feat: Reinitialize frontend with SvelteKit and TypeScript

- Delete old Vite+Svelte frontend
- Initialize new SvelteKit project with TypeScript
- Configure Tailwind CSS v4 + DaisyUI
- Implement JWT authentication with auto-refresh
- Create login page with form validation (Zod)
- Add protected route guards
- Update Docker configuration for single-stage build
- Add E2E tests with Playwright (6/11 passing)
- Fix Svelte 5 reactivity with $state() runes

Known issues:
- 5 E2E tests failing (timing/async issues)
- Token refresh implementation needs debugging
- Validation error display timing
This commit is contained in:
2026-02-17 16:19:59 -05:00
parent 54df6018f5
commit de2d83092e
28274 changed files with 3816354 additions and 90 deletions

View File

@@ -0,0 +1,14 @@
import ICookie from '../ICookie.js';
/**
* Cookie expire utility.
*/
export default class CookieExpireUtility {
/**
* Returns "true" if cookie has expired.
*
* @param cookie Cookie.
* @returns "true" if cookie has expired.
*/
static hasExpired(cookie: ICookie): boolean | null;
}
//# sourceMappingURL=CookieExpireUtility.d.ts.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieExpireUtility.d.ts","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieExpireUtility.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,eAAe,CAAC;AAEpC;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,mBAAmB;IACvC;;;;;OAKG;WACW,UAAU,CAAC,MAAM,EAAE,OAAO,GAAG,OAAO,GAAG,IAAI;CAGzD"}

View File

@@ -0,0 +1,15 @@
/**
* Cookie expire utility.
*/
export default class CookieExpireUtility {
/**
* Returns "true" if cookie has expired.
*
* @param cookie Cookie.
* @returns "true" if cookie has expired.
*/
static hasExpired(cookie) {
return cookie.expires && cookie.expires.getTime() < Date.now();
}
}
//# sourceMappingURL=CookieExpireUtility.js.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieExpireUtility.js","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieExpireUtility.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,mBAAmB;IACvC;;;;;OAKG;IACI,MAAM,CAAC,UAAU,CAAC,MAAe;QACvC,OAAO,MAAM,CAAC,OAAO,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAChE,CAAC;CACD"}

View File

@@ -0,0 +1,23 @@
import URL from '../../url/URL.js';
import ICookie from '../ICookie.js';
/**
* Cookie string.
*/
export default class CookieStringUtility {
/**
* Returns cookie.
*
* @param originURL Origin URL.
* @param cookieString Cookie string.
* @returns Cookie.
*/
static stringToCookie(originURL: URL, cookieString: string): ICookie | null;
/**
* Returns cookie string with key and value.
*
* @param cookies Cookies.
* @returns Cookie string.
*/
static cookiesToString(cookies: ICookie[]): string;
}
//# sourceMappingURL=CookieStringUtility.d.ts.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieStringUtility.d.ts","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieStringUtility.ts"],"names":[],"mappings":"AACA,OAAO,GAAG,MAAM,kBAAkB,CAAC;AACnC,OAAO,OAAO,MAAM,eAAe,CAAC;AAGpC;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,mBAAmB;IACvC;;;;;;OAMG;WACW,cAAc,CAAC,SAAS,EAAE,GAAG,EAAE,YAAY,EAAE,MAAM,GAAG,OAAO,GAAG,IAAI;IA6ElF;;;;;OAKG;WACW,eAAe,CAAC,OAAO,EAAE,OAAO,EAAE,GAAG,MAAM;CAazD"}

View File

@@ -0,0 +1,98 @@
import CookieSameSiteEnum from '../enums/CookieSameSiteEnum.js';
import DefaultCookie from '../DefaultCookie.js';
/**
* Cookie string.
*/
export default class CookieStringUtility {
/**
* Returns cookie.
*
* @param originURL Origin URL.
* @param cookieString Cookie string.
* @returns Cookie.
*/
static stringToCookie(originURL, cookieString) {
const parts = cookieString.split(';');
const part = parts.shift();
const index = part.indexOf('=');
const key = index !== -1 ? part.slice(0, index).trim() : part.trim();
const value = index !== -1 ? part.slice(index + 1).trim() : null;
const cookie = Object.assign({}, DefaultCookie, {
// Required
key,
value,
originURL
});
// Invalid if key is empty.
if (!cookie.key) {
return null;
}
for (const part of parts) {
const index = part.indexOf('=');
const key = index !== -1 ? part.slice(0, index).trim().toLowerCase() : part.trim().toLowerCase();
const value = index !== -1 ? part.slice(index + 1).trim() : '';
switch (key) {
case 'expires':
cookie.expires = new Date(value);
break;
case 'max-age':
cookie.expires = new Date(parseInt(value, 10) * 1000 + Date.now());
break;
case 'domain':
cookie.domain = value;
break;
case 'path':
cookie.path = value[0] === '/' ? value : `/${value}`;
break;
case 'httponly':
cookie.httpOnly = true;
break;
case 'secure':
cookie.secure = true;
break;
case 'samesite':
switch (value.toLowerCase()) {
case 'strict':
cookie.sameSite = CookieSameSiteEnum.strict;
break;
case 'lax':
cookie.sameSite = CookieSameSiteEnum.lax;
break;
case 'none':
cookie.sameSite = CookieSameSiteEnum.none;
}
break;
}
}
const lowerKey = cookie.key.toLowerCase();
// Invalid if __secure- prefix is used and cookie is not secure.
if (lowerKey.startsWith('__secure-') && !cookie.secure) {
return null;
}
// Invalid if __host- prefix is used and cookie is not secure, not on root path or has a domain.
if (lowerKey.startsWith('__host-') &&
(!cookie.secure || cookie.path !== '/' || cookie.domain)) {
return null;
}
return cookie;
}
/**
* Returns cookie string with key and value.
*
* @param cookies Cookies.
* @returns Cookie string.
*/
static cookiesToString(cookies) {
const cookieString = [];
for (const cookie of cookies) {
if (cookie.value !== null) {
cookieString.push(`${cookie.key}=${cookie.value}`);
}
else {
cookieString.push(cookie.key);
}
}
return cookieString.join('; ');
}
}
//# sourceMappingURL=CookieStringUtility.js.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieStringUtility.js","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieStringUtility.ts"],"names":[],"mappings":"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"}

View File

@@ -0,0 +1,16 @@
import URL from '../../url/URL.js';
import ICookie from '../ICookie.js';
/**
* Cookie string.
*/
export default class CookieURLUtility {
/**
* Returns "true" if cookie matches URL.
*
* @param cookie Cookie.
* @param url URL.
* @returns "true" if cookie matches URL.
*/
static cookieMatchesURL(cookie: ICookie, url: URL): boolean;
}
//# sourceMappingURL=CookieURLUtility.d.ts.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieURLUtility.d.ts","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieURLUtility.ts"],"names":[],"mappings":"AACA,OAAO,GAAG,MAAM,kBAAkB,CAAC;AACnC,OAAO,OAAO,MAAM,eAAe,CAAC;AAEpC;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,gBAAgB;IACpC;;;;;;OAMG;WACW,gBAAgB,CAAC,MAAM,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,GAAG,OAAO;CAWlE"}

View File

@@ -0,0 +1,23 @@
import CookieSameSiteEnum from '../enums/CookieSameSiteEnum.js';
/**
* Cookie string.
*/
export default class CookieURLUtility {
/**
* Returns "true" if cookie matches URL.
*
* @param cookie Cookie.
* @param url URL.
* @returns "true" if cookie matches URL.
*/
static cookieMatchesURL(cookie, url) {
const isLocalhost = url.hostname === 'localhost' || url.hostname.endsWith('.localhost');
return ((!cookie.secure || url.protocol === 'https:' || isLocalhost) &&
(!cookie.domain || url.hostname.endsWith(cookie.domain)) &&
(!cookie.path || url.pathname.startsWith(cookie.path)) &&
// @see https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#samesitesamesite-value
((cookie.sameSite === CookieSameSiteEnum.none && cookie.secure) ||
cookie.originURL.hostname === url.hostname));
}
}
//# sourceMappingURL=CookieURLUtility.js.map

View File

@@ -0,0 +1 @@
{"version":3,"file":"CookieURLUtility.js","sourceRoot":"","sources":["../../../src/cookie/urilities/CookieURLUtility.ts"],"names":[],"mappings":"AAAA,OAAO,kBAAkB,MAAM,gCAAgC,CAAC;AAIhE;;GAEG;AACH,MAAM,CAAC,OAAO,OAAO,gBAAgB;IACpC;;;;;;OAMG;IACI,MAAM,CAAC,gBAAgB,CAAC,MAAe,EAAE,GAAQ;QACvD,MAAM,WAAW,GAAG,GAAG,CAAC,QAAQ,KAAK,WAAW,IAAI,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;QACxF,OAAO,CACN,CAAC,CAAC,MAAM,CAAC,MAAM,IAAI,GAAG,CAAC,QAAQ,KAAK,QAAQ,IAAI,WAAW,CAAC;YAC5D,CAAC,CAAC,MAAM,CAAC,MAAM,IAAI,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACxD,CAAC,CAAC,MAAM,CAAC,IAAI,IAAI,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YACtD,mGAAmG;YACnG,CAAC,CAAC,MAAM,CAAC,QAAQ,KAAK,kBAAkB,CAAC,IAAI,IAAI,MAAM,CAAC,MAAM,CAAC;gBAC9D,MAAM,CAAC,SAAS,CAAC,QAAQ,KAAK,GAAG,CAAC,QAAQ,CAAC,CAC5C,CAAC;IACH,CAAC;CACD"}