Merge branch 'main' into feat/cert-key-parser

This commit is contained in:
sharevb 2024-05-05 19:41:06 +02:00 committed by GitHub
commit c10302b138
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
13 changed files with 133 additions and 56 deletions

View file

@ -15,7 +15,7 @@ jobs:
- run: corepack enable - run: corepack enable
- uses: actions/setup-node@v3 - uses: actions/setup-node@v3
with: with:
node-version: 16 node-version: 20
cache: 'pnpm' cache: 'pnpm'
- name: Install dependencies - name: Install dependencies

View file

@ -18,7 +18,7 @@ jobs:
- uses: actions/setup-node@v3 - uses: actions/setup-node@v3
with: with:
node-version: 16 node-version: 20
cache: 'pnpm' cache: 'pnpm'
- name: Get Playwright version - name: Get Playwright version

View file

@ -7,7 +7,7 @@ home:
toggleMenu: 'Toggle menu' toggleMenu: 'Toggle menu'
home: Home home: Home
uiLib: 'UI Lib' uiLib: 'UI Lib'
support: 'Support IT Tools development' support: 'Support IT-Tools development'
buyMeACoffee: 'Buy me a coffee' buyMeACoffee: 'Buy me a coffee'
follow: follow:
title: 'You like it-tools?' title: 'You like it-tools?'
@ -72,7 +72,7 @@ tools:
password-strength-analyser: password-strength-analyser:
title: Password strength analyser title: Password strength analyser
description: Discover the strength of your password with this client side only password strength analyser and crack time estimation tool. description: Discover the strength of your password with this client-side-only password strength analyser and crack time estimation tool.
chronometer: chronometer:
title: Chronometer title: Chronometer
@ -98,7 +98,7 @@ tools:
svg-placeholder-generator: svg-placeholder-generator:
title: SVG placeholder generator title: SVG placeholder generator
description: Generate svg images to use as placeholder in your applications. description: Generate svg images to use as a placeholder in your applications.
json-to-csv: json-to-csv:
title: JSON to CSV title: JSON to CSV
@ -126,11 +126,11 @@ tools:
crontab-generator: crontab-generator:
title: Crontab generator title: Crontab generator
description: Validate and generate crontab and get the human readable description of the cron schedule. description: Validate and generate crontab and get the human-readable description of the cron schedule.
http-status-codes: http-status-codes:
title: HTTP status codes title: HTTP status codes
description: The list of all HTTP status codes their name and their meaning. description: The list of all HTTP status codes, their name, and their meaning.
sql-prettify: sql-prettify:
title: SQL prettify and format title: SQL prettify and format
@ -142,7 +142,7 @@ tools:
git-memo: git-memo:
title: Git cheatsheet title: Git cheatsheet
description: Git is a decentralized version management software. With this cheatsheet you will have a quick access to the most common git commands. description: Git is a decentralized version management software. With this cheatsheet, you will have quick access to the most common git commands.
slugify-string: slugify-string:
title: Slugify string title: Slugify string
@ -150,7 +150,7 @@ tools:
encryption: encryption:
title: Encrypt / decrypt text title: Encrypt / decrypt text
description: Encrypt and decrypt text clear text using crypto algorithm like AES, TripleDES, Rabbit or RC4. description: Encrypt clear text and decrypt ciphertext using crypto algorithms like AES, TripleDES, Rabbit or RC4.
random-port-generator: random-port-generator:
title: Random port generator title: Random port generator
@ -158,11 +158,11 @@ tools:
yaml-prettify: yaml-prettify:
title: YAML prettify and format title: YAML prettify and format
description: Prettify your YAML string to a human friendly readable format. description: Prettify your YAML string into a friendly, human-readable format.
eta-calculator: eta-calculator:
title: ETA calculator title: ETA calculator
description: An ETA (Estimated Time of Arrival) calculator to know the approximate end time of a task, for example the moment of ending of a download. description: An ETA (Estimated Time of Arrival) calculator to determine the approximate end time of a task, for example, the end time and duration of a file download.
roman-numeral-converter: roman-numeral-converter:
title: Roman numeral converter title: Roman numeral converter
@ -174,11 +174,11 @@ tools:
bip39-generator: bip39-generator:
title: BIP39 passphrase generator title: BIP39 passphrase generator
description: Generate BIP39 passphrase from existing or random mnemonic, or get the mnemonic from the passphrase. description: Generate a BIP39 passphrase from an existing or random mnemonic, or get the mnemonic from the passphrase.
base64-file-converter: base64-file-converter:
title: Base64 file converter title: Base64 file converter
description: Convert string, files or images into a it\'s base64 representation. description: Convert a string, file, or image into its base64 representation.
list-converter: list-converter:
title: List converter title: List converter
@ -186,7 +186,7 @@ tools:
base64-string-converter: base64-string-converter:
title: Base64 string encoder/decoder title: Base64 string encoder/decoder
description: Simply encode and decode string into a their base64 representation. description: Simply encode and decode strings into their base64 representation.
toml-to-yaml: toml-to-yaml:
title: TOML to YAML title: TOML to YAML
@ -198,15 +198,15 @@ tools:
json-to-yaml-converter: json-to-yaml-converter:
title: JSON to YAML converter title: JSON to YAML converter
description: Simply convert JSON to YAML with this live online converter. description: Simply convert JSON to YAML with this online live converter.
url-parser: url-parser:
title: Url parser title: URL parser
description: Parse an url string to get all the different parts (protocol, origin, params, port, username-password, ...) description: Parse a URL into its separate constituent parts (protocol, origin, params, port, username-password, ...)
iban-validator-and-parser: iban-validator-and-parser:
title: IBAN validator and parser title: IBAN validator and parser
description: Validate and parse IBAN numbers. Check if IBAN is valid and get the country, BBAN, if it is a QR-IBAN and the IBAN friendly format. description: Validate and parse IBAN numbers. Check if an IBAN is valid and get the country, BBAN, if it is a QR-IBAN and the IBAN friendly format.
user-agent-parser: user-agent-parser:
title: User-agent parser title: User-agent parser
@ -218,27 +218,27 @@ tools:
case-converter: case-converter:
title: Case converter title: Case converter
description: Change the case of a string and chose between different formats description: Transform the case of a string and choose between different formats
html-entities: html-entities:
title: Escape html entities title: Escape HTML entities
description: Escape or unescape html entities (replace <,>, &, " and \' to their html version) description: Escape or unescape HTML entities (replace characters like <,>, &, " and \' with their HTML version)
json-prettify: json-prettify:
title: JSON prettify and format title: JSON prettify and format
description: Prettify your JSON string to a human friendly readable format. description: Prettify your JSON string into a friendly, human-readable format.
docker-run-to-docker-compose-converter: docker-run-to-docker-compose-converter:
title: Docker run to Docker compose converter title: Docker run to Docker compose converter
description: Turns docker run commands into docker-compose files! description: Transforms "docker run" commands into docker-compose files!
mac-address-lookup: mac-address-lookup:
title: MAC address lookup title: MAC address lookup
description: Find the vendor and manufacturer of a device by its MAC address. description: Find the vendor and manufacturer of a device by its MAC address.
mime-types: mime-types:
title: Mime types title: MIME types
description: Convert mime types to extensions and vice-versa. description: Convert MIME types to file extensions and vice-versa.
toml-to-json: toml-to-json:
title: TOML to JSON title: TOML to JSON
@ -250,19 +250,19 @@ tools:
qrcode-generator: qrcode-generator:
title: QR Code generator title: QR Code generator
description: Generate and download QR-code for an url or just a text and customize the background and foreground colors. description: Generate and download a QR code for a URL (or just plain text), and customize the background and foreground colors.
wifi-qrcode-generator: wifi-qrcode-generator:
title: WiFi QR Code generator title: WiFi QR Code generator
description: Generate and download QR-codes for quick connections to WiFi networks. description: Generate and download QR codes for quick connections to WiFi networks.
xml-formatter: xml-formatter:
title: XML formatter title: XML formatter
description: Prettify your XML string to a human friendly readable format. description: Prettify your XML string into a friendly, human-readable format.
temperature-converter: temperature-converter:
title: Temperature converter title: Temperature converter
description: Temperature degrees conversions for Kelvin, Celsius, Fahrenheit, Rankine, Delisle, Newton, Réaumur and Rømer. description: Degrees temperature conversions for Kelvin, Celsius, Fahrenheit, Rankine, Delisle, Newton, Réaumur, and Rømer.
chmod-calculator: chmod-calculator:
title: Chmod calculator title: Chmod calculator
@ -270,11 +270,11 @@ tools:
rsa-key-pair-generator: rsa-key-pair-generator:
title: RSA key pair generator title: RSA key pair generator
description: Generate new random RSA private and public key pem certificates. description: Generate a new random RSA private and public pem certificate key pair.
html-wysiwyg-editor: html-wysiwyg-editor:
title: HTML WYSIWYG editor title: HTML WYSIWYG editor
description: Online HTML editor with feature-rich WYSIWYG editor, get the source code of the content immediately. description: Online, feature-rich WYSIWYG HTML editor which generates the source code of the content immediately.
yaml-to-toml: yaml-to-toml:
title: YAML to TOML title: YAML to TOML
@ -302,15 +302,15 @@ tools:
ipv4-subnet-calculator: ipv4-subnet-calculator:
title: IPv4 subnet calculator title: IPv4 subnet calculator
description: Parse your IPv4 CIDR blocks and get all the info you need about your sub network. description: Parse your IPv4 CIDR blocks and get all the info you need about your subnet.
og-meta-generator: og-meta-generator:
title: Open graph meta generator title: Open graph meta generator
description: Generate open-graph and socials html meta tags for your website. description: Generate open-graph and socials HTML meta tags for your website.
ipv6-ula-generator: ipv6-ula-generator:
title: IPv6 ULA generator title: IPv6 ULA generator
description: Generate your own local, non-routable IP addresses on your network according to RFC4193. description: Generate your own local, non-routable IP addresses for your network according to RFC4193.
hash-text: hash-text:
title: Hash text title: Hash text
@ -330,7 +330,7 @@ tools:
json-minify: json-minify:
title: JSON minify title: JSON minify
description: Minify and compress your JSON by removing unnecessary white spaces. description: Minify and compress your JSON by removing unnecessary whitespace.
ulid-generator: ulid-generator:
title: ULID generator title: ULID generator
@ -342,31 +342,31 @@ tools:
base-converter: base-converter:
title: Integer base converter title: Integer base converter
description: Convert number between different bases (decimal, hexadecimal, binary, octal, base64, ...) description: Convert a number between different bases (decimal, hexadecimal, binary, octal, base64, ...)
yaml-to-json-converter: yaml-to-json-converter:
title: YAML to JSON converter title: YAML to JSON converter
description: Simply convert YAML to JSON with this live online converter. description: Simply convert YAML to JSON with this online live converter.
uuid-generator: uuid-generator:
title: UUIDs generator title: UUIDs generator
description: A Universally Unique Identifier (UUID) is a 128-bit number used to identify information in computer systems. The number of possible UUIDs is 16^32, which is 2^128 or about 3.4x10^38 (which is a lot!). description: A Universally Unique Identifier (UUID) is a 128-bit number used to identify information in computer systems. The number of possible UUIDs is 16^32, which is 2^128 or about 3.4x10^38 (which is a lot!).
ipv4-address-converter: ipv4-address-converter:
title: Ipv4 address converter title: IPv4 address converter
description: Convert an ip address into decimal, binary, hexadecimal or event in ipv6 description: Convert an IP address into decimal, binary, hexadecimal, or even an IPv6 representation of it.
text-statistics: text-statistics:
title: Text statistics title: Text statistics
description: Get information about a text, the amount of characters, the amount of words, it\'s size, ... description: Get information about a text, the number of characters, the number of words, its size in bytes, ...
text-to-nato-alphabet: text-to-nato-alphabet:
title: Text to NATO alphabet title: Text to NATO alphabet
description: Transform text into NATO phonetic alphabet for oral transmission. description: Transform text into the NATO phonetic alphabet for oral transmission.
basic-auth-generator: basic-auth-generator:
title: Basic auth generator title: Basic auth generator
description: Generate a base64 basic auth header from an username and a password. description: Generate a base64 basic auth header from a username and password.
text-to-unicode: text-to-unicode:
title: Text to Unicode title: Text to Unicode
@ -374,7 +374,7 @@ tools:
ipv4-range-expander: ipv4-range-expander:
title: IPv4 range expander title: IPv4 range expander
description: Given a start and an end IPv4 address this tool calculates a valid IPv4 network with its CIDR notation. description: Given a start and an end IPv4 address, this tool calculates a valid IPv4 subnet along with its CIDR notation.
text-diff: text-diff:
title: Text diff title: Text diff
@ -385,9 +385,9 @@ tools:
description: Generate and validate time-based OTP (one time password) for multi-factor authentication. description: Generate and validate time-based OTP (one time password) for multi-factor authentication.
url-encoder: url-encoder:
title: Encode/decode url formatted strings title: Encode/decode URL-formatted strings
description: Encode to url-encoded format (also known as "percent-encoded") or decode from it. description: Encode text to URL-encoded format (also known as "percent-encoded"), or decode from it.
text-to-binary: text-to-binary:
title: Text to ASCII binary title: Text to ASCII binary
description: Convert text to its ASCII binary representation and vice versa. description: Convert text to its ASCII binary representation and vice-versa.

View file

@ -9,7 +9,7 @@ const useWebServer = process.env.NO_WEB_SERVER !== 'true';
*/ */
export default defineConfig({ export default defineConfig({
testDir: './src', testDir: './src',
testMatch: /.*\.e2e\.(spec\.)?ts/, testMatch: /\.e2e\.(spec\.)?ts$/,
/* Run tests in files in parallel */ /* Run tests in files in parallel */
fullyParallel: true, fullyParallel: true,
/* Fail the build on CI if you accidentally left test.only in the source code. */ /* Fail the build on CI if you accidentally left test.only in the source code. */
@ -57,7 +57,7 @@ export default defineConfig({
&& { && {
webServer: { webServer: {
command: 'npm run preview', command: 'npm run preview',
url: 'http://127.0.0.1:5050', url: 'http://localhost:5050',
reuseExistingServer: !isCI, reuseExistingServer: !isCI,
}, },
} }

View file

@ -28,7 +28,7 @@ const compareMatch = computed(() => compareSync(compareString.value, compareHash
mb-2 mb-2
/> />
<n-form-item label="Salt count: " label-placement="left" label-width="120"> <n-form-item label="Salt count: " label-placement="left" label-width="120">
<n-input-number v-model:value="saltCount" placeholder="Salt rounds..." :max="10" :min="0" w-full /> <n-input-number v-model:value="saltCount" placeholder="Salt rounds..." :max="100" :min="0" w-full />
</n-form-item> </n-form-item>
<c-input-text :value="hashed" readonly text-center /> <c-input-text :value="hashed" readonly text-center />

View file

@ -6,6 +6,7 @@ import { tool as asciiTextDrawer } from './ascii-text-drawer';
import { tool as textToUnicode } from './text-to-unicode'; import { tool as textToUnicode } from './text-to-unicode';
import { tool as certificateKeyParser } from './certificate-key-parser'; import { tool as certificateKeyParser } from './certificate-key-parser';
import { tool as safelinkDecoder } from './safelink-decoder';
import { tool as pdfSignatureChecker } from './pdf-signature-checker'; import { tool as pdfSignatureChecker } from './pdf-signature-checker';
import { tool as numeronymGenerator } from './numeronym-generator'; import { tool as numeronymGenerator } from './numeronym-generator';
import { tool as macAddressGenerator } from './mac-address-generator'; import { tool as macAddressGenerator } from './mac-address-generator';
@ -140,6 +141,7 @@ export const toolsByCategory: ToolCategory[] = [
userAgentParser, userAgentParser,
httpStatusCodes, httpStatusCodes,
jsonDiff, jsonDiff,
safelinkDecoder,
], ],
}, },
{ {

View file

@ -11,6 +11,9 @@ describe('integer-base-converter', () => {
expect(convertBase({ value: '10100101', fromBase: 2, toBase: 16 })).toEqual('a5'); expect(convertBase({ value: '10100101', fromBase: 2, toBase: 16 })).toEqual('a5');
expect(convertBase({ value: '192654', fromBase: 10, toBase: 8 })).toEqual('570216'); expect(convertBase({ value: '192654', fromBase: 10, toBase: 8 })).toEqual('570216');
expect(convertBase({ value: 'zz', fromBase: 64, toBase: 10 })).toEqual('2275'); expect(convertBase({ value: 'zz', fromBase: 64, toBase: 10 })).toEqual('2275');
expect(convertBase({ value: '42540766411283223938465490632011909384', fromBase: 10, toBase: 10 })).toEqual('42540766411283223938465490632011909384');
expect(convertBase({ value: '42540766411283223938465490632011909384', fromBase: 10, toBase: 16 })).toEqual('20010db8000085a300000000ac1f8908');
expect(convertBase({ value: '20010db8000085a300000000ac1f8908', fromBase: 16, toBase: 10 })).toEqual('42540766411283223938465490632011909384');
}); });
}); });
}); });

View file

@ -5,16 +5,16 @@ export function convertBase({ value, fromBase, toBase }: { value: string; fromBa
let decValue = value let decValue = value
.split('') .split('')
.reverse() .reverse()
.reduce((carry: number, digit: string, index: number) => { .reduce((carry: bigint, digit: string, index: number) => {
if (!fromRange.includes(digit)) { if (!fromRange.includes(digit)) {
throw new Error(`Invalid digit "${digit}" for base ${fromBase}.`); throw new Error(`Invalid digit "${digit}" for base ${fromBase}.`);
} }
return (carry += fromRange.indexOf(digit) * fromBase ** index); return (carry += BigInt(fromRange.indexOf(digit)) * BigInt(fromBase) ** BigInt(index));
}, 0); }, 0n);
let newValue = ''; let newValue = '';
while (decValue > 0) { while (decValue > 0) {
newValue = toRange[decValue % toBase] + newValue; newValue = toRange[Number(decValue % BigInt(toBase))] + newValue;
decValue = (decValue - (decValue % toBase)) / toBase; decValue = (decValue - (decValue % BigInt(toBase))) / BigInt(toBase);
} }
return newValue || '0'; return newValue || '0';
} }

View file

@ -0,0 +1,12 @@
import { Mailbox } from '@vicons/tabler';
import { defineTool } from '../tool';
export const tool = defineTool({
name: 'Outlook Safelink decoder',
path: '/safelink-decoder',
description: 'Decode Outlook SafeLink links',
keywords: ['outlook', 'safelink', 'decoder'],
component: () => import('./safelink-decoder.vue'),
icon: Mailbox,
createdAt: new Date('2024-03-11'),
});

View file

@ -0,0 +1,21 @@
import { describe, expect, it } from 'vitest';
import { decodeSafeLinksURL } from './safelink-decoder.service';
describe('safelink-decoder', () => {
describe('decodeSafeLinksURL', () => {
describe('decode outlook safelink urls', () => {
it('should decode basic safelink urls', () => {
expect(decodeSafeLinksURL('https://aus01.safelinks.protection.outlook.com/?url=https%3A%2F%2Fwww.google.com%2Fsearch%3Fq%3Dsafelink%26rlz%3D1&data=05%7C02%7C%7C1ed07253975b46da1d1508dc3443752a%7C84df9e7fe9f640afb435aaaaaaaaaaaa%7C1%7C0%7C638442711583216725%7CUnknown%7CTWFpbGZsb3d8eyJWIjoiMC4wLjAwMDAiLCJQIjoiV2luMzIiLCJBTiI6Ik1haWwiLCJXVCI6Mn0%3D%7C0%7C%7C%7C&sdata=%2BQY0HBnnxfI7pzZoxzlhZdDvYu80LwQB0zUUjrffVnk%3D&reserved=0'))
.toBe('https://www.google.com/search?q=safelink&rlz=1');
});
it('should decode encoded safelink urls', () => {
expect(decodeSafeLinksURL('https://aus01.safelinks.protection.outlook.com/?url=https%3A%2F%2Fwww.google.com%2Fsearch%3Fq%3Dsafelink%26rlz%3D1&amp;data=05%7C02%7C%7C1ed07253975b46da1d1508dc3443752a%7C84df9e7fe9f640afb435aaaaaaaaaaaa%7C1%7C0%7C638442711583216725%7CUnknown%7CTWFpbGZsb3d8eyJWIjoiMC4wLjAwMDAiLCJQIjoiV2luMzIiLCJBTiI6Ik1haWwiLCJXVCI6Mn0%3D%7C0%7C%7C%7C&amp;sdata=%2BQY0HBnnxfI7pzZoxzlhZdDvYu80LwQB0zUUjrffVnk%3D&amp;reserved=0'))
.toBe('https://www.google.com/search?q=safelink&rlz=1');
});
it('throw on not outlook safelink urls', () => {
expect(() => decodeSafeLinksURL('https://google.com'))
.toThrow('Invalid SafeLinks URL provided');
});
});
});
});

View file

@ -0,0 +1,7 @@
export function decodeSafeLinksURL(safeLinksUrl: string) {
if (!safeLinksUrl.match(/\.safelinks\.protection\.outlook\.com/)) {
throw new Error('Invalid SafeLinks URL provided');
}
return new URL(safeLinksUrl).searchParams.get('url');
}

View file

@ -0,0 +1,32 @@
<script setup lang="ts">
import { decodeSafeLinksURL } from './safelink-decoder.service';
import TextareaCopyable from '@/components/TextareaCopyable.vue';
const inputSafeLinkUrl = ref('');
const outputDecodedUrl = computed(() => {
try {
return decodeSafeLinksURL(inputSafeLinkUrl.value);
}
catch (e: any) {
return e.toString();
}
});
</script>
<template>
<div>
<c-input-text
v-model:value="inputSafeLinkUrl"
raw-text
placeholder="Your input Outlook SafeLink Url..."
autofocus
label="Your input Outlook SafeLink Url:"
/>
<n-divider />
<n-form-item label="Output decoded URL:">
<TextareaCopyable :value="outputDecodedUrl" :word-wrap="true" />
</n-form-item>
</div>
</template>

View file

@ -23,7 +23,7 @@ const decodeInput = ref('Hello%20world%20%3A)');
const decodeOutput = computed(() => withDefaultOnError(() => decodeURIComponent(decodeInput.value), '')); const decodeOutput = computed(() => withDefaultOnError(() => decodeURIComponent(decodeInput.value), ''));
const decodeValidation = useValidation({ const decodeValidation = useValidation({
source: encodeInput, source: decodeInput,
rules: [ rules: [
{ {
validator: value => isNotThrowing(() => decodeURIComponent(value)), validator: value => isNotThrowing(() => decodeURIComponent(value)),