mirror of
https://github.com/quantum5/totp.git
synced 2025-04-24 05:31:59 -04:00
Implement basic app logic
This commit is contained in:
parent
284a8d3559
commit
d8636039a9
1407
package-lock.json
generated
1407
package-lock.json
generated
File diff suppressed because it is too large
Load diff
|
@ -10,8 +10,11 @@
|
||||||
"preview": "vite preview"
|
"preview": "vite preview"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@otplib/core": "^12.0.1",
|
||||||
|
"@otplib/plugin-crypto-js": "^12.0.1",
|
||||||
"@popperjs/core": "^2.11.8",
|
"@popperjs/core": "^2.11.8",
|
||||||
"bootstrap": "^5.3.3",
|
"bootstrap": "^5.3.3",
|
||||||
|
"classnames": "^2.5.1",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0"
|
"react-dom": "^18.2.0"
|
||||||
},
|
},
|
||||||
|
@ -27,6 +30,7 @@
|
||||||
"eslint-plugin-react-refresh": "^0.4.6",
|
"eslint-plugin-react-refresh": "^0.4.6",
|
||||||
"sass": "^1.74.1",
|
"sass": "^1.74.1",
|
||||||
"typescript": "^5.2.2",
|
"typescript": "^5.2.2",
|
||||||
"vite": "^5.2.0"
|
"vite": "^5.2.0",
|
||||||
|
"vite-plugin-node-polyfills": "^0.21.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
53
src/App.tsx
53
src/App.tsx
|
@ -1,17 +1,56 @@
|
||||||
import {useState} from 'react';
|
import React from 'react';
|
||||||
import Input from './Input.tsx';
|
import {NumberInput, TextInput} from './Input';
|
||||||
|
import OTPOutput, {HashAlgorithm} from './OTPOutput';
|
||||||
|
import Select from './Select';
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [secret, setSecret] = useState('');
|
const [secret, setSecret] = React.useState('');
|
||||||
|
const [step, setStep] = React.useState(30);
|
||||||
|
const [offset, setOffset] = React.useState(0);
|
||||||
|
const [digits, setDigits] = React.useState(6);
|
||||||
|
const [algorithm, setAlgorithm] = React.useState<HashAlgorithm>('sha1');
|
||||||
|
|
||||||
|
const validStep = step > 0;
|
||||||
|
const validDigits = digits > 0 && digits < 10;
|
||||||
|
const valid = validStep && validDigits;
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!validStep) return;
|
||||||
|
const now = Date.now();
|
||||||
|
setOffset(Math.floor(now / (1000 * step)));
|
||||||
|
}, [validStep, step]);
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!validStep) return;
|
||||||
|
const now = Date.now();
|
||||||
|
const nextOffset = Math.floor(now / (1000 * step)) + 1;
|
||||||
|
const nextUpdate = nextOffset * step * 1000;
|
||||||
|
const timer = setTimeout(() => setOffset(nextOffset), nextUpdate - now);
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}, [validStep, offset, step]);
|
||||||
|
|
||||||
|
const onReset = React.useCallback(() => {
|
||||||
|
setStep(30);
|
||||||
|
setDigits(6);
|
||||||
|
setAlgorithm('sha1');
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="totp-app">
|
<div className="totp-app">
|
||||||
<div className="totp-settings">
|
<div className="totp-settings">
|
||||||
<Input label="Secret key" value={secret} onChange={setSecret} />
|
<TextInput label="Secret key" value={secret} onChange={setSecret}/>
|
||||||
</div>
|
<NumberInput label="Time step" value={step} onChange={setStep} min={1}
|
||||||
<div className="totp-output">
|
error={!validStep && 'You must enter an integer time step ≥ 1 second'}/>
|
||||||
|
<NumberInput label="Code digits" value={digits} onChange={setDigits} min={1} max={10}
|
||||||
|
error={!validDigits && 'You must enter an integer digit count between 1 and 10'}/>
|
||||||
|
<Select label="Algorithm" value={algorithm} onChange={setAlgorithm} options={{
|
||||||
|
sha1: 'SHA-1',
|
||||||
|
sha256: 'SHA-256',
|
||||||
|
sha512: 'SHA-512',
|
||||||
|
}}/>
|
||||||
|
<button type="button" className="btn btn-secondary" onClick={onReset}>Reset</button>
|
||||||
</div>
|
</div>
|
||||||
|
{valid && <OTPOutput secret={secret} offset={offset} algorithm={algorithm} digits={digits}/>}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,19 +1,43 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
|
||||||
export default function Input({type, label, value, onChange}: {
|
type CommonProps = {
|
||||||
type?: string;
|
|
||||||
label: React.ReactNode;
|
label: React.ReactNode;
|
||||||
|
error?: React.ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
type TextInputProps = {
|
||||||
value: string;
|
value: string;
|
||||||
onChange: (value: string) => void;
|
onChange: (value: string) => void;
|
||||||
}) {
|
}
|
||||||
|
|
||||||
|
type HTMLInputProps = Omit<React.HTMLProps<HTMLInputElement>, 'onChange' | 'label'>;
|
||||||
|
|
||||||
|
function BaseInput({label, value, onChange, error, ...props}: CommonProps & HTMLInputProps & TextInputProps) {
|
||||||
const id = React.useId();
|
const id = React.useId();
|
||||||
const handleChange = React.useCallback(
|
const handleChange = React.useCallback(
|
||||||
(e: React.ChangeEvent<HTMLInputElement>) => onChange(e.target.value),
|
(e: React.ChangeEvent<HTMLInputElement>) => onChange(e.target.value),
|
||||||
[onChange],
|
[onChange],
|
||||||
);
|
);
|
||||||
|
|
||||||
return <div className="totp-input">
|
return <div className={classNames('totp-input', {'has-validation': !!error})}>
|
||||||
<label htmlFor={id} className="form-label">{label}</label>
|
<label htmlFor={id} className="form-label">{label}</label>
|
||||||
<input id={id} className="form-control" type={type || 'text'} value={value} onChange={handleChange}/>
|
<input id={id} className={classNames('form-control', {'is-invalid': !!error})}
|
||||||
|
value={value} onChange={handleChange} {...props}/>
|
||||||
|
{error && <div className="invalid-feedback">{error}</div>}
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function TextInput(props: CommonProps & TextInputProps) {
|
||||||
|
return <BaseInput {...props} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function NumberInput({value, onChange, ...props}: CommonProps & {
|
||||||
|
value: number;
|
||||||
|
onChange: (value: number) => void;
|
||||||
|
min?: number;
|
||||||
|
max?: number;
|
||||||
|
}) {
|
||||||
|
const handleChange = React.useCallback((value: string) => onChange(+value), [onChange]);
|
||||||
|
return <BaseInput type="number" value={`${value}`} onChange={handleChange} {...props}/>;
|
||||||
|
}
|
||||||
|
|
37
src/OTPOutput.tsx
Normal file
37
src/OTPOutput.tsx
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
import React from 'react';
|
||||||
|
import {HashAlgorithms, HOTP, HOTPOptions} from '@otplib/core';
|
||||||
|
import {createDigest} from '@otplib/plugin-crypto-js';
|
||||||
|
|
||||||
|
const ALGORITHMS = {
|
||||||
|
sha1: HashAlgorithms.SHA1,
|
||||||
|
sha256: HashAlgorithms.SHA256,
|
||||||
|
sha512: HashAlgorithms.SHA512,
|
||||||
|
};
|
||||||
|
|
||||||
|
export type HashAlgorithm = keyof typeof ALGORITHMS;
|
||||||
|
|
||||||
|
function OTPCode({code}: { code: string }) {
|
||||||
|
return <div className="totp-code">
|
||||||
|
{code}
|
||||||
|
</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function OTPOutput({secret, offset, algorithm, digits}: {
|
||||||
|
secret: string;
|
||||||
|
offset: number;
|
||||||
|
algorithm: HashAlgorithm;
|
||||||
|
digits: number;
|
||||||
|
}) {
|
||||||
|
const hotp = React.useMemo(() => new HOTP<HOTPOptions>({
|
||||||
|
createDigest,
|
||||||
|
digits,
|
||||||
|
algorithm: ALGORITHMS[algorithm],
|
||||||
|
}), [digits, algorithm]);
|
||||||
|
|
||||||
|
return <div className="totp-output">
|
||||||
|
{[...Array(21).keys()].map((i) => {
|
||||||
|
const current = offset + i - 10;
|
||||||
|
return <OTPCode key={current} code={hotp.generate(secret, current)}/>;
|
||||||
|
})}
|
||||||
|
</div>;
|
||||||
|
}
|
22
src/Select.tsx
Normal file
22
src/Select.tsx
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export default function Select<T extends string>({label, options, value, onChange}: {
|
||||||
|
label: React.ReactNode;
|
||||||
|
options: Record<T, string>;
|
||||||
|
value: T;
|
||||||
|
onChange: (value: T) => void;
|
||||||
|
}) {
|
||||||
|
const id = React.useId();
|
||||||
|
const handleChange = React.useCallback(
|
||||||
|
(e: React.ChangeEvent<HTMLSelectElement>) => onChange(e.target.value as T),
|
||||||
|
[onChange],
|
||||||
|
);
|
||||||
|
|
||||||
|
return <div className="totp-select">
|
||||||
|
<label htmlFor={id} className="form-label">{label}</label>
|
||||||
|
<select id={id} className="form-select" value={value} onChange={handleChange}>
|
||||||
|
{Object.entries(options).map(([key, value]) =>
|
||||||
|
<option key={key} value={key}>{value as string}</option>)}
|
||||||
|
</select>
|
||||||
|
</div>;
|
||||||
|
}
|
|
@ -4,9 +4,10 @@
|
||||||
@import 'bootstrap/scss/_mixins.scss';
|
@import 'bootstrap/scss/_mixins.scss';
|
||||||
@import 'bootstrap/scss/_root.scss';
|
@import 'bootstrap/scss/_root.scss';
|
||||||
@import 'bootstrap/scss/_grid.scss';
|
@import 'bootstrap/scss/_grid.scss';
|
||||||
|
@import 'bootstrap/scss/_buttons.scss';
|
||||||
@import 'bootstrap/scss/_forms.scss';
|
@import 'bootstrap/scss/_forms.scss';
|
||||||
|
|
||||||
.totp-input {
|
.totp-input, .totp-select {
|
||||||
label {
|
label {
|
||||||
font-weight: $font-weight-bold;
|
font-weight: $font-weight-bold;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,14 @@
|
||||||
import {defineConfig} from 'vite';
|
import {defineConfig} from 'vite';
|
||||||
import react from '@vitejs/plugin-react';
|
import react from '@vitejs/plugin-react';
|
||||||
|
import {nodePolyfills} from 'vite-plugin-node-polyfills';
|
||||||
|
|
||||||
// https://vitejs.dev/config/
|
// https://vitejs.dev/config/
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
plugins: [react()],
|
plugins: [
|
||||||
|
react(),
|
||||||
|
nodePolyfills({
|
||||||
|
include: ['buffer'],
|
||||||
|
globals: {Buffer: true},
|
||||||
|
}),
|
||||||
|
],
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue