@htlkg/astro 0.0.14 → 0.0.18
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{chunk-TJ5M2EBN.js → chunk-QWEHFUIH.js} +81 -11
- package/dist/chunk-QWEHFUIH.js.map +1 -0
- package/dist/htlkg/index.js +1 -2
- package/dist/index.js +1 -2
- package/dist/index.js.map +1 -1
- package/dist/middleware/index.js +4 -6
- package/dist/middleware/index.js.map +1 -1
- package/dist/runtime-config.js +6 -5
- package/dist/runtime-config.js.map +1 -1
- package/dist/vue-app-setup.js +2 -5
- package/dist/vue-app-setup.js.map +1 -1
- package/package.json +19 -23
- package/src/auth/LoginPage.astro +5 -3
- package/src/htlkg/index.ts +124 -12
- package/src/middleware/auth.ts +3 -3
- package/src/middleware/route-guards.ts +4 -4
- package/src/runtime-config.ts +12 -41
- package/src/vue-app-setup.ts +4 -4
- package/dist/chunk-LP4XHGBI.js +0 -22
- package/dist/chunk-LP4XHGBI.js.map +0 -1
- package/dist/chunk-TJ5M2EBN.js.map +0 -1
|
@@ -1,14 +1,65 @@
|
|
|
1
|
-
import {
|
|
2
|
-
setHtlkgConfig
|
|
3
|
-
} from "./chunk-LP4XHGBI.js";
|
|
4
|
-
|
|
5
1
|
// src/htlkg/index.ts
|
|
6
2
|
import tailwind from "@astrojs/tailwind";
|
|
7
3
|
import vue from "@astrojs/vue";
|
|
4
|
+
import inoxToolsRequestNanostores from "@inox-tools/request-nanostores";
|
|
8
5
|
var DEFAULT_ENV_VARS = [
|
|
9
6
|
"PUBLIC_COGNITO_USER_POOL_ID",
|
|
10
7
|
"PUBLIC_COGNITO_USER_POOL_CLIENT_ID"
|
|
11
8
|
];
|
|
9
|
+
var VIRTUAL_MODULE_ID = "virtual:htlkg";
|
|
10
|
+
var RESOLVED_VIRTUAL_MODULE_ID = "\0" + VIRTUAL_MODULE_ID;
|
|
11
|
+
function createVirtualConfigPlugin(config) {
|
|
12
|
+
return {
|
|
13
|
+
name: "htlkg-virtual-config",
|
|
14
|
+
resolveId(id) {
|
|
15
|
+
if (id === VIRTUAL_MODULE_ID) {
|
|
16
|
+
return RESOLVED_VIRTUAL_MODULE_ID;
|
|
17
|
+
}
|
|
18
|
+
},
|
|
19
|
+
load(id) {
|
|
20
|
+
if (id === RESOLVED_VIRTUAL_MODULE_ID) {
|
|
21
|
+
const serializedConfig = serializeConfig(config);
|
|
22
|
+
return `export const htlkgConfig = ${serializedConfig};`;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
function serializeConfig(config) {
|
|
28
|
+
const routeGuard = config.routeGuard || {};
|
|
29
|
+
const routeGuardParts = [];
|
|
30
|
+
if (routeGuard.publicRoutes) {
|
|
31
|
+
routeGuardParts.push(`publicRoutes: ${serializePatterns(routeGuard.publicRoutes)}`);
|
|
32
|
+
}
|
|
33
|
+
if (routeGuard.authenticatedRoutes) {
|
|
34
|
+
routeGuardParts.push(`authenticatedRoutes: ${serializePatterns(routeGuard.authenticatedRoutes)}`);
|
|
35
|
+
}
|
|
36
|
+
if (routeGuard.adminRoutes) {
|
|
37
|
+
routeGuardParts.push(`adminRoutes: ${serializePatterns(routeGuard.adminRoutes)}`);
|
|
38
|
+
}
|
|
39
|
+
if (routeGuard.brandRoutes && routeGuard.brandRoutes.length > 0) {
|
|
40
|
+
const brandRoutesStr = routeGuard.brandRoutes.map(
|
|
41
|
+
(br) => `{ pattern: ${br.pattern.toString()}, brandIdParam: ${br.brandIdParam} }`
|
|
42
|
+
).join(", ");
|
|
43
|
+
routeGuardParts.push(`brandRoutes: [${brandRoutesStr}]`);
|
|
44
|
+
}
|
|
45
|
+
if (routeGuard.loginUrl) {
|
|
46
|
+
routeGuardParts.push(`loginUrl: ${JSON.stringify(routeGuard.loginUrl)}`);
|
|
47
|
+
}
|
|
48
|
+
return `{
|
|
49
|
+
routeGuardConfig: { ${routeGuardParts.join(", ")} },
|
|
50
|
+
loginPageConfig: ${JSON.stringify(config.loginPage)},
|
|
51
|
+
amplifyConfig: ${JSON.stringify(config.amplify)}
|
|
52
|
+
}`;
|
|
53
|
+
}
|
|
54
|
+
function serializePatterns(patterns) {
|
|
55
|
+
const serialized = patterns.map((p) => {
|
|
56
|
+
if (typeof p === "string") {
|
|
57
|
+
return JSON.stringify(p);
|
|
58
|
+
}
|
|
59
|
+
return p.toString();
|
|
60
|
+
});
|
|
61
|
+
return `[${serialized.join(", ")}]`;
|
|
62
|
+
}
|
|
12
63
|
function htlkg(options = {}) {
|
|
13
64
|
const {
|
|
14
65
|
auth = {},
|
|
@@ -32,6 +83,12 @@ function htlkg(options = {}) {
|
|
|
32
83
|
} else {
|
|
33
84
|
integrations.push(vue());
|
|
34
85
|
}
|
|
86
|
+
integrations.push(inoxToolsRequestNanostores());
|
|
87
|
+
const htlkgVirtualConfig = {
|
|
88
|
+
routeGuard: auth,
|
|
89
|
+
loginPage: loginPage || null,
|
|
90
|
+
amplify: amplify || null
|
|
91
|
+
};
|
|
35
92
|
integrations.push({
|
|
36
93
|
name: "@htlkg/astro",
|
|
37
94
|
hooks: {
|
|
@@ -47,20 +104,24 @@ function htlkg(options = {}) {
|
|
|
47
104
|
logger.warn(`Missing env vars: ${missing.join(", ")}`);
|
|
48
105
|
}
|
|
49
106
|
}
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
amplify: amplify || null
|
|
54
|
-
});
|
|
107
|
+
const vitePlugins = [
|
|
108
|
+
createVirtualConfigPlugin(htlkgVirtualConfig)
|
|
109
|
+
];
|
|
55
110
|
const isDev = process.env.NODE_ENV !== "production";
|
|
56
111
|
if (isDev) {
|
|
57
112
|
try {
|
|
58
113
|
const vueDevTools = (await import("vite-plugin-vue-devtools")).default;
|
|
59
|
-
|
|
114
|
+
vitePlugins.push(vueDevTools());
|
|
60
115
|
logger.info("Vue DevTools enabled");
|
|
61
116
|
} catch {
|
|
62
117
|
}
|
|
63
118
|
}
|
|
119
|
+
updateConfig({
|
|
120
|
+
vite: {
|
|
121
|
+
plugins: vitePlugins
|
|
122
|
+
}
|
|
123
|
+
});
|
|
124
|
+
logger.info("virtual:htlkg module configured");
|
|
64
125
|
if (useFullVueSetup) {
|
|
65
126
|
addMiddleware({ entrypoint: "@htlkg/astro/middleware", order: "pre" });
|
|
66
127
|
logger.info("Authentication middleware injected");
|
|
@@ -86,6 +147,15 @@ function htlkg(options = {}) {
|
|
|
86
147
|
filename: "htlkg.d.ts",
|
|
87
148
|
content: `
|
|
88
149
|
import type { AuthUser } from '@htlkg/core/types';
|
|
150
|
+
import type { RouteGuardConfig, LoginPageConfig } from '@htlkg/astro/htlkg/config';
|
|
151
|
+
|
|
152
|
+
declare module 'virtual:htlkg' {
|
|
153
|
+
export const htlkgConfig: {
|
|
154
|
+
routeGuardConfig: RouteGuardConfig | null;
|
|
155
|
+
loginPageConfig: LoginPageConfig | null;
|
|
156
|
+
amplifyConfig: Record<string, unknown> | null;
|
|
157
|
+
};
|
|
158
|
+
}
|
|
89
159
|
|
|
90
160
|
declare global {
|
|
91
161
|
namespace App {
|
|
@@ -107,4 +177,4 @@ export {};
|
|
|
107
177
|
export {
|
|
108
178
|
htlkg
|
|
109
179
|
};
|
|
110
|
-
//# sourceMappingURL=chunk-
|
|
180
|
+
//# sourceMappingURL=chunk-QWEHFUIH.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/htlkg/index.ts"],"sourcesContent":["/**\n * htlkg Astro Integration\n *\n * Supports static, hybrid, and full SSR output modes.\n * Uses a virtual Vite module (virtual:htlkg) to provide configuration\n * to middleware and Vue app setup.\n */\n\nimport tailwind from '@astrojs/tailwind';\nimport vue from '@astrojs/vue';\nimport inoxToolsRequestNanostores from '@inox-tools/request-nanostores';\nimport type { AstroIntegration } from 'astro';\nimport type { Plugin } from 'vite';\nimport type { HtlkgIntegrationOptions, RouteGuardConfig, LoginPageConfig } from './config.js';\n\nconst DEFAULT_ENV_VARS = [\n\t'PUBLIC_COGNITO_USER_POOL_ID',\n\t'PUBLIC_COGNITO_USER_POOL_CLIENT_ID'\n];\n\nconst VIRTUAL_MODULE_ID = 'virtual:htlkg';\nconst RESOLVED_VIRTUAL_MODULE_ID = '\\0' + VIRTUAL_MODULE_ID;\n\n/**\n * Create Vite plugin that provides the virtual:htlkg module\n */\nfunction createVirtualConfigPlugin(config: {\n\trouteGuard: RouteGuardConfig;\n\tloginPage: LoginPageConfig | null;\n\tamplify: Record<string, unknown> | null;\n}): Plugin {\n\treturn {\n\t\tname: 'htlkg-virtual-config',\n\t\tresolveId(id) {\n\t\t\tif (id === VIRTUAL_MODULE_ID) {\n\t\t\t\treturn RESOLVED_VIRTUAL_MODULE_ID;\n\t\t\t}\n\t\t},\n\t\tload(id) {\n\t\t\tif (id === RESOLVED_VIRTUAL_MODULE_ID) {\n\t\t\t\t// Serialize the config to be loaded as a module\n\t\t\t\t// We need to handle RegExp serialization for route patterns\n\t\t\t\tconst serializedConfig = serializeConfig(config);\n\t\t\t\treturn `export const htlkgConfig = ${serializedConfig};`;\n\t\t\t}\n\t\t},\n\t};\n}\n\n/**\n * Serialize config object to JavaScript code\n * Handles RegExp patterns which can't be JSON.stringify'd\n */\nfunction serializeConfig(config: {\n\trouteGuard: RouteGuardConfig;\n\tloginPage: LoginPageConfig | null;\n\tamplify: Record<string, unknown> | null;\n}): string {\n\tconst routeGuard = config.routeGuard || {};\n\n\t// Build route guard config with proper RegExp handling\n\tconst routeGuardParts: string[] = [];\n\n\tif (routeGuard.publicRoutes) {\n\t\trouteGuardParts.push(`publicRoutes: ${serializePatterns(routeGuard.publicRoutes)}`);\n\t}\n\tif (routeGuard.authenticatedRoutes) {\n\t\trouteGuardParts.push(`authenticatedRoutes: ${serializePatterns(routeGuard.authenticatedRoutes)}`);\n\t}\n\tif (routeGuard.adminRoutes) {\n\t\trouteGuardParts.push(`adminRoutes: ${serializePatterns(routeGuard.adminRoutes)}`);\n\t}\n\tif (routeGuard.brandRoutes && routeGuard.brandRoutes.length > 0) {\n\t\tconst brandRoutesStr = routeGuard.brandRoutes.map(br =>\n\t\t\t`{ pattern: ${br.pattern.toString()}, brandIdParam: ${br.brandIdParam} }`\n\t\t).join(', ');\n\t\trouteGuardParts.push(`brandRoutes: [${brandRoutesStr}]`);\n\t}\n\tif (routeGuard.loginUrl) {\n\t\trouteGuardParts.push(`loginUrl: ${JSON.stringify(routeGuard.loginUrl)}`);\n\t}\n\n\treturn `{\n\trouteGuardConfig: { ${routeGuardParts.join(', ')} },\n\tloginPageConfig: ${JSON.stringify(config.loginPage)},\n\tamplifyConfig: ${JSON.stringify(config.amplify)}\n}`;\n}\n\n/**\n * Serialize an array of route patterns (strings or RegExp)\n */\nfunction serializePatterns(patterns: (string | RegExp)[]): string {\n\tconst serialized = patterns.map(p => {\n\t\tif (typeof p === 'string') {\n\t\t\treturn JSON.stringify(p);\n\t\t}\n\t\t// RegExp - output as literal\n\t\treturn p.toString();\n\t});\n\treturn `[${serialized.join(', ')}]`;\n}\n\nexport function htlkg(\n\toptions: HtlkgIntegrationOptions = {},\n): AstroIntegration | AstroIntegration[] {\n\tconst {\n\t\tauth = {},\n\t\tloginPage = { path: '/login', title: 'Sign In', redirectUrl: '/admin' },\n\t\tvalidateEnv = true,\n\t\trequiredEnvVars = DEFAULT_ENV_VARS,\n\t\ttailwind: tailwindOptions,\n\t\tamplify,\n\t\tvueAppSetup = 'auto',\n\t} = options;\n\n\tconst integrations: AstroIntegration[] = [];\n\n\t// Add Tailwind integration (enabled by default)\n\tif (tailwindOptions !== false) {\n\t\tconst tailwindConfig =\n\t\t\ttypeof tailwindOptions === 'object' ? tailwindOptions : undefined;\n\t\tintegrations.push(\n\t\t\ttailwind(tailwindConfig as Parameters<typeof tailwind>[0]),\n\t\t);\n\t}\n\n\t// Determine Vue setup mode:\n\t// - 'full': Use Amplify app entrypoint (requires SSR)\n\t// - 'basic': Basic Vue without app entrypoint (works with static)\n\t// - 'auto': Default to 'basic' for compatibility with static builds\n\tconst useFullVueSetup = vueAppSetup === 'full';\n\n\t// Add Vue integration\n\tif (useFullVueSetup) {\n\t\tintegrations.push(vue({ appEntrypoint: '@htlkg/astro/vue-app-setup' }));\n\t} else {\n\t\tintegrations.push(vue());\n\t}\n\n\t// Add request nanostores integration (for SSR-safe nanostores)\n\tintegrations.push(inoxToolsRequestNanostores());\n\n\t// Prepare configuration for virtual module\n\tconst htlkgVirtualConfig = {\n\t\trouteGuard: auth,\n\t\tloginPage: loginPage || null,\n\t\tamplify: amplify || null,\n\t};\n\n\t// Add the main htlkg integration\n\tintegrations.push({\n\t\tname: '@htlkg/astro',\n\t\thooks: {\n\t\t\t'astro:config:setup': async ({ updateConfig, addMiddleware, injectRoute, logger }) => {\n\t\t\t\ttry {\n\t\t\t\t\tlogger.info(useFullVueSetup\n\t\t\t\t\t\t? 'Vue configured with Amplify app setup'\n\t\t\t\t\t\t: 'Vue configured (basic mode)');\n\n\t\t\t\t\tif (amplify) {\n\t\t\t\t\t\tlogger.info('Amplify configuration provided');\n\t\t\t\t\t}\n\n\t\t\t\t\t// Validate env vars (only for full setup)\n\t\t\t\t\tif (validateEnv && !amplify && useFullVueSetup) {\n\t\t\t\t\t\tconst missing = requiredEnvVars.filter(v => !process.env[v]);\n\t\t\t\t\t\tif (missing.length > 0) {\n\t\t\t\t\t\t\tlogger.warn(`Missing env vars: ${missing.join(', ')}`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Create virtual module plugin for config\n\t\t\t\t\tconst vitePlugins: Plugin[] = [\n\t\t\t\t\t\tcreateVirtualConfigPlugin(htlkgVirtualConfig),\n\t\t\t\t\t];\n\n\t\t\t\t\t// Only load Vue DevTools during development\n\t\t\t\t\tconst isDev = process.env.NODE_ENV !== 'production';\n\t\t\t\t\tif (isDev) {\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tconst vueDevTools = (await import('vite-plugin-vue-devtools')).default;\n\t\t\t\t\t\t\tvitePlugins.push(vueDevTools());\n\t\t\t\t\t\t\tlogger.info('Vue DevTools enabled');\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t// DevTools not available, skip silently\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Update Vite config with plugins\n\t\t\t\t\tupdateConfig({\n\t\t\t\t\t\tvite: {\n\t\t\t\t\t\t\tplugins: vitePlugins\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\n\t\t\t\t\tlogger.info('virtual:htlkg module configured');\n\n\t\t\t\t\t// Inject middleware (only for full setup)\n\t\t\t\t\tif (useFullVueSetup) {\n\t\t\t\t\t\taddMiddleware({ entrypoint: '@htlkg/astro/middleware', order: 'pre' });\n\t\t\t\t\t\tlogger.info('Authentication middleware injected');\n\t\t\t\t\t}\n\n\t\t\t\t\t// Inject login page (only for full setup)\n\t\t\t\t\tif (loginPage !== false && useFullVueSetup) {\n\t\t\t\t\t\tconst loginPath = loginPage.path || '/login';\n\t\t\t\t\t\tinjectRoute({\n\t\t\t\t\t\t\tpattern: loginPath,\n\t\t\t\t\t\t\tentrypoint: '@htlkg/astro/auth/LoginPage.astro',\n\t\t\t\t\t\t\tprerender: false,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tlogger.info(`Login page injected at ${loginPath}`);\n\t\t\t\t\t}\n\n\t\t\t\t\tlogger.info('htlkg integration configured');\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconst msg = error instanceof Error ? error.message : 'Unknown error';\n\t\t\t\t\tlogger.error(`htlkg configuration failed: ${msg}`);\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t},\n\t\t\t'astro:config:done': ({ injectTypes }) => {\n\t\t\t\tinjectTypes({\n\t\t\t\t\tfilename: 'htlkg.d.ts',\n\t\t\t\t\tcontent: `\nimport type { AuthUser } from '@htlkg/core/types';\nimport type { RouteGuardConfig, LoginPageConfig } from '@htlkg/astro/htlkg/config';\n\ndeclare module 'virtual:htlkg' {\n export const htlkgConfig: {\n routeGuardConfig: RouteGuardConfig | null;\n loginPageConfig: LoginPageConfig | null;\n amplifyConfig: Record<string, unknown> | null;\n };\n}\n\ndeclare global {\n namespace App {\n interface Locals {\n user: AuthUser | null;\n }\n }\n}\n\nexport {};\n`,\n\t\t\t\t});\n\t\t\t},\n\t\t},\n\t});\n\n\treturn integrations;\n}\n"],"mappings":"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","names":[]}
|
package/dist/htlkg/index.js
CHANGED
package/dist/index.js
CHANGED
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/factories/createListPage.ts"],"sourcesContent":["/**\n * List Page Factory\n *\n * Factory function for creating standardized list pages with automatic\n * data fetching, filtering, sorting, and pagination.\n */\n\nimport type { AstroGlobal } from \"astro\";\nimport type { WritableAtom } from \"nanostores\";\n\nimport {\n\tparseListParams,\n\ttype ListParams,\n\ttype ListParamConfig,\n\ttype FilterFieldConfig,\n} from \"../utils/params\";\nimport {\n\tbuildGraphQLFilter,\n\tapplyClientFilters,\n\tsortItems,\n\tpaginateItems,\n} from \"../utils/filters\";\n\n/**\n * Breadcrumb item\n */\nexport interface BreadcrumbItem {\n\tlabel: string;\n\thref?: string;\n}\n\n/**\n * Layout props returned by the factory\n */\nexport interface LayoutProps {\n\ttitle: string;\n\tdescription?: string;\n\tcurrentPage: string;\n\tbreadcrumbs: BreadcrumbItem[];\n}\n\n/**\n * Table initial state\n */\nexport interface TableInitialState {\n\tcurrentPage: number;\n\tpageSize: number;\n\tsortKey: string;\n\tsortOrder: \"asc\" | \"desc\";\n\ttotalItems: number;\n\ttotalPages: number;\n\tfilters: Record<string, any>;\n\tsearch?: string;\n}\n\n/**\n * Related store configuration\n */\nexport interface RelatedStoreConfig<T = any> {\n\tstore: WritableAtom<T[]>;\n\tfetch: () => Promise<T[]>;\n}\n\n/**\n * Configuration for createListPage\n */\nexport interface ListPageConfig<T> {\n\t// Page metadata\n\t/** Page title */\n\ttitle: string;\n\t/** Page description */\n\tdescription?: string;\n\t/** Page identifier (for sidebar active state) */\n\tpageId: string;\n\t/** Custom breadcrumbs (auto-generated if not provided) */\n\tbreadcrumbs?: BreadcrumbItem[];\n\n\t// Data fetching\n\t/** Fetch function that retrieves data */\n\tfetchFn: (params: {\n\t\tfilter: any;\n\t\tlimit?: number;\n\t\tnextToken?: string;\n\t}) => Promise<{ data: T[]; nextToken?: string | null }>;\n\n\t// Data handling\n\t/** Transform function to apply to fetched data */\n\ttransform?: (item: any) => T;\n\t/** Main store to set with data */\n\tstore: WritableAtom<T[]>;\n\t/** Related stores to populate */\n\trelatedStores?: Record<string, RelatedStoreConfig>;\n\n\t// Filtering/sorting config\n\t/** Searchable field keys */\n\tsearchableFields?: string[];\n\t/** Filterable field configurations */\n\tfilterableFields?: FilterFieldConfig[];\n\t/** Sortable field keys (all by default) */\n\tsortableFields?: string[];\n\t/** Default sort configuration */\n\tdefaultSort?: { key: string; order: \"asc\" | \"desc\" };\n\t/** Default page size */\n\tdefaultPageSize?: number;\n\t/** Maximum items to fetch from API */\n\tfetchLimit?: number;\n}\n\n/**\n * Result from createListPage\n */\nexport interface ListPageResult<T> {\n\t/** Props to spread to Layout component */\n\tlayoutProps: LayoutProps;\n\t/** Initial state for table component */\n\tinitialState: TableInitialState;\n\t/** Processed items (paginated) */\n\titems: T[];\n\t/** All items (pre-pagination, for client-side operations) */\n\tallItems: T[];\n\t/** Parsed URL parameters */\n\tparams: ListParams;\n}\n\n/**\n * Default breadcrumb generator\n */\nfunction generateBreadcrumbs(pageId: string, title: string): BreadcrumbItem[] {\n\treturn [\n\t\t{ label: \"Admin\", href: \"/admin\" },\n\t\t{ label: title },\n\t];\n}\n\n/**\n * Create a list page with automatic data handling\n *\n * This factory handles:\n * - URL parameter parsing (page, pageSize, sortKey, sortOrder, filters)\n * - Data fetching with GraphQL filter building\n * - Data transformation\n * - Client-side filtering (for computed fields)\n * - Sorting\n * - Pagination\n * - Store population\n *\n * @example\n * ```astro\n * ---\n * import { createListPage } from '@htlkg/astro/factories';\n * import { $accounts } from '@/stores/accounts';\n *\n * const { layoutProps, initialState, items } = await createListPage(Astro, {\n * title: 'Accounts',\n * pageId: 'accounts',\n * store: $accounts,\n *\n * fetchFn: async ({ filter }) => {\n * const client = getServerClient(Astro);\n * return await client.models.Account.list({ filter });\n * },\n *\n * transform: (account) => ({\n * id: account.id,\n * name: account.name,\n * brandCount: account.brands?.length ?? 0,\n * }),\n *\n * filterableFields: [\n * { key: 'name', type: 'text', graphql: true },\n * { key: 'status', type: 'select', graphql: true, options: ['active', 'inactive'] },\n * ],\n *\n * defaultSort: { key: 'name', order: 'asc' },\n * });\n * ---\n *\n * <Layout {...layoutProps}>\n * <AccountsTable client:load initialState={initialState} />\n * </Layout>\n * ```\n */\nexport async function createListPage<T extends Record<string, any>>(\n\tastro: AstroGlobal,\n\tconfig: ListPageConfig<T>\n): Promise<ListPageResult<T>> {\n\tconst {\n\t\ttitle,\n\t\tdescription,\n\t\tpageId,\n\t\tbreadcrumbs,\n\t\tfetchFn,\n\t\ttransform,\n\t\tstore,\n\t\trelatedStores,\n\t\tsearchableFields,\n\t\tfilterableFields = [],\n\t\tdefaultSort,\n\t\tdefaultPageSize = 25,\n\t\tfetchLimit = 1000,\n\t} = config;\n\n\t// 1. Parse URL parameters\n\tconst paramConfig: ListParamConfig = {\n\t\tdefaultPageSize,\n\t\tdefaultSort,\n\t\tsearchableFields,\n\t\tfilterableFields,\n\t};\n\tconst params = parseListParams(astro.url, paramConfig);\n\n\t// 2. Build GraphQL filter from URL params\n\tconst graphqlFilter = buildGraphQLFilter(params.filters, filterableFields, {\n\t\tsearch: params.search,\n\t\tsearchFields: searchableFields,\n\t});\n\n\t// 3. Fetch data with pagination\n\tlet allItems: T[] = [];\n\tlet nextToken: string | null | undefined = undefined;\n\n\tdo {\n\t\tconst result = await fetchFn({\n\t\t\tfilter: graphqlFilter,\n\t\t\tlimit: fetchLimit,\n\t\t\tnextToken: nextToken ?? undefined,\n\t\t});\n\n\t\tconst data = result.data ?? [];\n\t\tallItems = allItems.concat(data);\n\t\tnextToken = result.nextToken;\n\t} while (nextToken);\n\n\t// 4. Transform data if needed\n\tif (transform) {\n\t\tallItems = allItems.map(transform);\n\t}\n\n\t// 5. Apply client-side filtering (for computed fields not in GraphQL)\n\tallItems = applyClientFilters(allItems, params.filters, filterableFields, {\n\t\tsearch: params.search,\n\t\tsearchFields: searchableFields,\n\t});\n\n\t// 6. Sort\n\tallItems = sortItems(allItems, params.sortKey, params.sortOrder);\n\n\t// 7. Paginate\n\tconst { paginatedItems, totalItems, totalPages, currentPage, pageSize } = paginateItems(\n\t\tallItems,\n\t\tparams.page,\n\t\tparams.pageSize\n\t);\n\n\t// 8. Set main store\n\tstore.set(paginatedItems);\n\n\t// 9. Fetch and set related stores\n\tif (relatedStores) {\n\t\tawait Promise.all(\n\t\t\tObject.entries(relatedStores).map(async ([, config]) => {\n\t\t\t\tconst data = await config.fetch();\n\t\t\t\tconfig.store.set(data);\n\t\t\t})\n\t\t);\n\t}\n\n\t// 10. Build result\n\treturn {\n\t\tlayoutProps: {\n\t\t\ttitle,\n\t\t\tdescription,\n\t\t\tcurrentPage: pageId,\n\t\t\tbreadcrumbs: breadcrumbs ?? generateBreadcrumbs(pageId, title),\n\t\t},\n\t\tinitialState: {\n\t\t\tcurrentPage,\n\t\t\tpageSize,\n\t\t\tsortKey: params.sortKey,\n\t\t\tsortOrder: params.sortOrder,\n\t\t\ttotalItems,\n\t\t\ttotalPages,\n\t\t\tfilters: params.filters,\n\t\t\tsearch: params.search,\n\t\t},\n\t\titems: paginatedItems,\n\t\tallItems,\n\t\tparams,\n\t};\n}\n"],"mappings":"
|
|
1
|
+
{"version":3,"sources":["../src/factories/createListPage.ts"],"sourcesContent":["/**\n * List Page Factory\n *\n * Factory function for creating standardized list pages with automatic\n * data fetching, filtering, sorting, and pagination.\n */\n\nimport type { AstroGlobal } from \"astro\";\nimport type { WritableAtom } from \"nanostores\";\n\nimport {\n\tparseListParams,\n\ttype ListParams,\n\ttype ListParamConfig,\n\ttype FilterFieldConfig,\n} from \"../utils/params\";\nimport {\n\tbuildGraphQLFilter,\n\tapplyClientFilters,\n\tsortItems,\n\tpaginateItems,\n} from \"../utils/filters\";\n\n/**\n * Breadcrumb item\n */\nexport interface BreadcrumbItem {\n\tlabel: string;\n\thref?: string;\n}\n\n/**\n * Layout props returned by the factory\n */\nexport interface LayoutProps {\n\ttitle: string;\n\tdescription?: string;\n\tcurrentPage: string;\n\tbreadcrumbs: BreadcrumbItem[];\n}\n\n/**\n * Table initial state\n */\nexport interface TableInitialState {\n\tcurrentPage: number;\n\tpageSize: number;\n\tsortKey: string;\n\tsortOrder: \"asc\" | \"desc\";\n\ttotalItems: number;\n\ttotalPages: number;\n\tfilters: Record<string, any>;\n\tsearch?: string;\n}\n\n/**\n * Related store configuration\n */\nexport interface RelatedStoreConfig<T = any> {\n\tstore: WritableAtom<T[]>;\n\tfetch: () => Promise<T[]>;\n}\n\n/**\n * Configuration for createListPage\n */\nexport interface ListPageConfig<T> {\n\t// Page metadata\n\t/** Page title */\n\ttitle: string;\n\t/** Page description */\n\tdescription?: string;\n\t/** Page identifier (for sidebar active state) */\n\tpageId: string;\n\t/** Custom breadcrumbs (auto-generated if not provided) */\n\tbreadcrumbs?: BreadcrumbItem[];\n\n\t// Data fetching\n\t/** Fetch function that retrieves data */\n\tfetchFn: (params: {\n\t\tfilter: any;\n\t\tlimit?: number;\n\t\tnextToken?: string;\n\t}) => Promise<{ data: T[]; nextToken?: string | null }>;\n\n\t// Data handling\n\t/** Transform function to apply to fetched data */\n\ttransform?: (item: any) => T;\n\t/** Main store to set with data */\n\tstore: WritableAtom<T[]>;\n\t/** Related stores to populate */\n\trelatedStores?: Record<string, RelatedStoreConfig>;\n\n\t// Filtering/sorting config\n\t/** Searchable field keys */\n\tsearchableFields?: string[];\n\t/** Filterable field configurations */\n\tfilterableFields?: FilterFieldConfig[];\n\t/** Sortable field keys (all by default) */\n\tsortableFields?: string[];\n\t/** Default sort configuration */\n\tdefaultSort?: { key: string; order: \"asc\" | \"desc\" };\n\t/** Default page size */\n\tdefaultPageSize?: number;\n\t/** Maximum items to fetch from API */\n\tfetchLimit?: number;\n}\n\n/**\n * Result from createListPage\n */\nexport interface ListPageResult<T> {\n\t/** Props to spread to Layout component */\n\tlayoutProps: LayoutProps;\n\t/** Initial state for table component */\n\tinitialState: TableInitialState;\n\t/** Processed items (paginated) */\n\titems: T[];\n\t/** All items (pre-pagination, for client-side operations) */\n\tallItems: T[];\n\t/** Parsed URL parameters */\n\tparams: ListParams;\n}\n\n/**\n * Default breadcrumb generator\n */\nfunction generateBreadcrumbs(pageId: string, title: string): BreadcrumbItem[] {\n\treturn [\n\t\t{ label: \"Admin\", href: \"/admin\" },\n\t\t{ label: title },\n\t];\n}\n\n/**\n * Create a list page with automatic data handling\n *\n * This factory handles:\n * - URL parameter parsing (page, pageSize, sortKey, sortOrder, filters)\n * - Data fetching with GraphQL filter building\n * - Data transformation\n * - Client-side filtering (for computed fields)\n * - Sorting\n * - Pagination\n * - Store population\n *\n * @example\n * ```astro\n * ---\n * import { createListPage } from '@htlkg/astro/factories';\n * import { $accounts } from '@/stores/accounts';\n *\n * const { layoutProps, initialState, items } = await createListPage(Astro, {\n * title: 'Accounts',\n * pageId: 'accounts',\n * store: $accounts,\n *\n * fetchFn: async ({ filter }) => {\n * const client = getServerClient(Astro);\n * return await client.models.Account.list({ filter });\n * },\n *\n * transform: (account) => ({\n * id: account.id,\n * name: account.name,\n * brandCount: account.brands?.length ?? 0,\n * }),\n *\n * filterableFields: [\n * { key: 'name', type: 'text', graphql: true },\n * { key: 'status', type: 'select', graphql: true, options: ['active', 'inactive'] },\n * ],\n *\n * defaultSort: { key: 'name', order: 'asc' },\n * });\n * ---\n *\n * <Layout {...layoutProps}>\n * <AccountsTable client:load initialState={initialState} />\n * </Layout>\n * ```\n */\nexport async function createListPage<T extends Record<string, any>>(\n\tastro: AstroGlobal,\n\tconfig: ListPageConfig<T>\n): Promise<ListPageResult<T>> {\n\tconst {\n\t\ttitle,\n\t\tdescription,\n\t\tpageId,\n\t\tbreadcrumbs,\n\t\tfetchFn,\n\t\ttransform,\n\t\tstore,\n\t\trelatedStores,\n\t\tsearchableFields,\n\t\tfilterableFields = [],\n\t\tdefaultSort,\n\t\tdefaultPageSize = 25,\n\t\tfetchLimit = 1000,\n\t} = config;\n\n\t// 1. Parse URL parameters\n\tconst paramConfig: ListParamConfig = {\n\t\tdefaultPageSize,\n\t\tdefaultSort,\n\t\tsearchableFields,\n\t\tfilterableFields,\n\t};\n\tconst params = parseListParams(astro.url, paramConfig);\n\n\t// 2. Build GraphQL filter from URL params\n\tconst graphqlFilter = buildGraphQLFilter(params.filters, filterableFields, {\n\t\tsearch: params.search,\n\t\tsearchFields: searchableFields,\n\t});\n\n\t// 3. Fetch data with pagination\n\tlet allItems: T[] = [];\n\tlet nextToken: string | null | undefined = undefined;\n\n\tdo {\n\t\tconst result = await fetchFn({\n\t\t\tfilter: graphqlFilter,\n\t\t\tlimit: fetchLimit,\n\t\t\tnextToken: nextToken ?? undefined,\n\t\t});\n\n\t\tconst data = result.data ?? [];\n\t\tallItems = allItems.concat(data);\n\t\tnextToken = result.nextToken;\n\t} while (nextToken);\n\n\t// 4. Transform data if needed\n\tif (transform) {\n\t\tallItems = allItems.map(transform);\n\t}\n\n\t// 5. Apply client-side filtering (for computed fields not in GraphQL)\n\tallItems = applyClientFilters(allItems, params.filters, filterableFields, {\n\t\tsearch: params.search,\n\t\tsearchFields: searchableFields,\n\t});\n\n\t// 6. Sort\n\tallItems = sortItems(allItems, params.sortKey, params.sortOrder);\n\n\t// 7. Paginate\n\tconst { paginatedItems, totalItems, totalPages, currentPage, pageSize } = paginateItems(\n\t\tallItems,\n\t\tparams.page,\n\t\tparams.pageSize\n\t);\n\n\t// 8. Set main store\n\tstore.set(paginatedItems);\n\n\t// 9. Fetch and set related stores\n\tif (relatedStores) {\n\t\tawait Promise.all(\n\t\t\tObject.entries(relatedStores).map(async ([, config]) => {\n\t\t\t\tconst data = await config.fetch();\n\t\t\t\tconfig.store.set(data);\n\t\t\t})\n\t\t);\n\t}\n\n\t// 10. Build result\n\treturn {\n\t\tlayoutProps: {\n\t\t\ttitle,\n\t\t\tdescription,\n\t\t\tcurrentPage: pageId,\n\t\t\tbreadcrumbs: breadcrumbs ?? generateBreadcrumbs(pageId, title),\n\t\t},\n\t\tinitialState: {\n\t\t\tcurrentPage,\n\t\t\tpageSize,\n\t\t\tsortKey: params.sortKey,\n\t\t\tsortOrder: params.sortOrder,\n\t\t\ttotalItems,\n\t\t\ttotalPages,\n\t\t\tfilters: params.filters,\n\t\t\tsearch: params.search,\n\t\t},\n\t\titems: paginatedItems,\n\t\tallItems,\n\t\tparams,\n\t};\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA+HA,SAAS,oBAAoB,QAAgB,OAAiC;AAC7E,SAAO;AAAA,IACN,EAAE,OAAO,SAAS,MAAM,SAAS;AAAA,IACjC,EAAE,OAAO,MAAM;AAAA,EAChB;AACD;AAkDA,eAAsB,eACrB,OACA,QAC6B;AAC7B,QAAM;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,mBAAmB,CAAC;AAAA,IACpB;AAAA,IACA,kBAAkB;AAAA,IAClB,aAAa;AAAA,EACd,IAAI;AAGJ,QAAM,cAA+B;AAAA,IACpC;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACD;AACA,QAAM,SAAS,gBAAgB,MAAM,KAAK,WAAW;AAGrD,QAAM,gBAAgB,mBAAmB,OAAO,SAAS,kBAAkB;AAAA,IAC1E,QAAQ,OAAO;AAAA,IACf,cAAc;AAAA,EACf,CAAC;AAGD,MAAI,WAAgB,CAAC;AACrB,MAAI,YAAuC;AAE3C,KAAG;AACF,UAAM,SAAS,MAAM,QAAQ;AAAA,MAC5B,QAAQ;AAAA,MACR,OAAO;AAAA,MACP,WAAW,aAAa;AAAA,IACzB,CAAC;AAED,UAAM,OAAO,OAAO,QAAQ,CAAC;AAC7B,eAAW,SAAS,OAAO,IAAI;AAC/B,gBAAY,OAAO;AAAA,EACpB,SAAS;AAGT,MAAI,WAAW;AACd,eAAW,SAAS,IAAI,SAAS;AAAA,EAClC;AAGA,aAAW,mBAAmB,UAAU,OAAO,SAAS,kBAAkB;AAAA,IACzE,QAAQ,OAAO;AAAA,IACf,cAAc;AAAA,EACf,CAAC;AAGD,aAAW,UAAU,UAAU,OAAO,SAAS,OAAO,SAAS;AAG/D,QAAM,EAAE,gBAAgB,YAAY,YAAY,aAAa,SAAS,IAAI;AAAA,IACzE;AAAA,IACA,OAAO;AAAA,IACP,OAAO;AAAA,EACR;AAGA,QAAM,IAAI,cAAc;AAGxB,MAAI,eAAe;AAClB,UAAM,QAAQ;AAAA,MACb,OAAO,QAAQ,aAAa,EAAE,IAAI,OAAO,CAAC,EAAEA,OAAM,MAAM;AACvD,cAAM,OAAO,MAAMA,QAAO,MAAM;AAChC,QAAAA,QAAO,MAAM,IAAI,IAAI;AAAA,MACtB,CAAC;AAAA,IACF;AAAA,EACD;AAGA,SAAO;AAAA,IACN,aAAa;AAAA,MACZ;AAAA,MACA;AAAA,MACA,aAAa;AAAA,MACb,aAAa,eAAe,oBAAoB,QAAQ,KAAK;AAAA,IAC9D;AAAA,IACA,cAAc;AAAA,MACb;AAAA,MACA;AAAA,MACA,SAAS,OAAO;AAAA,MAChB,WAAW,OAAO;AAAA,MAClB;AAAA,MACA;AAAA,MACA,SAAS,OAAO;AAAA,MAChB,QAAQ,OAAO;AAAA,IAChB;AAAA,IACA,OAAO;AAAA,IACP;AAAA,IACA;AAAA,EACD;AACD;","names":["config"]}
|
package/dist/middleware/index.js
CHANGED
|
@@ -1,7 +1,3 @@
|
|
|
1
|
-
import {
|
|
2
|
-
getHtlkgConfig
|
|
3
|
-
} from "../chunk-LP4XHGBI.js";
|
|
4
|
-
|
|
5
1
|
// src/middleware/index.ts
|
|
6
2
|
import { sequence } from "astro:middleware";
|
|
7
3
|
|
|
@@ -9,11 +5,12 @@ import { sequence } from "astro:middleware";
|
|
|
9
5
|
import { Amplify } from "aws-amplify";
|
|
10
6
|
import { getUser } from "@htlkg/core/auth";
|
|
11
7
|
import { globalSettings } from "@htlkg/core/amplify-astro-adapter";
|
|
8
|
+
import { htlkgConfig } from "virtual:htlkg";
|
|
12
9
|
var amplifyConfigured = false;
|
|
13
10
|
function ensureAmplifyConfigured() {
|
|
14
11
|
if (amplifyConfigured) return;
|
|
15
12
|
try {
|
|
16
|
-
const { amplifyConfig } =
|
|
13
|
+
const { amplifyConfig } = htlkgConfig;
|
|
17
14
|
if (!amplifyConfig) {
|
|
18
15
|
console.warn("[htlkg Auth] No Amplify configuration provided");
|
|
19
16
|
return;
|
|
@@ -48,6 +45,7 @@ var authMiddleware = async (context, next) => {
|
|
|
48
45
|
};
|
|
49
46
|
|
|
50
47
|
// src/middleware/route-guards.ts
|
|
48
|
+
import { htlkgConfig as htlkgConfig2 } from "virtual:htlkg";
|
|
51
49
|
function matchesPattern(pathname, patterns) {
|
|
52
50
|
return patterns.some((pattern) => {
|
|
53
51
|
try {
|
|
@@ -70,7 +68,7 @@ function matchesPattern(pathname, patterns) {
|
|
|
70
68
|
var routeGuard = async (context, next) => {
|
|
71
69
|
const { locals, url, redirect } = context;
|
|
72
70
|
const pathname = url.pathname;
|
|
73
|
-
const { routeGuardConfig } =
|
|
71
|
+
const { routeGuardConfig } = htlkgConfig2;
|
|
74
72
|
const {
|
|
75
73
|
publicRoutes = [],
|
|
76
74
|
authenticatedRoutes = [],
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/middleware/index.ts","../../src/middleware/auth.ts","../../src/middleware/route-guards.ts"],"sourcesContent":["/**\n * Middleware for @htlkg/astro\n * \n * This module exports:\n * - authMiddleware: Retrieves authenticated user and injects into locals\n * - routeGuard: Enforces access control based on route configuration\n * - onRequest: Composed middleware chain (auth + route guard)\n * - Helper functions: requireAuth, requireAdminAccess, requireBrandAccess\n */\n\nimport { sequence } from 'astro:middleware';\nimport { authMiddleware } from './auth.js';\nimport { routeGuard } from './route-guards.js';\n\n// Export individual middleware\nexport { authMiddleware } from './auth.js';\nexport {\n\trouteGuard,\n\trequireAuth,\n\trequireAdminAccess,\n\trequireBrandAccess,\n} from './route-guards.js';\n\n/**\n * Export composed middleware chain\n * Auth middleware runs first to populate locals.user\n * Route guard middleware runs second to enforce access control\n * \n * This is the default export that gets injected by the htlkg integration\n */\nexport const onRequest = sequence(authMiddleware, routeGuard);\n","/**\n * Authentication middleware for htlkg integration\n * \n * This middleware:\n * - Configures AWS Amplify for server-side auth\n * - Retrieves the authenticated user from AWS Amplify\n * - Injects the user into Astro.locals for use in pages and API routes\n * - Handles authentication errors gracefully\n */\n\nimport type { MiddlewareHandler } from 'astro';\nimport { Amplify } from 'aws-amplify';\nimport { getUser } from '@htlkg/core/auth';\nimport { globalSettings } from '@htlkg/core/amplify-astro-adapter';\nimport { getHtlkgConfig } from '../runtime-config.js';\n\n// Track if Amplify has been configured\nlet amplifyConfigured = false;\n\n/**\n * Configure Amplify on first request\n */\nfunction ensureAmplifyConfigured(): void {\n\tif (amplifyConfigured) return;\n\n\ttry {\n\t\tconst { amplifyConfig } = getHtlkgConfig();\n\t\tif (!amplifyConfig) {\n\t\t\tconsole.warn('[htlkg Auth] No Amplify configuration provided');\n\t\t\treturn;\n\t\t}\n\n\t\t// Configure Amplify with SSR support\n\t\tAmplify.configure(amplifyConfig as any, { ssr: true });\n\t\t\n\t\t// Enable server-side auth in the adapter\n\t\tglobalSettings.enableServerSideAuth();\n\t\t\n\t\t// Check if we're on HTTPS (for secure cookies)\n\t\tconst isSSL = typeof window !== 'undefined' \n\t\t\t? window.location.protocol === 'https:' \n\t\t\t: process.env.NODE_ENV === 'production';\n\t\tglobalSettings.setIsSSLOrigin(isSSL);\n\t\t\n\t\tamplifyConfigured = true;\n\t\tconsole.info('[htlkg Auth] Amplify configured for server-side auth');\n\t} catch (error) {\n\t\tconst errorMsg = error instanceof Error ? error.message : 'Unknown error';\n\t\tconsole.error(`[htlkg Auth] Failed to configure Amplify: ${errorMsg}`);\n\t}\n}\n\n/**\n * Auth middleware - retrieves authenticated user and injects into locals\n * \n * This middleware runs on every request and attempts to get the current\n * authenticated user from AWS Amplify. If successful, the user is injected\n * into context.locals.user. If authentication fails, locals.user is set to null.\n * \n * @example\n * // In an Astro page\n * const { user } = Astro.locals;\n * if (user) {\n * console.log('Authenticated user:', user.email);\n * }\n */\nexport const authMiddleware: MiddlewareHandler = async (context, next) => {\n\tconst { locals } = context;\n\n\t// Ensure Amplify is configured before attempting auth\n\tensureAmplifyConfigured();\n\n\ttry {\n\t\tconst user = await getUser(context);\n\t\tlocals.user = user;\n\t} catch (error) {\n\t\t// Set user to null on any authentication error\n\t\tlocals.user = null;\n\n\t\t// Log error without exposing sensitive information\n\t\tif (error instanceof Error) {\n\t\t\t// Filter out sensitive data from error messages\n\t\t\tconst safeMessage = error.message\n\t\t\t\t.replace(/token[=:]\\s*[^\\s,}]+/gi, 'token=***')\n\t\t\t\t.replace(/key[=:]\\s*[^\\s,}]+/gi, 'key=***')\n\t\t\t\t.replace(/secret[=:]\\s*[^\\s,}]+/gi, 'secret=***')\n\t\t\t\t.replace(/password[=:]\\s*[^\\s,}]+/gi, 'password=***');\n\n\t\t\tconsole.error('[htlkg Auth] Authentication failed:', safeMessage);\n\t\t} else {\n\t\t\tconsole.error('[htlkg Auth] Authentication failed: Unknown error');\n\t\t}\n\t}\n\n\treturn next();\n};\n","/**\n * Route guard middleware for htlkg integration\n * \n * This middleware enforces access control based on route configuration:\n * - Public routes: accessible to everyone\n * - Authenticated routes: require any logged-in user\n * - Admin routes: require admin privileges\n * - Brand routes: require brand-specific access or admin privileges\n */\n\nimport type { MiddlewareHandler } from 'astro';\nimport type { RoutePattern, RouteGuardConfig } from '../htlkg/config.js';\nimport { getHtlkgConfig } from '../runtime-config.js';\n\n/**\n * Helper: Check if pathname matches any of the provided patterns\n */\nfunction matchesPattern(pathname: string, patterns: RoutePattern[]): boolean {\n\treturn patterns.some((pattern) => {\n\t\ttry {\n\t\t\tif (typeof pattern === 'string') {\n\t\t\t\t// Special case: \"/\" only matches exactly\n\t\t\t\tif (pattern === '/') {\n\t\t\t\t\treturn pathname === '/';\n\t\t\t\t}\n\t\t\t\t// String pattern: exact match or starts with (for sub-routes)\n\t\t\t\treturn pathname === pattern || pathname.startsWith(pattern + '/');\n\t\t\t}\n\t\t\t// RegExp pattern: test against pathname\n\t\t\treturn pattern.test(pathname);\n\t\t} catch (error) {\n\t\t\t// Log pattern matching errors but don't block the request\n\t\t\tconsole.error(\n\t\t\t\t'[htlkg] Error matching route pattern:',\n\t\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t\t);\n\t\t\treturn false;\n\t\t}\n\t});\n}\n\n/**\n * Route guard middleware - protects routes based on configuration\n * \n * This middleware runs after authMiddleware and enforces access control\n * based on the route configuration provided to the htlkg integration.\n * \n * @example\n * // In astro.config.mjs\n * htlkg({\n * auth: {\n * publicRoutes: ['/login', '/'],\n * adminRoutes: [/^\\/admin/],\n * brandRoutes: [\n * { pattern: /^\\/brands\\/(\\d+)/, brandIdParam: 1 }\n * ],\n * loginUrl: '/login'\n * }\n * })\n */\nexport const routeGuard: MiddlewareHandler = async (context, next) => {\n\tconst { locals, url, redirect } = context;\n\tconst pathname = url.pathname;\n\n\tconst { routeGuardConfig } = getHtlkgConfig();\n\tconst {\n\t\tpublicRoutes = [],\n\t\tauthenticatedRoutes = [],\n\t\tadminRoutes = [],\n\t\tbrandRoutes = [],\n\t\tloginUrl = '/login',\n\t} = routeGuardConfig || {};\n\n\ttry {\n\t\t// Public routes - no auth required\n\t\tif (matchesPattern(pathname, publicRoutes)) {\n\t\t\treturn next();\n\t\t}\n\n\t\tconst user = locals.user;\n\n\t\t// Admin routes - require admin role\n\t\tif (matchesPattern(pathname, adminRoutes)) {\n\t\t\tif (!user || !user.isAdmin) {\n\t\t\t\treturn redirect(`${loginUrl}?error=admin_required`);\n\t\t\t}\n\t\t\treturn next();\n\t\t}\n\n\t\t// Brand routes - require brand access or admin\n\t\tfor (const brandRoute of brandRoutes) {\n\t\t\ttry {\n\t\t\t\tconst match = pathname.match(brandRoute.pattern);\n\t\t\t\tif (match) {\n\t\t\t\t\tconst brandId = Number.parseInt(match[brandRoute.brandIdParam], 10);\n\n\t\t\t\t\tif (Number.isNaN(brandId)) {\n\t\t\t\t\t\tconsole.warn(`[htlkg] Invalid brandId extracted from ${pathname}`);\n\t\t\t\t\t\treturn redirect(`${loginUrl}?error=invalid_brand`);\n\t\t\t\t\t}\n\n\t\t\t\t\tif (!user || (!user.isAdmin && !user.brandIds.includes(brandId))) {\n\t\t\t\t\t\treturn redirect(`${loginUrl}?error=access_denied`);\n\t\t\t\t\t}\n\t\t\t\t\treturn next();\n\t\t\t\t}\n\t\t\t} catch (error) {\n\t\t\t\tconsole.error(\n\t\t\t\t\t`[htlkg] Error processing brand route for ${pathname}:`,\n\t\t\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t\t\t);\n\t\t\t\t// Fail-safe: deny access on error\n\t\t\t\treturn redirect(`${loginUrl}?error=access_denied`);\n\t\t\t}\n\t\t}\n\n\t\t// Authenticated routes - require any logged-in user\n\t\tif (matchesPattern(pathname, authenticatedRoutes)) {\n\t\t\tif (!user) {\n\t\t\t\tconst returnUrl = encodeURIComponent(pathname + url.search);\n\t\t\t\treturn redirect(`${loginUrl}?redirect=${returnUrl}`);\n\t\t\t}\n\t\t\treturn next();\n\t\t}\n\n\t\t// Default: allow access\n\t\treturn next();\n\t} catch (error) {\n\t\t// Catch-all error handler for route guard\n\t\tconsole.error(\n\t\t\t'[htlkg] Unexpected error in route guard:',\n\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t);\n\t\t// Fail-safe: deny access and redirect to login\n\t\treturn redirect(`${loginUrl}?error=server_error`);\n\t}\n};\n\n/**\n * Helper functions for programmatic route protection in pages\n * These can be used in Astro pages for more fine-grained control\n */\n\n/**\n * Require authentication for a page\n * Redirects to login if user is not authenticated\n */\nexport async function requireAuth(context: any, loginUrl = '/login') {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\tconst currentUrl = context.url.pathname + context.url.search;\n\t\tconst encodedReturnUrl = encodeURIComponent(currentUrl);\n\t\treturn context.redirect(`${loginUrl}?redirect=${encodedReturnUrl}`);\n\t}\n\treturn user;\n}\n\n/**\n * Require admin access for a page\n * Redirects to login if user is not an admin\n */\nexport async function requireAdminAccess(context: any, loginUrl = '/login') {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\treturn context.redirect(`${loginUrl}?error=not_authenticated`);\n\t}\n\tif (!user.isAdmin) {\n\t\treturn context.redirect(`${loginUrl}?error=admin_required`);\n\t}\n\treturn user;\n}\n\n/**\n * Require brand access for a page\n * Redirects to login if user doesn't have access to the specified brand\n */\nexport async function requireBrandAccess(\n\tcontext: any,\n\tbrandId: number,\n\tloginUrl = '/login'\n) {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\treturn context.redirect(`${loginUrl}?error=not_authenticated`);\n\t}\n\tif (!user.isAdmin && !user.brandIds.includes(brandId)) {\n\t\treturn context.redirect(`${loginUrl}?error=access_denied`);\n\t}\n\treturn user;\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../../src/middleware/index.ts","../../src/middleware/auth.ts","../../src/middleware/route-guards.ts"],"sourcesContent":["/**\n * Middleware for @htlkg/astro\n * \n * This module exports:\n * - authMiddleware: Retrieves authenticated user and injects into locals\n * - routeGuard: Enforces access control based on route configuration\n * - onRequest: Composed middleware chain (auth + route guard)\n * - Helper functions: requireAuth, requireAdminAccess, requireBrandAccess\n */\n\nimport { sequence } from 'astro:middleware';\nimport { authMiddleware } from './auth.js';\nimport { routeGuard } from './route-guards.js';\n\n// Export individual middleware\nexport { authMiddleware } from './auth.js';\nexport {\n\trouteGuard,\n\trequireAuth,\n\trequireAdminAccess,\n\trequireBrandAccess,\n} from './route-guards.js';\n\n/**\n * Export composed middleware chain\n * Auth middleware runs first to populate locals.user\n * Route guard middleware runs second to enforce access control\n * \n * This is the default export that gets injected by the htlkg integration\n */\nexport const onRequest = sequence(authMiddleware, routeGuard);\n","/**\n * Authentication middleware for htlkg integration\n *\n * This middleware:\n * - Configures AWS Amplify for server-side auth\n * - Retrieves the authenticated user from AWS Amplify\n * - Injects the user into Astro.locals for use in pages and API routes\n * - Handles authentication errors gracefully\n */\n\nimport type { MiddlewareHandler } from 'astro';\nimport { Amplify } from 'aws-amplify';\nimport { getUser } from '@htlkg/core/auth';\nimport { globalSettings } from '@htlkg/core/amplify-astro-adapter';\nimport { htlkgConfig } from 'virtual:htlkg';\n\n// Track if Amplify has been configured\nlet amplifyConfigured = false;\n\n/**\n * Configure Amplify on first request\n */\nfunction ensureAmplifyConfigured(): void {\n\tif (amplifyConfigured) return;\n\n\ttry {\n\t\tconst { amplifyConfig } = htlkgConfig;\n\t\tif (!amplifyConfig) {\n\t\t\tconsole.warn('[htlkg Auth] No Amplify configuration provided');\n\t\t\treturn;\n\t\t}\n\n\t\t// Configure Amplify with SSR support\n\t\tAmplify.configure(amplifyConfig as any, { ssr: true });\n\t\t\n\t\t// Enable server-side auth in the adapter\n\t\tglobalSettings.enableServerSideAuth();\n\t\t\n\t\t// Check if we're on HTTPS (for secure cookies)\n\t\tconst isSSL = typeof window !== 'undefined' \n\t\t\t? window.location.protocol === 'https:' \n\t\t\t: process.env.NODE_ENV === 'production';\n\t\tglobalSettings.setIsSSLOrigin(isSSL);\n\t\t\n\t\tamplifyConfigured = true;\n\t\tconsole.info('[htlkg Auth] Amplify configured for server-side auth');\n\t} catch (error) {\n\t\tconst errorMsg = error instanceof Error ? error.message : 'Unknown error';\n\t\tconsole.error(`[htlkg Auth] Failed to configure Amplify: ${errorMsg}`);\n\t}\n}\n\n/**\n * Auth middleware - retrieves authenticated user and injects into locals\n * \n * This middleware runs on every request and attempts to get the current\n * authenticated user from AWS Amplify. If successful, the user is injected\n * into context.locals.user. If authentication fails, locals.user is set to null.\n * \n * @example\n * // In an Astro page\n * const { user } = Astro.locals;\n * if (user) {\n * console.log('Authenticated user:', user.email);\n * }\n */\nexport const authMiddleware: MiddlewareHandler = async (context, next) => {\n\tconst { locals } = context;\n\n\t// Ensure Amplify is configured before attempting auth\n\tensureAmplifyConfigured();\n\n\ttry {\n\t\tconst user = await getUser(context);\n\t\tlocals.user = user;\n\t} catch (error) {\n\t\t// Set user to null on any authentication error\n\t\tlocals.user = null;\n\n\t\t// Log error without exposing sensitive information\n\t\tif (error instanceof Error) {\n\t\t\t// Filter out sensitive data from error messages\n\t\t\tconst safeMessage = error.message\n\t\t\t\t.replace(/token[=:]\\s*[^\\s,}]+/gi, 'token=***')\n\t\t\t\t.replace(/key[=:]\\s*[^\\s,}]+/gi, 'key=***')\n\t\t\t\t.replace(/secret[=:]\\s*[^\\s,}]+/gi, 'secret=***')\n\t\t\t\t.replace(/password[=:]\\s*[^\\s,}]+/gi, 'password=***');\n\n\t\t\tconsole.error('[htlkg Auth] Authentication failed:', safeMessage);\n\t\t} else {\n\t\t\tconsole.error('[htlkg Auth] Authentication failed: Unknown error');\n\t\t}\n\t}\n\n\treturn next();\n};\n","/**\n * Route guard middleware for htlkg integration\n *\n * This middleware enforces access control based on route configuration:\n * - Public routes: accessible to everyone\n * - Authenticated routes: require any logged-in user\n * - Admin routes: require admin privileges\n * - Brand routes: require brand-specific access or admin privileges\n */\n\nimport type { MiddlewareHandler } from 'astro';\nimport type { RoutePattern } from '../htlkg/config.js';\nimport { htlkgConfig } from 'virtual:htlkg';\n\n/**\n * Helper: Check if pathname matches any of the provided patterns\n */\nfunction matchesPattern(pathname: string, patterns: RoutePattern[]): boolean {\n\treturn patterns.some((pattern) => {\n\t\ttry {\n\t\t\tif (typeof pattern === 'string') {\n\t\t\t\t// Special case: \"/\" only matches exactly\n\t\t\t\tif (pattern === '/') {\n\t\t\t\t\treturn pathname === '/';\n\t\t\t\t}\n\t\t\t\t// String pattern: exact match or starts with (for sub-routes)\n\t\t\t\treturn pathname === pattern || pathname.startsWith(pattern + '/');\n\t\t\t}\n\t\t\t// RegExp pattern: test against pathname\n\t\t\treturn pattern.test(pathname);\n\t\t} catch (error) {\n\t\t\t// Log pattern matching errors but don't block the request\n\t\t\tconsole.error(\n\t\t\t\t'[htlkg] Error matching route pattern:',\n\t\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t\t);\n\t\t\treturn false;\n\t\t}\n\t});\n}\n\n/**\n * Route guard middleware - protects routes based on configuration\n * \n * This middleware runs after authMiddleware and enforces access control\n * based on the route configuration provided to the htlkg integration.\n * \n * @example\n * // In astro.config.mjs\n * htlkg({\n * auth: {\n * publicRoutes: ['/login', '/'],\n * adminRoutes: [/^\\/admin/],\n * brandRoutes: [\n * { pattern: /^\\/brands\\/(\\d+)/, brandIdParam: 1 }\n * ],\n * loginUrl: '/login'\n * }\n * })\n */\nexport const routeGuard: MiddlewareHandler = async (context, next) => {\n\tconst { locals, url, redirect } = context;\n\tconst pathname = url.pathname;\n\n\tconst { routeGuardConfig } = htlkgConfig;\n\tconst {\n\t\tpublicRoutes = [],\n\t\tauthenticatedRoutes = [],\n\t\tadminRoutes = [],\n\t\tbrandRoutes = [],\n\t\tloginUrl = '/login',\n\t} = routeGuardConfig || {};\n\n\ttry {\n\t\t// Public routes - no auth required\n\t\tif (matchesPattern(pathname, publicRoutes)) {\n\t\t\treturn next();\n\t\t}\n\n\t\tconst user = locals.user;\n\n\t\t// Admin routes - require admin role\n\t\tif (matchesPattern(pathname, adminRoutes)) {\n\t\t\tif (!user || !user.isAdmin) {\n\t\t\t\treturn redirect(`${loginUrl}?error=admin_required`);\n\t\t\t}\n\t\t\treturn next();\n\t\t}\n\n\t\t// Brand routes - require brand access or admin\n\t\tfor (const brandRoute of brandRoutes) {\n\t\t\ttry {\n\t\t\t\tconst match = pathname.match(brandRoute.pattern);\n\t\t\t\tif (match) {\n\t\t\t\t\tconst brandId = Number.parseInt(match[brandRoute.brandIdParam], 10);\n\n\t\t\t\t\tif (Number.isNaN(brandId)) {\n\t\t\t\t\t\tconsole.warn(`[htlkg] Invalid brandId extracted from ${pathname}`);\n\t\t\t\t\t\treturn redirect(`${loginUrl}?error=invalid_brand`);\n\t\t\t\t\t}\n\n\t\t\t\t\tif (!user || (!user.isAdmin && !user.brandIds.includes(brandId))) {\n\t\t\t\t\t\treturn redirect(`${loginUrl}?error=access_denied`);\n\t\t\t\t\t}\n\t\t\t\t\treturn next();\n\t\t\t\t}\n\t\t\t} catch (error) {\n\t\t\t\tconsole.error(\n\t\t\t\t\t`[htlkg] Error processing brand route for ${pathname}:`,\n\t\t\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t\t\t);\n\t\t\t\t// Fail-safe: deny access on error\n\t\t\t\treturn redirect(`${loginUrl}?error=access_denied`);\n\t\t\t}\n\t\t}\n\n\t\t// Authenticated routes - require any logged-in user\n\t\tif (matchesPattern(pathname, authenticatedRoutes)) {\n\t\t\tif (!user) {\n\t\t\t\tconst returnUrl = encodeURIComponent(pathname + url.search);\n\t\t\t\treturn redirect(`${loginUrl}?redirect=${returnUrl}`);\n\t\t\t}\n\t\t\treturn next();\n\t\t}\n\n\t\t// Default: allow access\n\t\treturn next();\n\t} catch (error) {\n\t\t// Catch-all error handler for route guard\n\t\tconsole.error(\n\t\t\t'[htlkg] Unexpected error in route guard:',\n\t\t\terror instanceof Error ? error.message : 'Unknown error',\n\t\t);\n\t\t// Fail-safe: deny access and redirect to login\n\t\treturn redirect(`${loginUrl}?error=server_error`);\n\t}\n};\n\n/**\n * Helper functions for programmatic route protection in pages\n * These can be used in Astro pages for more fine-grained control\n */\n\n/**\n * Require authentication for a page\n * Redirects to login if user is not authenticated\n */\nexport async function requireAuth(context: any, loginUrl = '/login') {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\tconst currentUrl = context.url.pathname + context.url.search;\n\t\tconst encodedReturnUrl = encodeURIComponent(currentUrl);\n\t\treturn context.redirect(`${loginUrl}?redirect=${encodedReturnUrl}`);\n\t}\n\treturn user;\n}\n\n/**\n * Require admin access for a page\n * Redirects to login if user is not an admin\n */\nexport async function requireAdminAccess(context: any, loginUrl = '/login') {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\treturn context.redirect(`${loginUrl}?error=not_authenticated`);\n\t}\n\tif (!user.isAdmin) {\n\t\treturn context.redirect(`${loginUrl}?error=admin_required`);\n\t}\n\treturn user;\n}\n\n/**\n * Require brand access for a page\n * Redirects to login if user doesn't have access to the specified brand\n */\nexport async function requireBrandAccess(\n\tcontext: any,\n\tbrandId: number,\n\tloginUrl = '/login'\n) {\n\tconst user = context.locals.user;\n\tif (!user) {\n\t\treturn context.redirect(`${loginUrl}?error=not_authenticated`);\n\t}\n\tif (!user.isAdmin && !user.brandIds.includes(brandId)) {\n\t\treturn context.redirect(`${loginUrl}?error=access_denied`);\n\t}\n\treturn user;\n}\n"],"mappings":"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","names":["htlkgConfig"]}
|
package/dist/runtime-config.js
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
|
+
// src/runtime-config.ts
|
|
1
2
|
import {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
} from "
|
|
3
|
+
htlkgConfig,
|
|
4
|
+
htlkgConfig as htlkgConfig2
|
|
5
|
+
} from "virtual:htlkg";
|
|
5
6
|
export {
|
|
6
|
-
getHtlkgConfig,
|
|
7
|
-
|
|
7
|
+
htlkgConfig2 as getHtlkgConfig,
|
|
8
|
+
htlkgConfig
|
|
8
9
|
};
|
|
9
10
|
//# sourceMappingURL=runtime-config.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/runtime-config.ts"],"sourcesContent":["/**\n * Runtime Configuration for @htlkg/astro\n *\n * @deprecated This module is deprecated. Import from 'virtual:htlkg' instead.\n *\n * @example\n * // OLD (deprecated)\n * import { getHtlkgConfig } from '@htlkg/astro/runtime-config';\n * const config = getHtlkgConfig();\n *\n * // NEW\n * import { htlkgConfig } from 'virtual:htlkg';\n * const { routeGuardConfig, loginPageConfig, amplifyConfig } = htlkgConfig;\n */\n\nexport {\n\thtlkgConfig,\n\thtlkgConfig as getHtlkgConfig,\n} from 'virtual:htlkg';\n"],"mappings":";AAeA;AAAA,EACC;AAAA,EACe,eAAfA;AAAA,OACM;","names":["htlkgConfig"]}
|
package/dist/vue-app-setup.js
CHANGED
|
@@ -1,9 +1,6 @@
|
|
|
1
|
-
import {
|
|
2
|
-
getHtlkgConfig
|
|
3
|
-
} from "./chunk-LP4XHGBI.js";
|
|
4
|
-
|
|
5
1
|
// src/vue-app-setup.ts
|
|
6
2
|
import { Amplify } from "aws-amplify";
|
|
3
|
+
import { htlkgConfig } from "virtual:htlkg";
|
|
7
4
|
function setupVueApp(app) {
|
|
8
5
|
if (import.meta.env.DEV && typeof window !== "undefined") {
|
|
9
6
|
try {
|
|
@@ -15,7 +12,7 @@ function setupVueApp(app) {
|
|
|
15
12
|
}
|
|
16
13
|
}
|
|
17
14
|
try {
|
|
18
|
-
const { amplifyConfig } =
|
|
15
|
+
const { amplifyConfig } = htlkgConfig;
|
|
19
16
|
if (!amplifyConfig) {
|
|
20
17
|
console.warn("[htlkg] No Amplify configuration provided");
|
|
21
18
|
return;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/vue-app-setup.ts"],"sourcesContent":["/**\n * Vue App Setup for htlkg Integration\n
|
|
1
|
+
{"version":3,"sources":["../src/vue-app-setup.ts"],"sourcesContent":["/**\n * Vue App Setup for htlkg Integration\n *\n * This file is automatically loaded by the Vue integration when vueAppSetup is enabled.\n * It configures AWS Amplify for client-side authentication in all Vue components.\n * It also sets up nanostores devtools integration for Vue DevTools.\n */\n\n/// <reference types=\"vite/client\" />\n\nimport type { App } from \"vue\";\nimport type { ResourcesConfig } from \"aws-amplify\";\nimport { Amplify } from \"aws-amplify\";\nimport { htlkgConfig } from \"virtual:htlkg\";\n\n/**\n * Setup function called by Astro's Vue integration\n * Configures Amplify for client-side authentication and nanostores devtools\n */\nexport default function setupVueApp(app: App): void {\n\t// Setup nanostores devtools in development\n\t// The devtools plugin will automatically detect stores used in components\n\tif (import.meta.env.DEV && typeof window !== 'undefined') {\n\t\ttry {\n\t\t\t// Dynamically import devtools plugin (only in browser)\n\t\t\timport('@nanostores/vue/devtools').then(({ devtools }) => {\n\t\t\t\t// Install devtools plugin - it will detect stores automatically\n\t\t\t\tapp.use(devtools, {});\n\t\t\t}).catch(() => {\n\t\t\t\t// Silently ignore - devtools are optional\n\t\t\t});\n\t\t} catch {\n\t\t\t// Silently ignore - devtools are optional\n\t\t}\n\t}\n\n\t// Setup Amplify\n\ttry {\n\t\tconst { amplifyConfig } = htlkgConfig;\n\t\tif (!amplifyConfig) {\n\t\t\tconsole.warn(\"[htlkg] No Amplify configuration provided\");\n\t\t\treturn;\n\t\t}\n\n\t\t// Check if this is a full amplify_outputs.json config\n\t\tif (\n\t\t\t\"auth\" in amplifyConfig ||\n\t\t\t\"data\" in amplifyConfig ||\n\t\t\t\"storage\" in amplifyConfig\n\t\t) {\n\t\t\tAmplify.configure(amplifyConfig as ResourcesConfig, { ssr: true });\n\t\t} else {\n\t\t\t// Legacy individual config properties\n\t\t\tconst { userPoolId, userPoolClientId, region } = amplifyConfig as {\n\t\t\t\tuserPoolId?: string;\n\t\t\t\tuserPoolClientId?: string;\n\t\t\t\tregion?: string;\n\t\t\t};\n\n\t\t\tif (userPoolId && userPoolClientId) {\n\t\t\t\tconst config: ResourcesConfig = {\n\t\t\t\t\tAuth: {\n\t\t\t\t\t\tCognito: {\n\t\t\t\t\t\t\tuserPoolId,\n\t\t\t\t\t\t\tuserPoolClientId,\n\t\t\t\t\t\t\t...(region && { region }),\n\t\t\t\t\t\t},\n\t\t\t\t\t},\n\t\t\t\t};\n\n\t\t\t\tAmplify.configure(config, { ssr: true });\n\t\t\t} else {\n\t\t\t\tconsole.error(\n\t\t\t\t\t\"[htlkg] Missing required Amplify configuration (userPoolId, userPoolClientId)\",\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t} catch (error) {\n\t\tconsole.error(\"[htlkg] Failed to setup Vue app:\", error);\n\t\t// Don't throw - allow app to continue even if Amplify setup fails\n\t}\n}\n"],"mappings":";AAYA,SAAS,eAAe;AACxB,SAAS,mBAAmB;AAMb,SAAR,YAA6B,KAAgB;AAGnD,MAAI,YAAY,IAAI,OAAO,OAAO,WAAW,aAAa;AACzD,QAAI;AAEH,aAAO,0BAA0B,EAAE,KAAK,CAAC,EAAE,SAAS,MAAM;AAEzD,YAAI,IAAI,UAAU,CAAC,CAAC;AAAA,MACrB,CAAC,EAAE,MAAM,MAAM;AAAA,MAEf,CAAC;AAAA,IACF,QAAQ;AAAA,IAER;AAAA,EACD;AAGA,MAAI;AACH,UAAM,EAAE,cAAc,IAAI;AAC1B,QAAI,CAAC,eAAe;AACnB,cAAQ,KAAK,2CAA2C;AACxD;AAAA,IACD;AAGA,QACC,UAAU,iBACV,UAAU,iBACV,aAAa,eACZ;AACD,cAAQ,UAAU,eAAkC,EAAE,KAAK,KAAK,CAAC;AAAA,IAClE,OAAO;AAEN,YAAM,EAAE,YAAY,kBAAkB,OAAO,IAAI;AAMjD,UAAI,cAAc,kBAAkB;AACnC,cAAM,SAA0B;AAAA,UAC/B,MAAM;AAAA,YACL,SAAS;AAAA,cACR;AAAA,cACA;AAAA,cACA,GAAI,UAAU,EAAE,OAAO;AAAA,YACxB;AAAA,UACD;AAAA,QACD;AAEA,gBAAQ,UAAU,QAAQ,EAAE,KAAK,KAAK,CAAC;AAAA,MACxC,OAAO;AACN,gBAAQ;AAAA,UACP;AAAA,QACD;AAAA,MACD;AAAA,IACD;AAAA,EACD,SAAS,OAAO;AACf,YAAQ,MAAM,oCAAoC,KAAK;AAAA,EAExD;AACD;","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@htlkg/astro",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.18",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"exports": {
|
|
6
6
|
".": {
|
|
@@ -23,6 +23,10 @@
|
|
|
23
23
|
"import": "./dist/vue-app-setup.js",
|
|
24
24
|
"types": "./src/vue-app-setup.ts"
|
|
25
25
|
},
|
|
26
|
+
"./runtime-config": {
|
|
27
|
+
"import": "./dist/runtime-config.js",
|
|
28
|
+
"types": "./src/runtime-config.ts"
|
|
29
|
+
},
|
|
26
30
|
"./auth/LoginPage.astro": "./src/auth/LoginPage.astro",
|
|
27
31
|
"./layouts": "./src/layouts/index.ts",
|
|
28
32
|
"./layouts/*.astro": "./src/layouts/*.astro",
|
|
@@ -56,38 +60,30 @@
|
|
|
56
60
|
],
|
|
57
61
|
"dependencies": {
|
|
58
62
|
"@hotelinking/ui": "^16.49.16",
|
|
59
|
-
"@
|
|
60
|
-
"@htlkg/core": "0.0.9"
|
|
61
|
-
},
|
|
62
|
-
"peerDependencies": {
|
|
63
|
-
"@astrojs/tailwind": "^6.0.0",
|
|
63
|
+
"@astrojs/tailwind": "^6.0.2",
|
|
64
64
|
"@astrojs/vue": "^5.0.0",
|
|
65
65
|
"@inox-tools/request-nanostores": "^0.6.0",
|
|
66
66
|
"@nanostores/logger": "^1.0.0",
|
|
67
|
-
"@nanostores/vue": "^
|
|
68
|
-
"@vue/devtools-api": "^
|
|
67
|
+
"@nanostores/vue": "^1.0.1",
|
|
68
|
+
"@vue/devtools-api": "^8.0.5",
|
|
69
|
+
"nanostores": "^1.0.1",
|
|
70
|
+
"tailwindcss": "^3.4.18",
|
|
71
|
+
"vue": "^3.5.22",
|
|
72
|
+
"vite-plugin-vue-devtools": "^7.6.10",
|
|
73
|
+
"@htlkg/components": "0.0.11",
|
|
74
|
+
"@htlkg/core": "0.0.10"
|
|
75
|
+
},
|
|
76
|
+
"peerDependencies": {
|
|
69
77
|
"astro": "^5.0.0",
|
|
70
|
-
"aws-amplify": "^6.0.0"
|
|
71
|
-
"nanostores": "^0.11.0 || ^1.0.0",
|
|
72
|
-
"tailwindcss": "^3.0.0",
|
|
73
|
-
"vue": "^3.5.0"
|
|
78
|
+
"aws-amplify": "^6.0.0"
|
|
74
79
|
},
|
|
75
80
|
"devDependencies": {
|
|
76
|
-
"@astrojs/tailwind": "^6.0.2",
|
|
77
|
-
"@astrojs/vue": "^5.0.0",
|
|
78
|
-
"@inox-tools/request-nanostores": "^0.6.0",
|
|
79
|
-
"@nanostores/logger": "^1.0.0",
|
|
80
|
-
"@nanostores/vue": "^0.10.0",
|
|
81
|
-
"@vue/devtools-api": "^6.6.4",
|
|
82
81
|
"astro": "^5.14.7",
|
|
83
82
|
"aws-amplify": "^6.11.3",
|
|
84
|
-
"nanostores": "^0.11.3",
|
|
85
|
-
"tailwindcss": "^3.4.18",
|
|
86
83
|
"tsup": "^8.0.0",
|
|
87
84
|
"typescript": "^5.9.2",
|
|
88
|
-
"vite
|
|
89
|
-
"vitest": "^3.2.4"
|
|
90
|
-
"vue": "^3.5.22"
|
|
85
|
+
"vite": "^6.0.0",
|
|
86
|
+
"vitest": "^3.2.4"
|
|
91
87
|
},
|
|
92
88
|
"publishConfig": {
|
|
93
89
|
"access": "public"
|
package/src/auth/LoginPage.astro
CHANGED
|
@@ -16,12 +16,14 @@
|
|
|
16
16
|
|
|
17
17
|
import DefaultLayout from "../layouts/DefaultLayout.astro";
|
|
18
18
|
import LoginForm from "./LoginForm.vue";
|
|
19
|
-
import {
|
|
19
|
+
import { htlkgConfig } from "virtual:htlkg";
|
|
20
20
|
|
|
21
21
|
export const prerender = false;
|
|
22
22
|
|
|
23
|
-
// Get login page configuration from
|
|
24
|
-
const { loginPageConfig } =
|
|
23
|
+
// Get login page configuration from virtual module
|
|
24
|
+
const { loginPageConfig } = htlkgConfig;
|
|
25
|
+
|
|
26
|
+
console.log(loginPageConfig);
|
|
25
27
|
|
|
26
28
|
const {
|
|
27
29
|
logo = "https://images.hotelinking.com/ui/WiFiBot.svg",
|
package/src/htlkg/index.ts
CHANGED
|
@@ -2,19 +2,105 @@
|
|
|
2
2
|
* htlkg Astro Integration
|
|
3
3
|
*
|
|
4
4
|
* Supports static, hybrid, and full SSR output modes.
|
|
5
|
+
* Uses a virtual Vite module (virtual:htlkg) to provide configuration
|
|
6
|
+
* to middleware and Vue app setup.
|
|
5
7
|
*/
|
|
6
8
|
|
|
7
9
|
import tailwind from '@astrojs/tailwind';
|
|
8
10
|
import vue from '@astrojs/vue';
|
|
11
|
+
import inoxToolsRequestNanostores from '@inox-tools/request-nanostores';
|
|
9
12
|
import type { AstroIntegration } from 'astro';
|
|
10
|
-
import type {
|
|
11
|
-
import {
|
|
13
|
+
import type { Plugin } from 'vite';
|
|
14
|
+
import type { HtlkgIntegrationOptions, RouteGuardConfig, LoginPageConfig } from './config.js';
|
|
12
15
|
|
|
13
16
|
const DEFAULT_ENV_VARS = [
|
|
14
17
|
'PUBLIC_COGNITO_USER_POOL_ID',
|
|
15
18
|
'PUBLIC_COGNITO_USER_POOL_CLIENT_ID'
|
|
16
19
|
];
|
|
17
20
|
|
|
21
|
+
const VIRTUAL_MODULE_ID = 'virtual:htlkg';
|
|
22
|
+
const RESOLVED_VIRTUAL_MODULE_ID = '\0' + VIRTUAL_MODULE_ID;
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Create Vite plugin that provides the virtual:htlkg module
|
|
26
|
+
*/
|
|
27
|
+
function createVirtualConfigPlugin(config: {
|
|
28
|
+
routeGuard: RouteGuardConfig;
|
|
29
|
+
loginPage: LoginPageConfig | null;
|
|
30
|
+
amplify: Record<string, unknown> | null;
|
|
31
|
+
}): Plugin {
|
|
32
|
+
return {
|
|
33
|
+
name: 'htlkg-virtual-config',
|
|
34
|
+
resolveId(id) {
|
|
35
|
+
if (id === VIRTUAL_MODULE_ID) {
|
|
36
|
+
return RESOLVED_VIRTUAL_MODULE_ID;
|
|
37
|
+
}
|
|
38
|
+
},
|
|
39
|
+
load(id) {
|
|
40
|
+
if (id === RESOLVED_VIRTUAL_MODULE_ID) {
|
|
41
|
+
// Serialize the config to be loaded as a module
|
|
42
|
+
// We need to handle RegExp serialization for route patterns
|
|
43
|
+
const serializedConfig = serializeConfig(config);
|
|
44
|
+
return `export const htlkgConfig = ${serializedConfig};`;
|
|
45
|
+
}
|
|
46
|
+
},
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Serialize config object to JavaScript code
|
|
52
|
+
* Handles RegExp patterns which can't be JSON.stringify'd
|
|
53
|
+
*/
|
|
54
|
+
function serializeConfig(config: {
|
|
55
|
+
routeGuard: RouteGuardConfig;
|
|
56
|
+
loginPage: LoginPageConfig | null;
|
|
57
|
+
amplify: Record<string, unknown> | null;
|
|
58
|
+
}): string {
|
|
59
|
+
const routeGuard = config.routeGuard || {};
|
|
60
|
+
|
|
61
|
+
// Build route guard config with proper RegExp handling
|
|
62
|
+
const routeGuardParts: string[] = [];
|
|
63
|
+
|
|
64
|
+
if (routeGuard.publicRoutes) {
|
|
65
|
+
routeGuardParts.push(`publicRoutes: ${serializePatterns(routeGuard.publicRoutes)}`);
|
|
66
|
+
}
|
|
67
|
+
if (routeGuard.authenticatedRoutes) {
|
|
68
|
+
routeGuardParts.push(`authenticatedRoutes: ${serializePatterns(routeGuard.authenticatedRoutes)}`);
|
|
69
|
+
}
|
|
70
|
+
if (routeGuard.adminRoutes) {
|
|
71
|
+
routeGuardParts.push(`adminRoutes: ${serializePatterns(routeGuard.adminRoutes)}`);
|
|
72
|
+
}
|
|
73
|
+
if (routeGuard.brandRoutes && routeGuard.brandRoutes.length > 0) {
|
|
74
|
+
const brandRoutesStr = routeGuard.brandRoutes.map(br =>
|
|
75
|
+
`{ pattern: ${br.pattern.toString()}, brandIdParam: ${br.brandIdParam} }`
|
|
76
|
+
).join(', ');
|
|
77
|
+
routeGuardParts.push(`brandRoutes: [${brandRoutesStr}]`);
|
|
78
|
+
}
|
|
79
|
+
if (routeGuard.loginUrl) {
|
|
80
|
+
routeGuardParts.push(`loginUrl: ${JSON.stringify(routeGuard.loginUrl)}`);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return `{
|
|
84
|
+
routeGuardConfig: { ${routeGuardParts.join(', ')} },
|
|
85
|
+
loginPageConfig: ${JSON.stringify(config.loginPage)},
|
|
86
|
+
amplifyConfig: ${JSON.stringify(config.amplify)}
|
|
87
|
+
}`;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Serialize an array of route patterns (strings or RegExp)
|
|
92
|
+
*/
|
|
93
|
+
function serializePatterns(patterns: (string | RegExp)[]): string {
|
|
94
|
+
const serialized = patterns.map(p => {
|
|
95
|
+
if (typeof p === 'string') {
|
|
96
|
+
return JSON.stringify(p);
|
|
97
|
+
}
|
|
98
|
+
// RegExp - output as literal
|
|
99
|
+
return p.toString();
|
|
100
|
+
});
|
|
101
|
+
return `[${serialized.join(', ')}]`;
|
|
102
|
+
}
|
|
103
|
+
|
|
18
104
|
export function htlkg(
|
|
19
105
|
options: HtlkgIntegrationOptions = {},
|
|
20
106
|
): AstroIntegration | AstroIntegration[] {
|
|
@@ -52,14 +138,24 @@ export function htlkg(
|
|
|
52
138
|
integrations.push(vue());
|
|
53
139
|
}
|
|
54
140
|
|
|
141
|
+
// Add request nanostores integration (for SSR-safe nanostores)
|
|
142
|
+
integrations.push(inoxToolsRequestNanostores());
|
|
143
|
+
|
|
144
|
+
// Prepare configuration for virtual module
|
|
145
|
+
const htlkgVirtualConfig = {
|
|
146
|
+
routeGuard: auth,
|
|
147
|
+
loginPage: loginPage || null,
|
|
148
|
+
amplify: amplify || null,
|
|
149
|
+
};
|
|
150
|
+
|
|
55
151
|
// Add the main htlkg integration
|
|
56
152
|
integrations.push({
|
|
57
153
|
name: '@htlkg/astro',
|
|
58
154
|
hooks: {
|
|
59
155
|
'astro:config:setup': async ({ updateConfig, addMiddleware, injectRoute, logger }) => {
|
|
60
156
|
try {
|
|
61
|
-
logger.info(useFullVueSetup
|
|
62
|
-
? 'Vue configured with Amplify app setup'
|
|
157
|
+
logger.info(useFullVueSetup
|
|
158
|
+
? 'Vue configured with Amplify app setup'
|
|
63
159
|
: 'Vue configured (basic mode)');
|
|
64
160
|
|
|
65
161
|
if (amplify) {
|
|
@@ -74,25 +170,32 @@ export function htlkg(
|
|
|
74
170
|
}
|
|
75
171
|
}
|
|
76
172
|
|
|
77
|
-
//
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
amplify: amplify || null,
|
|
82
|
-
});
|
|
173
|
+
// Create virtual module plugin for config
|
|
174
|
+
const vitePlugins: Plugin[] = [
|
|
175
|
+
createVirtualConfigPlugin(htlkgVirtualConfig),
|
|
176
|
+
];
|
|
83
177
|
|
|
84
|
-
// Only load Vue DevTools during development
|
|
178
|
+
// Only load Vue DevTools during development
|
|
85
179
|
const isDev = process.env.NODE_ENV !== 'production';
|
|
86
180
|
if (isDev) {
|
|
87
181
|
try {
|
|
88
182
|
const vueDevTools = (await import('vite-plugin-vue-devtools')).default;
|
|
89
|
-
|
|
183
|
+
vitePlugins.push(vueDevTools());
|
|
90
184
|
logger.info('Vue DevTools enabled');
|
|
91
185
|
} catch {
|
|
92
186
|
// DevTools not available, skip silently
|
|
93
187
|
}
|
|
94
188
|
}
|
|
95
189
|
|
|
190
|
+
// Update Vite config with plugins
|
|
191
|
+
updateConfig({
|
|
192
|
+
vite: {
|
|
193
|
+
plugins: vitePlugins
|
|
194
|
+
}
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
logger.info('virtual:htlkg module configured');
|
|
198
|
+
|
|
96
199
|
// Inject middleware (only for full setup)
|
|
97
200
|
if (useFullVueSetup) {
|
|
98
201
|
addMiddleware({ entrypoint: '@htlkg/astro/middleware', order: 'pre' });
|
|
@@ -122,6 +225,15 @@ export function htlkg(
|
|
|
122
225
|
filename: 'htlkg.d.ts',
|
|
123
226
|
content: `
|
|
124
227
|
import type { AuthUser } from '@htlkg/core/types';
|
|
228
|
+
import type { RouteGuardConfig, LoginPageConfig } from '@htlkg/astro/htlkg/config';
|
|
229
|
+
|
|
230
|
+
declare module 'virtual:htlkg' {
|
|
231
|
+
export const htlkgConfig: {
|
|
232
|
+
routeGuardConfig: RouteGuardConfig | null;
|
|
233
|
+
loginPageConfig: LoginPageConfig | null;
|
|
234
|
+
amplifyConfig: Record<string, unknown> | null;
|
|
235
|
+
};
|
|
236
|
+
}
|
|
125
237
|
|
|
126
238
|
declare global {
|
|
127
239
|
namespace App {
|
package/src/middleware/auth.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Authentication middleware for htlkg integration
|
|
3
|
-
*
|
|
3
|
+
*
|
|
4
4
|
* This middleware:
|
|
5
5
|
* - Configures AWS Amplify for server-side auth
|
|
6
6
|
* - Retrieves the authenticated user from AWS Amplify
|
|
@@ -12,7 +12,7 @@ import type { MiddlewareHandler } from 'astro';
|
|
|
12
12
|
import { Amplify } from 'aws-amplify';
|
|
13
13
|
import { getUser } from '@htlkg/core/auth';
|
|
14
14
|
import { globalSettings } from '@htlkg/core/amplify-astro-adapter';
|
|
15
|
-
import {
|
|
15
|
+
import { htlkgConfig } from 'virtual:htlkg';
|
|
16
16
|
|
|
17
17
|
// Track if Amplify has been configured
|
|
18
18
|
let amplifyConfigured = false;
|
|
@@ -24,7 +24,7 @@ function ensureAmplifyConfigured(): void {
|
|
|
24
24
|
if (amplifyConfigured) return;
|
|
25
25
|
|
|
26
26
|
try {
|
|
27
|
-
const { amplifyConfig } =
|
|
27
|
+
const { amplifyConfig } = htlkgConfig;
|
|
28
28
|
if (!amplifyConfig) {
|
|
29
29
|
console.warn('[htlkg Auth] No Amplify configuration provided');
|
|
30
30
|
return;
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Route guard middleware for htlkg integration
|
|
3
|
-
*
|
|
3
|
+
*
|
|
4
4
|
* This middleware enforces access control based on route configuration:
|
|
5
5
|
* - Public routes: accessible to everyone
|
|
6
6
|
* - Authenticated routes: require any logged-in user
|
|
@@ -9,8 +9,8 @@
|
|
|
9
9
|
*/
|
|
10
10
|
|
|
11
11
|
import type { MiddlewareHandler } from 'astro';
|
|
12
|
-
import type { RoutePattern
|
|
13
|
-
import {
|
|
12
|
+
import type { RoutePattern } from '../htlkg/config.js';
|
|
13
|
+
import { htlkgConfig } from 'virtual:htlkg';
|
|
14
14
|
|
|
15
15
|
/**
|
|
16
16
|
* Helper: Check if pathname matches any of the provided patterns
|
|
@@ -62,7 +62,7 @@ export const routeGuard: MiddlewareHandler = async (context, next) => {
|
|
|
62
62
|
const { locals, url, redirect } = context;
|
|
63
63
|
const pathname = url.pathname;
|
|
64
64
|
|
|
65
|
-
const { routeGuardConfig } =
|
|
65
|
+
const { routeGuardConfig } = htlkgConfig;
|
|
66
66
|
const {
|
|
67
67
|
publicRoutes = [],
|
|
68
68
|
authenticatedRoutes = [],
|
package/src/runtime-config.ts
CHANGED
|
@@ -1,48 +1,19 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Runtime Configuration for @htlkg/astro
|
|
3
3
|
*
|
|
4
|
-
* This module
|
|
5
|
-
* and makes it available to middleware and pages at request time.
|
|
4
|
+
* @deprecated This module is deprecated. Import from 'virtual:htlkg' instead.
|
|
6
5
|
*
|
|
7
|
-
*
|
|
8
|
-
*
|
|
6
|
+
* @example
|
|
7
|
+
* // OLD (deprecated)
|
|
8
|
+
* import { getHtlkgConfig } from '@htlkg/astro/runtime-config';
|
|
9
|
+
* const config = getHtlkgConfig();
|
|
9
10
|
*
|
|
10
|
-
*
|
|
11
|
+
* // NEW
|
|
12
|
+
* import { htlkgConfig } from 'virtual:htlkg';
|
|
13
|
+
* const { routeGuardConfig, loginPageConfig, amplifyConfig } = htlkgConfig;
|
|
11
14
|
*/
|
|
12
15
|
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
let _loginPageConfig: LoginPageConfig | null = null;
|
|
18
|
-
let _amplifyConfig: Record<string, unknown> | null = null;
|
|
19
|
-
|
|
20
|
-
/**
|
|
21
|
-
* Set the runtime configuration
|
|
22
|
-
* Called by the htlkg integration during astro:config:setup
|
|
23
|
-
*/
|
|
24
|
-
export function setHtlkgConfig(config: {
|
|
25
|
-
routeGuard: RouteGuardConfig;
|
|
26
|
-
loginPage: LoginPageConfig | null;
|
|
27
|
-
amplify: Record<string, unknown> | null;
|
|
28
|
-
}): void {
|
|
29
|
-
_routeGuardConfig = config.routeGuard;
|
|
30
|
-
_loginPageConfig = config.loginPage;
|
|
31
|
-
_amplifyConfig = config.amplify;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* Get the runtime configuration
|
|
36
|
-
* Called by middleware and pages at request time
|
|
37
|
-
*/
|
|
38
|
-
export function getHtlkgConfig(): {
|
|
39
|
-
routeGuardConfig: RouteGuardConfig | null;
|
|
40
|
-
loginPageConfig: LoginPageConfig | null;
|
|
41
|
-
amplifyConfig: Record<string, unknown> | null;
|
|
42
|
-
} {
|
|
43
|
-
return {
|
|
44
|
-
routeGuardConfig: _routeGuardConfig,
|
|
45
|
-
loginPageConfig: _loginPageConfig,
|
|
46
|
-
amplifyConfig: _amplifyConfig,
|
|
47
|
-
};
|
|
48
|
-
}
|
|
16
|
+
export {
|
|
17
|
+
htlkgConfig,
|
|
18
|
+
htlkgConfig as getHtlkgConfig,
|
|
19
|
+
} from 'virtual:htlkg';
|
package/src/vue-app-setup.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Vue App Setup for htlkg Integration
|
|
3
|
-
*
|
|
3
|
+
*
|
|
4
4
|
* This file is automatically loaded by the Vue integration when vueAppSetup is enabled.
|
|
5
5
|
* It configures AWS Amplify for client-side authentication in all Vue components.
|
|
6
6
|
* It also sets up nanostores devtools integration for Vue DevTools.
|
|
@@ -11,7 +11,7 @@
|
|
|
11
11
|
import type { App } from "vue";
|
|
12
12
|
import type { ResourcesConfig } from "aws-amplify";
|
|
13
13
|
import { Amplify } from "aws-amplify";
|
|
14
|
-
import {
|
|
14
|
+
import { htlkgConfig } from "virtual:htlkg";
|
|
15
15
|
|
|
16
16
|
/**
|
|
17
17
|
* Setup function called by Astro's Vue integration
|
|
@@ -33,10 +33,10 @@ export default function setupVueApp(app: App): void {
|
|
|
33
33
|
// Silently ignore - devtools are optional
|
|
34
34
|
}
|
|
35
35
|
}
|
|
36
|
-
|
|
36
|
+
|
|
37
37
|
// Setup Amplify
|
|
38
38
|
try {
|
|
39
|
-
const { amplifyConfig } =
|
|
39
|
+
const { amplifyConfig } = htlkgConfig;
|
|
40
40
|
if (!amplifyConfig) {
|
|
41
41
|
console.warn("[htlkg] No Amplify configuration provided");
|
|
42
42
|
return;
|
package/dist/chunk-LP4XHGBI.js
DELETED
|
@@ -1,22 +0,0 @@
|
|
|
1
|
-
// src/runtime-config.ts
|
|
2
|
-
var _routeGuardConfig = null;
|
|
3
|
-
var _loginPageConfig = null;
|
|
4
|
-
var _amplifyConfig = null;
|
|
5
|
-
function setHtlkgConfig(config) {
|
|
6
|
-
_routeGuardConfig = config.routeGuard;
|
|
7
|
-
_loginPageConfig = config.loginPage;
|
|
8
|
-
_amplifyConfig = config.amplify;
|
|
9
|
-
}
|
|
10
|
-
function getHtlkgConfig() {
|
|
11
|
-
return {
|
|
12
|
-
routeGuardConfig: _routeGuardConfig,
|
|
13
|
-
loginPageConfig: _loginPageConfig,
|
|
14
|
-
amplifyConfig: _amplifyConfig
|
|
15
|
-
};
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
export {
|
|
19
|
-
setHtlkgConfig,
|
|
20
|
-
getHtlkgConfig
|
|
21
|
-
};
|
|
22
|
-
//# sourceMappingURL=chunk-LP4XHGBI.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/runtime-config.ts"],"sourcesContent":["/**\n * Runtime Configuration for @htlkg/astro\n *\n * This module stores configuration set during Astro's config setup phase\n * and makes it available to middleware and pages at request time.\n *\n * This replaces the virtual:htlkg-config virtual module approach,\n * which required bundling to resolve virtual imports.\n *\n * @internal This module is internal to @htlkg/astro\n */\n\nimport type { RouteGuardConfig, LoginPageConfig } from './htlkg/config.js';\n\n// Module-level storage (persists across requests in the same Node.js process)\nlet _routeGuardConfig: RouteGuardConfig | null = null;\nlet _loginPageConfig: LoginPageConfig | null = null;\nlet _amplifyConfig: Record<string, unknown> | null = null;\n\n/**\n * Set the runtime configuration\n * Called by the htlkg integration during astro:config:setup\n */\nexport function setHtlkgConfig(config: {\n\trouteGuard: RouteGuardConfig;\n\tloginPage: LoginPageConfig | null;\n\tamplify: Record<string, unknown> | null;\n}): void {\n\t_routeGuardConfig = config.routeGuard;\n\t_loginPageConfig = config.loginPage;\n\t_amplifyConfig = config.amplify;\n}\n\n/**\n * Get the runtime configuration\n * Called by middleware and pages at request time\n */\nexport function getHtlkgConfig(): {\n\trouteGuardConfig: RouteGuardConfig | null;\n\tloginPageConfig: LoginPageConfig | null;\n\tamplifyConfig: Record<string, unknown> | null;\n} {\n\treturn {\n\t\trouteGuardConfig: _routeGuardConfig,\n\t\tloginPageConfig: _loginPageConfig,\n\t\tamplifyConfig: _amplifyConfig,\n\t};\n}\n"],"mappings":";AAeA,IAAI,oBAA6C;AACjD,IAAI,mBAA2C;AAC/C,IAAI,iBAAiD;AAM9C,SAAS,eAAe,QAItB;AACR,sBAAoB,OAAO;AAC3B,qBAAmB,OAAO;AAC1B,mBAAiB,OAAO;AACzB;AAMO,SAAS,iBAId;AACD,SAAO;AAAA,IACN,kBAAkB;AAAA,IAClB,iBAAiB;AAAA,IACjB,eAAe;AAAA,EAChB;AACD;","names":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/htlkg/index.ts"],"sourcesContent":["/**\n * htlkg Astro Integration\n *\n * Supports static, hybrid, and full SSR output modes.\n */\n\nimport tailwind from '@astrojs/tailwind';\nimport vue from '@astrojs/vue';\nimport type { AstroIntegration } from 'astro';\nimport type { HtlkgIntegrationOptions } from './config.js';\nimport { setHtlkgConfig } from '../runtime-config.js';\n\nconst DEFAULT_ENV_VARS = [\n\t'PUBLIC_COGNITO_USER_POOL_ID',\n\t'PUBLIC_COGNITO_USER_POOL_CLIENT_ID'\n];\n\nexport function htlkg(\n\toptions: HtlkgIntegrationOptions = {},\n): AstroIntegration | AstroIntegration[] {\n\tconst {\n\t\tauth = {},\n\t\tloginPage = { path: '/login', title: 'Sign In', redirectUrl: '/admin' },\n\t\tvalidateEnv = true,\n\t\trequiredEnvVars = DEFAULT_ENV_VARS,\n\t\ttailwind: tailwindOptions,\n\t\tamplify,\n\t\tvueAppSetup = 'auto',\n\t} = options;\n\n\tconst integrations: AstroIntegration[] = [];\n\n\t// Add Tailwind integration (enabled by default)\n\tif (tailwindOptions !== false) {\n\t\tconst tailwindConfig =\n\t\t\ttypeof tailwindOptions === 'object' ? tailwindOptions : undefined;\n\t\tintegrations.push(\n\t\t\ttailwind(tailwindConfig as Parameters<typeof tailwind>[0]),\n\t\t);\n\t}\n\n\t// Determine Vue setup mode:\n\t// - 'full': Use Amplify app entrypoint (requires SSR)\n\t// - 'basic': Basic Vue without app entrypoint (works with static)\n\t// - 'auto': Default to 'basic' for compatibility with static builds\n\tconst useFullVueSetup = vueAppSetup === 'full';\n\n\t// Add Vue integration\n\tif (useFullVueSetup) {\n\t\tintegrations.push(vue({ appEntrypoint: '@htlkg/astro/vue-app-setup' }));\n\t} else {\n\t\tintegrations.push(vue());\n\t}\n\n\t// Add the main htlkg integration\n\tintegrations.push({\n\t\tname: '@htlkg/astro',\n\t\thooks: {\n\t\t\t'astro:config:setup': async ({ updateConfig, addMiddleware, injectRoute, logger }) => {\n\t\t\t\ttry {\n\t\t\t\t\tlogger.info(useFullVueSetup \n\t\t\t\t\t\t? 'Vue configured with Amplify app setup' \n\t\t\t\t\t\t: 'Vue configured (basic mode)');\n\n\t\t\t\t\tif (amplify) {\n\t\t\t\t\t\tlogger.info('Amplify configuration provided');\n\t\t\t\t\t}\n\n\t\t\t\t\t// Validate env vars (only for full setup)\n\t\t\t\t\tif (validateEnv && !amplify && useFullVueSetup) {\n\t\t\t\t\t\tconst missing = requiredEnvVars.filter(v => !process.env[v]);\n\t\t\t\t\t\tif (missing.length > 0) {\n\t\t\t\t\t\t\tlogger.warn(`Missing env vars: ${missing.join(', ')}`);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Set runtime configuration (replaces virtual module approach)\n\t\t\t\t\tsetHtlkgConfig({\n\t\t\t\t\t\trouteGuard: auth,\n\t\t\t\t\t\tloginPage: loginPage || null,\n\t\t\t\t\t\tamplify: amplify || null,\n\t\t\t\t\t});\n\n\t\t\t\t\t// Only load Vue DevTools during development (dynamic import to avoid bundling in production)\n\t\t\t\t\tconst isDev = process.env.NODE_ENV !== 'production';\n\t\t\t\t\tif (isDev) {\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tconst vueDevTools = (await import('vite-plugin-vue-devtools')).default;\n\t\t\t\t\t\t\tupdateConfig({ vite: { plugins: [vueDevTools()] } });\n\t\t\t\t\t\t\tlogger.info('Vue DevTools enabled');\n\t\t\t\t\t\t} catch {\n\t\t\t\t\t\t\t// DevTools not available, skip silently\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Inject middleware (only for full setup)\n\t\t\t\t\tif (useFullVueSetup) {\n\t\t\t\t\t\taddMiddleware({ entrypoint: '@htlkg/astro/middleware', order: 'pre' });\n\t\t\t\t\t\tlogger.info('Authentication middleware injected');\n\t\t\t\t\t}\n\n\t\t\t\t\t// Inject login page (only for full setup)\n\t\t\t\t\tif (loginPage !== false && useFullVueSetup) {\n\t\t\t\t\t\tconst loginPath = loginPage.path || '/login';\n\t\t\t\t\t\tinjectRoute({\n\t\t\t\t\t\t\tpattern: loginPath,\n\t\t\t\t\t\t\tentrypoint: '@htlkg/astro/auth/LoginPage.astro',\n\t\t\t\t\t\t\tprerender: false,\n\t\t\t\t\t\t});\n\t\t\t\t\t\tlogger.info(`Login page injected at ${loginPath}`);\n\t\t\t\t\t}\n\n\t\t\t\t\tlogger.info('htlkg integration configured');\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconst msg = error instanceof Error ? error.message : 'Unknown error';\n\t\t\t\t\tlogger.error(`htlkg configuration failed: ${msg}`);\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t},\n\t\t\t'astro:config:done': ({ injectTypes }) => {\n\t\t\t\tinjectTypes({\n\t\t\t\t\tfilename: 'htlkg.d.ts',\n\t\t\t\t\tcontent: `\nimport type { AuthUser } from '@htlkg/core/types';\n\ndeclare global {\n namespace App {\n interface Locals {\n user: AuthUser | null;\n }\n }\n}\n\nexport {};\n`,\n\t\t\t\t});\n\t\t\t},\n\t\t},\n\t});\n\n\treturn integrations;\n}\n"],"mappings":";;;;;AAMA,OAAO,cAAc;AACrB,OAAO,SAAS;AAKhB,IAAM,mBAAmB;AAAA,EACxB;AAAA,EACA;AACD;AAEO,SAAS,MACf,UAAmC,CAAC,GACI;AACxC,QAAM;AAAA,IACL,OAAO,CAAC;AAAA,IACR,YAAY,EAAE,MAAM,UAAU,OAAO,WAAW,aAAa,SAAS;AAAA,IACtE,cAAc;AAAA,IACd,kBAAkB;AAAA,IAClB,UAAU;AAAA,IACV;AAAA,IACA,cAAc;AAAA,EACf,IAAI;AAEJ,QAAM,eAAmC,CAAC;AAG1C,MAAI,oBAAoB,OAAO;AAC9B,UAAM,iBACL,OAAO,oBAAoB,WAAW,kBAAkB;AACzD,iBAAa;AAAA,MACZ,SAAS,cAAgD;AAAA,IAC1D;AAAA,EACD;AAMA,QAAM,kBAAkB,gBAAgB;AAGxC,MAAI,iBAAiB;AACpB,iBAAa,KAAK,IAAI,EAAE,eAAe,6BAA6B,CAAC,CAAC;AAAA,EACvE,OAAO;AACN,iBAAa,KAAK,IAAI,CAAC;AAAA,EACxB;AAGA,eAAa,KAAK;AAAA,IACjB,MAAM;AAAA,IACN,OAAO;AAAA,MACN,sBAAsB,OAAO,EAAE,cAAc,eAAe,aAAa,OAAO,MAAM;AACrF,YAAI;AACH,iBAAO,KAAK,kBACT,0CACA,6BAA6B;AAEhC,cAAI,SAAS;AACZ,mBAAO,KAAK,gCAAgC;AAAA,UAC7C;AAGA,cAAI,eAAe,CAAC,WAAW,iBAAiB;AAC/C,kBAAM,UAAU,gBAAgB,OAAO,OAAK,CAAC,QAAQ,IAAI,CAAC,CAAC;AAC3D,gBAAI,QAAQ,SAAS,GAAG;AACvB,qBAAO,KAAK,qBAAqB,QAAQ,KAAK,IAAI,CAAC,EAAE;AAAA,YACtD;AAAA,UACD;AAGA,yBAAe;AAAA,YACd,YAAY;AAAA,YACZ,WAAW,aAAa;AAAA,YACxB,SAAS,WAAW;AAAA,UACrB,CAAC;AAGD,gBAAM,QAAQ,QAAQ,IAAI,aAAa;AACvC,cAAI,OAAO;AACV,gBAAI;AACH,oBAAM,eAAe,MAAM,OAAO,0BAA0B,GAAG;AAC/D,2BAAa,EAAE,MAAM,EAAE,SAAS,CAAC,YAAY,CAAC,EAAE,EAAE,CAAC;AACnD,qBAAO,KAAK,sBAAsB;AAAA,YACnC,QAAQ;AAAA,YAER;AAAA,UACD;AAGA,cAAI,iBAAiB;AACpB,0BAAc,EAAE,YAAY,2BAA2B,OAAO,MAAM,CAAC;AACrE,mBAAO,KAAK,oCAAoC;AAAA,UACjD;AAGA,cAAI,cAAc,SAAS,iBAAiB;AAC3C,kBAAM,YAAY,UAAU,QAAQ;AACpC,wBAAY;AAAA,cACX,SAAS;AAAA,cACT,YAAY;AAAA,cACZ,WAAW;AAAA,YACZ,CAAC;AACD,mBAAO,KAAK,0BAA0B,SAAS,EAAE;AAAA,UAClD;AAEA,iBAAO,KAAK,8BAA8B;AAAA,QAC3C,SAAS,OAAO;AACf,gBAAM,MAAM,iBAAiB,QAAQ,MAAM,UAAU;AACrD,iBAAO,MAAM,+BAA+B,GAAG,EAAE;AACjD,gBAAM;AAAA,QACP;AAAA,MACD;AAAA,MACA,qBAAqB,CAAC,EAAE,YAAY,MAAM;AACzC,oBAAY;AAAA,UACX,UAAU;AAAA,UACV,SAAS;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,QAaV,CAAC;AAAA,MACF;AAAA,IACD;AAAA,EACD,CAAC;AAED,SAAO;AACR;","names":[]}
|