diff options
author | 2023-05-15 14:35:44 +0200 | |
---|---|---|
committer | 2023-05-15 14:35:44 +0200 | |
commit | 0b20f1c16a42aef7f165dc47929bbee4bb201d54 (patch) | |
tree | 63667d3ffa067037a7035e8cde3f56cc18acc06d /src | |
parent | 8c92d56318b21fb0cddd8364e903b9e693fd53ed (diff) | |
download | it-tools-0b20f1c16a42aef7f165dc47929bbee4bb201d54.tar.gz it-tools-0b20f1c16a42aef7f165dc47929bbee4bb201d54.tar.zst it-tools-0b20f1c16a42aef7f165dc47929bbee4bb201d54.zip |
feat(base64-string-converter): switch to encode and decode url safe base64 strings (#392)
* feat(base64-string-converter): switch to encode and decode url safe
* feat(base64-string-converter): changes based on review comments, use config object instead of boolean argument.
* feat(base64-string-converter): fix validation, add option to watch additional refs for changes which interfere with validation rules
Diffstat (limited to 'src')
-rw-r--r-- | src/tools/base64-string-converter/base64-string-converter.vue | 23 | ||||
-rw-r--r-- | src/ui/c-input-text/c-input-text.vue | 4 | ||||
-rw-r--r-- | src/utils/base64.test.ts | 18 | ||||
-rw-r--r-- | src/utils/base64.ts | 39 |
4 files changed, 72 insertions, 12 deletions
diff --git a/src/tools/base64-string-converter/base64-string-converter.vue b/src/tools/base64-string-converter/base64-string-converter.vue index 0bcd966..9d574c9 100644 --- a/src/tools/base64-string-converter/base64-string-converter.vue +++ b/src/tools/base64-string-converter/base64-string-converter.vue @@ -1,5 +1,8 @@ <template> <c-card title="String to base64"> + <n-form-item label="Encode URL safe" label-placement="left"> + <n-switch v-model:value="encodeUrlSafe" /> + </n-form-item> <c-input-text v-model:value="textInput" multiline @@ -26,12 +29,16 @@ </c-card> <c-card title="Base64 to string"> + <n-form-item label="Decode URL safe" label-placement="left"> + <n-switch v-model:value="decodeUrlSafe" /> + </n-form-item> <c-input-text v-model:value="base64Input" multiline placeholder="Your base64 string..." rows="5" :validation-rules="b64ValidationRules" + :validation-watch="b64ValidationWatch" label="Base64 string to decode" mb-5 /> @@ -58,15 +65,23 @@ import { base64ToText, isValidBase64, textToBase64 } from '@/utils/base64'; import { withDefaultOnError } from '@/utils/defaults'; import { computed, ref } from 'vue'; +const encodeUrlSafe = useStorage('base64-string-converter--encode-url-safe', false); +const decodeUrlSafe = useStorage('base64-string-converter--decode-url-safe', false); + const textInput = ref(''); -const base64Output = computed(() => textToBase64(textInput.value)); +const base64Output = computed(() => textToBase64(textInput.value, { makeUrlSafe: encodeUrlSafe.value })); const { copy: copyTextBase64 } = useCopy({ source: base64Output, text: 'Base64 string copied to the clipboard' }); const base64Input = ref(''); -const textOutput = computed(() => withDefaultOnError(() => base64ToText(base64Input.value.trim()), '')); +const textOutput = computed(() => + withDefaultOnError(() => base64ToText(base64Input.value.trim(), { makeUrlSafe: decodeUrlSafe.value }), ''), +); const { copy: copyText } = useCopy({ source: textOutput, text: 'String copied to the clipboard' }); - const b64ValidationRules = [ - { message: 'Invalid base64 string', validator: (value: string) => isValidBase64(value.trim()) }, + { + message: 'Invalid base64 string', + validator: (value: string) => isValidBase64(value.trim(), { makeUrlSafe: decodeUrlSafe.value }), + }, ]; +const b64ValidationWatch = [decodeUrlSafe]; </script> diff --git a/src/ui/c-input-text/c-input-text.vue b/src/ui/c-input-text/c-input-text.vue index d1dd3c6..cd5f067 100644 --- a/src/ui/c-input-text/c-input-text.vue +++ b/src/ui/c-input-text/c-input-text.vue @@ -61,6 +61,7 @@ <script lang="ts" setup> import { generateRandomId } from '@/utils/random'; import { useValidation, type UseValidationRule } from '@/composable/validation'; +import type { Ref } from 'vue'; import { useTheme } from './c-input-text.theme'; import { useAppTheme } from '../theme/themes'; @@ -73,6 +74,7 @@ const props = withDefaults( readonly?: boolean; disabled?: boolean; validationRules?: UseValidationRule<string>[]; + validationWatch?: Ref<unknown>[]; validation?: ReturnType<typeof useValidation>; labelPosition?: 'top' | 'left'; labelWidth?: string; @@ -97,6 +99,7 @@ const props = withDefaults( readonly: false, disabled: false, validationRules: () => [], + validationWatch: undefined, validation: undefined, labelPosition: 'top', labelWidth: 'auto', @@ -125,6 +128,7 @@ const validation = useValidation({ rules: validationRules, source: value, + watch: props.validationWatch, }); const theme = useTheme(); diff --git a/src/utils/base64.test.ts b/src/utils/base64.test.ts index 0496b79..994f1b1 100644 --- a/src/utils/base64.test.ts +++ b/src/utils/base64.test.ts @@ -8,18 +8,34 @@ describe('base64 utils', () => { expect(textToBase64('a')).to.eql('YQ=='); expect(textToBase64('lorem ipsum')).to.eql('bG9yZW0gaXBzdW0='); expect(textToBase64('-1')).to.eql('LTE='); + expect(textToBase64('<<<????????>>>', { makeUrlSafe: false })).to.eql('PDw8Pz8/Pz8/Pz8+Pj4='); + }); + it('should convert string into url safe base64', () => { + expect(textToBase64('', { makeUrlSafe: true })).to.eql(''); + expect(textToBase64('a', { makeUrlSafe: true })).to.eql('YQ'); + expect(textToBase64('lorem ipsum', { makeUrlSafe: true })).to.eql('bG9yZW0gaXBzdW0'); + expect(textToBase64('<<<????????>>>', { makeUrlSafe: true })).to.eql('PDw8Pz8_Pz8_Pz8-Pj4'); }); }); describe('base64ToText', () => { it('should convert base64 into text', () => { expect(base64ToText('')).to.eql(''); - expect(base64ToText('YQ==')).to.eql('a'); + expect(base64ToText('YQ==', { makeUrlSafe: false })).to.eql('a'); expect(base64ToText('bG9yZW0gaXBzdW0=')).to.eql('lorem ipsum'); expect(base64ToText('data:text/plain;base64,bG9yZW0gaXBzdW0=')).to.eql('lorem ipsum'); expect(base64ToText('LTE=')).to.eql('-1'); }); + it('should convert url safe base64 into text', () => { + expect(base64ToText('', { makeUrlSafe: true })).to.eql(''); + expect(base64ToText('YQ', { makeUrlSafe: true })).to.eql('a'); + expect(base64ToText('bG9yZW0gaXBzdW0', { makeUrlSafe: true })).to.eql('lorem ipsum'); + expect(base64ToText('data:text/plain;base64,bG9yZW0gaXBzdW0', { makeUrlSafe: true })).to.eql('lorem ipsum'); + expect(base64ToText('LTE', { makeUrlSafe: true })).to.eql('-1'); + expect(base64ToText('PDw8Pz8_Pz8_Pz8-Pj4', { makeUrlSafe: true })).to.eql('<<<????????>>>'); + }); + it('should throw for incorrect base64 string', () => { expect(() => base64ToText('a')).to.throw('Incorrect base64 string'); expect(() => base64ToText(' ')).to.throw('Incorrect base64 string'); diff --git a/src/utils/base64.ts b/src/utils/base64.ts index c0ef96a..44fda1e 100644 --- a/src/utils/base64.ts +++ b/src/utils/base64.ts @@ -1,15 +1,19 @@ export { textToBase64, base64ToText, isValidBase64, removePotentialDataAndMimePrefix }; -function textToBase64(str: string) { - return window.btoa(str); +function textToBase64(str: string, { makeUrlSafe = false }: { makeUrlSafe?: boolean } = {}) { + const encoded = window.btoa(str); + return makeUrlSafe ? makeUriSafe(encoded) : encoded; } -function base64ToText(str: string) { - if (!isValidBase64(str)) { +function base64ToText(str: string, { makeUrlSafe = false }: { makeUrlSafe?: boolean } = {}) { + if (!isValidBase64(str, { makeUrlSafe: makeUrlSafe })) { throw new Error('Incorrect base64 string'); } - const cleanStr = removePotentialDataAndMimePrefix(str); + let cleanStr = removePotentialDataAndMimePrefix(str); + if (makeUrlSafe) { + cleanStr = unURI(cleanStr); + } try { return window.atob(cleanStr); @@ -22,12 +26,33 @@ function removePotentialDataAndMimePrefix(str: string) { return str.replace(/^data:.*?;base64,/, ''); } -function isValidBase64(str: string) { - const cleanStr = removePotentialDataAndMimePrefix(str); +function isValidBase64(str: string, { makeUrlSafe = false }: { makeUrlSafe?: boolean } = {}) { + let cleanStr = removePotentialDataAndMimePrefix(str); + if (makeUrlSafe) { + cleanStr = unURI(cleanStr); + } try { + if (makeUrlSafe) { + return removePotentialPadding(window.btoa(window.atob(cleanStr))) === cleanStr; + } return window.btoa(window.atob(cleanStr)) === cleanStr; } catch (err) { return false; } } + +function makeUriSafe(encoded: string) { + return encoded.replace(/=/g, '').replace(/\+/g, '-').replace(/\//g, '_'); +} + +function unURI(encoded: string): string { + return encoded + .replace(/-/g, '+') + .replace(/_/g, '/') + .replace(/[^A-Za-z0-9+/]/g, ''); +} + +function removePotentialPadding(str: string) { + return str.replace(/=/g, ''); +} |