Merge branch 'cli-improvements-1' with @reviq/utils password hashing
- Use executeBootstrap helper from @reviq/db for CLI bootstrap - Update @reviq/db to use @reviq/utils for PBKDF2-SHA256 password hashing (Cloudflare Workers compatible) - Keep @scure/base for base58 token encoding - Remove redundant password.ts from @reviq/db (import directly from @reviq/utils) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
@@ -21,6 +21,7 @@
|
|||||||
"@reviq/db": "workspace:*",
|
"@reviq/db": "workspace:*",
|
||||||
"@reviq/db-schema": "workspace:*",
|
"@reviq/db-schema": "workspace:*",
|
||||||
"@reviq/utils": "workspace:*",
|
"@reviq/utils": "workspace:*",
|
||||||
|
"@scure/base": "^2.0.0",
|
||||||
"@simplewebauthn/server": "^13.2.2",
|
"@simplewebauthn/server": "^13.2.2",
|
||||||
"@simplewebauthn/types": "^12.0.0",
|
"@simplewebauthn/types": "^12.0.0",
|
||||||
"kysely": "^0.28.2",
|
"kysely": "^0.28.2",
|
||||||
|
|||||||
@@ -61,6 +61,7 @@ function createAuthenticatedContext(
|
|||||||
userId: number,
|
userId: number,
|
||||||
email: string,
|
email: string,
|
||||||
): AuthenticatedContext {
|
): AuthenticatedContext {
|
||||||
|
const now = new Date();
|
||||||
return {
|
return {
|
||||||
...createAPIContext(),
|
...createAPIContext(),
|
||||||
user: {
|
user: {
|
||||||
@@ -73,7 +74,13 @@ function createAuthenticatedContext(
|
|||||||
session: {
|
session: {
|
||||||
id: "1",
|
id: "1",
|
||||||
trustedMode: false,
|
trustedMode: false,
|
||||||
createdAt: new Date(),
|
createdAt: now,
|
||||||
|
},
|
||||||
|
auth: {
|
||||||
|
method: "session",
|
||||||
|
sessionId: "1",
|
||||||
|
expiresAt: new Date(now.getTime() + 24 * 60 * 60 * 1000),
|
||||||
|
createdAt: now,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -44,6 +44,33 @@ export interface Session {
|
|||||||
createdAt: Date;
|
createdAt: Date;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* API token authentication info
|
||||||
|
*/
|
||||||
|
export interface ApiTokenAuth {
|
||||||
|
method: "api_token";
|
||||||
|
tokenId: string;
|
||||||
|
tokenName: string;
|
||||||
|
expiresAt: Date;
|
||||||
|
lastUsedAt: Date | null;
|
||||||
|
createdAt: Date;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Session authentication info
|
||||||
|
*/
|
||||||
|
export interface SessionAuth {
|
||||||
|
method: "session";
|
||||||
|
sessionId: string;
|
||||||
|
expiresAt: Date;
|
||||||
|
createdAt: Date;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Union type for authentication method info
|
||||||
|
*/
|
||||||
|
export type AuthInfo = ApiTokenAuth | SessionAuth;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Authenticated API context for protected handlers
|
* Authenticated API context for protected handlers
|
||||||
*/
|
*/
|
||||||
@@ -52,6 +79,8 @@ export interface AuthenticatedContext extends APIContext {
|
|||||||
user: SessionUser;
|
user: SessionUser;
|
||||||
/** Current session */
|
/** Current session */
|
||||||
session: Session;
|
session: Session;
|
||||||
|
/** Authentication method and details */
|
||||||
|
auth: AuthInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
import type {
|
import type {
|
||||||
APIContext,
|
APIContext,
|
||||||
AuthenticatedContext,
|
AuthenticatedContext,
|
||||||
|
AuthInfo,
|
||||||
Session,
|
Session,
|
||||||
SessionUser,
|
SessionUser,
|
||||||
} from "../context.js";
|
} from "../context.js";
|
||||||
@@ -111,24 +112,49 @@ export const createAuthMiddleware = () => {
|
|||||||
isSuperuser: user.is_superuser,
|
isSuperuser: user.is_superuser,
|
||||||
};
|
};
|
||||||
|
|
||||||
const sessionInfo: Session = session
|
// Build session and auth info based on authentication method
|
||||||
? {
|
let sessionInfo: Session;
|
||||||
|
let authInfo: AuthInfo;
|
||||||
|
|
||||||
|
if (session) {
|
||||||
|
sessionInfo = {
|
||||||
id: session.id,
|
id: session.id,
|
||||||
trustedMode: session.trusted_mode,
|
trustedMode: session.trusted_mode,
|
||||||
createdAt: session.created_at,
|
createdAt: session.created_at,
|
||||||
}
|
};
|
||||||
: {
|
authInfo = {
|
||||||
|
method: "session",
|
||||||
|
sessionId: session.id,
|
||||||
|
expiresAt: session.expires_at,
|
||||||
|
createdAt: session.created_at,
|
||||||
|
};
|
||||||
|
} else if (apiToken) {
|
||||||
|
sessionInfo = {
|
||||||
// For API token auth, create a synthetic session object
|
// For API token auth, create a synthetic session object
|
||||||
// We know apiToken exists because userId came from it
|
|
||||||
id: "0",
|
id: "0",
|
||||||
trustedMode: true,
|
trustedMode: true,
|
||||||
createdAt: apiToken?.created_at ?? new Date(),
|
createdAt: apiToken.created_at,
|
||||||
};
|
};
|
||||||
|
authInfo = {
|
||||||
|
method: "api_token",
|
||||||
|
tokenId: apiToken.id,
|
||||||
|
tokenName: apiToken.name,
|
||||||
|
expiresAt: apiToken.expires_at,
|
||||||
|
lastUsedAt: apiToken.last_used_at,
|
||||||
|
createdAt: apiToken.created_at,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// This should never happen since we checked userId above
|
||||||
|
throw new ORPCError("UNAUTHORIZED", {
|
||||||
|
message: "Invalid authentication state",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return next({
|
return next({
|
||||||
context: {
|
context: {
|
||||||
user: sessionUser,
|
user: sessionUser,
|
||||||
session: sessionInfo,
|
session: sessionInfo,
|
||||||
|
auth: authInfo,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
import type {
|
import type {
|
||||||
APIContext,
|
APIContext,
|
||||||
AuthenticatedContext,
|
AuthenticatedContext,
|
||||||
|
AuthInfo,
|
||||||
LoginRequestContext,
|
LoginRequestContext,
|
||||||
Session,
|
Session,
|
||||||
SessionUser,
|
SessionUser,
|
||||||
@@ -109,23 +110,49 @@ export const authMiddleware = os.middleware(async ({ context, next }) => {
|
|||||||
isSuperuser: user.is_superuser,
|
isSuperuser: user.is_superuser,
|
||||||
};
|
};
|
||||||
|
|
||||||
const sessionInfo: Session = session
|
// Build session and auth info based on authentication method
|
||||||
? {
|
let sessionInfo: Session;
|
||||||
|
let authInfo: AuthInfo;
|
||||||
|
|
||||||
|
if (session) {
|
||||||
|
sessionInfo = {
|
||||||
id: session.id,
|
id: session.id,
|
||||||
trustedMode: session.trusted_mode,
|
trustedMode: session.trusted_mode,
|
||||||
createdAt: session.created_at,
|
createdAt: session.created_at,
|
||||||
}
|
};
|
||||||
: {
|
authInfo = {
|
||||||
|
method: "session",
|
||||||
|
sessionId: session.id,
|
||||||
|
expiresAt: session.expires_at,
|
||||||
|
createdAt: session.created_at,
|
||||||
|
};
|
||||||
|
} else if (apiToken) {
|
||||||
|
sessionInfo = {
|
||||||
// For API token auth, create a synthetic session object
|
// For API token auth, create a synthetic session object
|
||||||
id: "0",
|
id: "0",
|
||||||
trustedMode: true,
|
trustedMode: true,
|
||||||
createdAt: apiToken?.created_at ?? new Date(),
|
createdAt: apiToken.created_at,
|
||||||
};
|
};
|
||||||
|
authInfo = {
|
||||||
|
method: "api_token",
|
||||||
|
tokenId: apiToken.id,
|
||||||
|
tokenName: apiToken.name,
|
||||||
|
expiresAt: apiToken.expires_at,
|
||||||
|
lastUsedAt: apiToken.last_used_at,
|
||||||
|
createdAt: apiToken.created_at,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// This should never happen since we checked userId above
|
||||||
|
throw new ORPCError("UNAUTHORIZED", {
|
||||||
|
message: "Invalid authentication state",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return next({
|
return next({
|
||||||
context: {
|
context: {
|
||||||
user: sessionUser,
|
user: sessionUser,
|
||||||
session: sessionInfo,
|
session: sessionInfo,
|
||||||
|
auth: authInfo,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -187,6 +187,40 @@ const meGet = os.me.get.use(authMiddleware).handler(async ({ context }) => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const meAuthStatus = os.me.authStatus
|
||||||
|
.use(authMiddleware)
|
||||||
|
.handler(async ({ context }) => {
|
||||||
|
const user = await context.db
|
||||||
|
.selectFrom("users")
|
||||||
|
.select([
|
||||||
|
"id",
|
||||||
|
"email",
|
||||||
|
"display_name",
|
||||||
|
"full_name",
|
||||||
|
"phone_number",
|
||||||
|
"avatar_url",
|
||||||
|
"email_verified_at",
|
||||||
|
"is_superuser",
|
||||||
|
])
|
||||||
|
.where("id", "=", context.user.id)
|
||||||
|
.executeTakeFirstOrThrow();
|
||||||
|
|
||||||
|
return {
|
||||||
|
user: {
|
||||||
|
id: user.id,
|
||||||
|
email: user.email,
|
||||||
|
displayName: user.display_name,
|
||||||
|
fullName: user.full_name,
|
||||||
|
phoneNumber: user.phone_number,
|
||||||
|
avatarUrl: user.avatar_url,
|
||||||
|
emailVerified: user.email_verified_at !== null,
|
||||||
|
needsSetup: user.display_name === null,
|
||||||
|
isSuperuser: user.is_superuser,
|
||||||
|
},
|
||||||
|
auth: context.auth,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
const setupProfile = os.me.setupProfile
|
const setupProfile = os.me.setupProfile
|
||||||
.use(authMiddleware)
|
.use(authMiddleware)
|
||||||
.handler(async ({ input, context }) => {
|
.handler(async ({ input, context }) => {
|
||||||
@@ -238,6 +272,7 @@ export const router = os.router({
|
|||||||
},
|
},
|
||||||
me: {
|
me: {
|
||||||
get: meGet,
|
get: meGet,
|
||||||
|
authStatus: meAuthStatus,
|
||||||
setupProfile,
|
setupProfile,
|
||||||
updateProfile,
|
updateProfile,
|
||||||
delete: meDelete,
|
delete: meDelete,
|
||||||
|
|||||||
@@ -1,3 +1,10 @@
|
|||||||
|
import { base58 } from "@scure/base";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Token prefix for all RevIQ API tokens
|
||||||
|
*/
|
||||||
|
export const TOKEN_PREFIX = "reviq_";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Hash a token with SHA-256 for storage in database
|
* Hash a token with SHA-256 for storage in database
|
||||||
* Never store raw tokens - always hash first
|
* Never store raw tokens - always hash first
|
||||||
@@ -13,6 +20,34 @@ export const hashToken = async (token: string): Promise<string> => {
|
|||||||
.join("");
|
.join("");
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate that a token has the correct format
|
||||||
|
* Returns the raw bytes if valid, null if invalid
|
||||||
|
*/
|
||||||
|
export const parseToken = (token: string): Uint8Array | null => {
|
||||||
|
if (!token.startsWith(TOKEN_PREFIX)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const encoded = token.slice(TOKEN_PREFIX.length);
|
||||||
|
try {
|
||||||
|
const bytes = base58.decode(encoded);
|
||||||
|
// Expect 32 bytes of entropy
|
||||||
|
if (bytes.length !== 32) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return bytes;
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a token has the valid reviq_ prefix format
|
||||||
|
*/
|
||||||
|
export const isValidTokenFormat = (token: string): boolean => {
|
||||||
|
return parseToken(token) !== null;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a session token (UUID v4)
|
* Generate a session token (UUID v4)
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -12,14 +12,15 @@
|
|||||||
"cli": "bun run src/bin/reviq.ts",
|
"cli": "bun run src/bin/reviq.ts",
|
||||||
"typecheck": "tsc --noEmit",
|
"typecheck": "tsc --noEmit",
|
||||||
"lint": "eslint . --cache",
|
"lint": "eslint . --cache",
|
||||||
"clean": "rm -rf dist .eslintcache"
|
"clean": "rm -rf dist .eslintcache",
|
||||||
|
"test": "bun test"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@stricli/core": "^1.2.5",
|
"@noble/hashes": "^2.0.1",
|
||||||
"@stricli/auto-complete": "^1.0.0",
|
|
||||||
"@reviq/db": "workspace:*",
|
"@reviq/db": "workspace:*",
|
||||||
"@reviq/utils": "workspace:*",
|
"@scure/base": "^2.0.0",
|
||||||
"@noble/hashes": "^2.0.1"
|
"@stricli/auto-complete": "^1.0.0",
|
||||||
|
"@stricli/core": "^1.2.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@macalinao/eslint-config": "catalog:",
|
"@macalinao/eslint-config": "catalog:",
|
||||||
|
|||||||
@@ -1,6 +1,61 @@
|
|||||||
import type { LocalContext } from "../../context.js";
|
import type { LocalContext } from "../../context.js";
|
||||||
import { buildCommand } from "@stricli/core";
|
import { buildCommand } from "@stricli/core";
|
||||||
|
import { createApiClient } from "../../utils/api-client.js";
|
||||||
import { getConfigPath, readConfig } from "../../utils/config.js";
|
import { getConfigPath, readConfig } from "../../utils/config.js";
|
||||||
|
import { TOKEN_PREFIX } from "../../utils/token.js";
|
||||||
|
|
||||||
|
interface AuthStatusResponse {
|
||||||
|
user: {
|
||||||
|
id: number;
|
||||||
|
email: string;
|
||||||
|
displayName: string | null;
|
||||||
|
fullName: string | null;
|
||||||
|
isSuperuser: boolean;
|
||||||
|
emailVerified: boolean;
|
||||||
|
};
|
||||||
|
auth:
|
||||||
|
| {
|
||||||
|
method: "api_token";
|
||||||
|
tokenId: string;
|
||||||
|
tokenName: string;
|
||||||
|
expiresAt: string;
|
||||||
|
lastUsedAt: string | null;
|
||||||
|
createdAt: string;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
method: "session";
|
||||||
|
sessionId: string;
|
||||||
|
expiresAt: string;
|
||||||
|
createdAt: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatDate(dateStr: string): string {
|
||||||
|
const date = new Date(dateStr);
|
||||||
|
return date.toLocaleString();
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatRelativeTime(dateStr: string): string {
|
||||||
|
const date = new Date(dateStr);
|
||||||
|
const now = new Date();
|
||||||
|
const diffMs = date.getTime() - now.getTime();
|
||||||
|
const diffDays = Math.floor(diffMs / (1000 * 60 * 60 * 24));
|
||||||
|
|
||||||
|
if (diffDays < 0) {
|
||||||
|
return `${String(Math.abs(diffDays))} days ago`;
|
||||||
|
}
|
||||||
|
if (diffDays === 0) {
|
||||||
|
const diffHours = Math.floor(diffMs / (1000 * 60 * 60));
|
||||||
|
if (diffHours <= 0) {
|
||||||
|
return "expired";
|
||||||
|
}
|
||||||
|
return `in ${String(diffHours)} hours`;
|
||||||
|
}
|
||||||
|
if (diffDays === 1) {
|
||||||
|
return "tomorrow";
|
||||||
|
}
|
||||||
|
return `in ${String(diffDays)} days`;
|
||||||
|
}
|
||||||
|
|
||||||
async function status(this: LocalContext): Promise<void> {
|
async function status(this: LocalContext): Promise<void> {
|
||||||
const config = await readConfig();
|
const config = await readConfig();
|
||||||
@@ -16,10 +71,66 @@ async function status(this: LocalContext): Promise<void> {
|
|||||||
|
|
||||||
console.log("Authentication Status");
|
console.log("Authentication Status");
|
||||||
console.log("=====================\n");
|
console.log("=====================\n");
|
||||||
console.log(`Email: ${config.email}`);
|
|
||||||
console.log(`API URL: ${config.apiUrl}`);
|
// Show local config info
|
||||||
|
console.log("Local Configuration:");
|
||||||
console.log(` Config file: ${getConfigPath()}`);
|
console.log(` Config file: ${getConfigPath()}`);
|
||||||
console.log("Token: [configured]");
|
console.log(` API URL: ${config.apiUrl}`);
|
||||||
|
|
||||||
|
// Show token format info
|
||||||
|
const hasNewFormat = config.token.startsWith(TOKEN_PREFIX);
|
||||||
|
console.log(
|
||||||
|
` Token format: ${hasNewFormat ? "reviq_<base58>" : "legacy (hex)"}`,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Try to fetch status from API
|
||||||
|
console.log("\nAPI Status:");
|
||||||
|
try {
|
||||||
|
const client = await createApiClient();
|
||||||
|
const response = await client.call<AuthStatusResponse>("me.authStatus");
|
||||||
|
|
||||||
|
// User info
|
||||||
|
console.log("\n User:");
|
||||||
|
console.log(` Email: ${response.user.email}`);
|
||||||
|
if (response.user.displayName) {
|
||||||
|
console.log(` Display name: ${response.user.displayName}`);
|
||||||
|
}
|
||||||
|
if (response.user.fullName) {
|
||||||
|
console.log(` Full name: ${response.user.fullName}`);
|
||||||
|
}
|
||||||
|
console.log(
|
||||||
|
` Email verified: ${response.user.emailVerified ? "yes" : "no"}`,
|
||||||
|
);
|
||||||
|
console.log(` Superuser: ${response.user.isSuperuser ? "yes" : "no"}`);
|
||||||
|
|
||||||
|
// Auth method info
|
||||||
|
if (response.auth.method === "api_token") {
|
||||||
|
console.log("\n API Token:");
|
||||||
|
console.log(` Name: ${response.auth.tokenName}`);
|
||||||
|
console.log(` Token ID: ${response.auth.tokenId}`);
|
||||||
|
console.log(` Created: ${formatDate(response.auth.createdAt)}`);
|
||||||
|
console.log(
|
||||||
|
` Expires: ${formatDate(response.auth.expiresAt)} (${formatRelativeTime(response.auth.expiresAt)})`,
|
||||||
|
);
|
||||||
|
if (response.auth.lastUsedAt) {
|
||||||
|
console.log(` Last used: ${formatDate(response.auth.lastUsedAt)}`);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log("\n Session:");
|
||||||
|
console.log(` Session ID: ${response.auth.sessionId}`);
|
||||||
|
console.log(` Created: ${formatDate(response.auth.createdAt)}`);
|
||||||
|
console.log(
|
||||||
|
` Expires: ${formatDate(response.auth.expiresAt)} (${formatRelativeTime(response.auth.expiresAt)})`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.log(
|
||||||
|
` Error: ${error instanceof Error ? error.message : String(error)}`,
|
||||||
|
);
|
||||||
|
console.log(
|
||||||
|
"\n Unable to connect to API. Local credentials may be invalid.",
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const statusCommand = buildCommand({
|
export const statusCommand = buildCommand({
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
import type { LocalContext } from "../context.js";
|
import type { LocalContext } from "../context.js";
|
||||||
import { createDb } from "@reviq/db";
|
import { createDb, executeBootstrap } from "@reviq/db";
|
||||||
import { hashPassword } from "@reviq/utils";
|
|
||||||
import { buildCommand } from "@stricli/core";
|
import { buildCommand } from "@stricli/core";
|
||||||
import { writeConfig } from "../utils/config.js";
|
import { writeConfig } from "../utils/config.js";
|
||||||
import { generateToken, hashToken } from "../utils/token.js";
|
|
||||||
|
|
||||||
interface BootstrapFlags {
|
interface BootstrapFlags {
|
||||||
email: string;
|
email: string;
|
||||||
@@ -17,98 +15,23 @@ async function bootstrap(
|
|||||||
console.log("RevIQ Bootstrap - Create Superuser");
|
console.log("RevIQ Bootstrap - Create Superuser");
|
||||||
console.log("===================================\n");
|
console.log("===================================\n");
|
||||||
|
|
||||||
// Validate password length
|
|
||||||
if (flags.password.length < 8) {
|
|
||||||
console.error("Error: Password must be at least 8 characters");
|
|
||||||
this.process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
const db = createDb();
|
const db = createDb();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Check if user already exists
|
// Execute the bootstrap operation
|
||||||
const existing = await db
|
const result = await executeBootstrap(db, {
|
||||||
.selectFrom("users")
|
email: flags.email,
|
||||||
.where("email", "=", flags.email.toLowerCase())
|
password: flags.password,
|
||||||
.select("id")
|
});
|
||||||
.executeTakeFirst();
|
|
||||||
|
|
||||||
if (existing) {
|
console.log(`Created superuser: ${result.user.email}`);
|
||||||
console.error(`Error: User with email ${flags.email} already exists`);
|
console.log(`Created org: ${result.org.slug}`);
|
||||||
await db.destroy();
|
|
||||||
this.process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hash the password
|
|
||||||
const passwordHash = await hashPassword(flags.password);
|
|
||||||
|
|
||||||
// Create superuser
|
|
||||||
const [user] = await db
|
|
||||||
.insertInto("users")
|
|
||||||
.values({
|
|
||||||
email: flags.email.toLowerCase(),
|
|
||||||
password_hash: passwordHash,
|
|
||||||
is_superuser: true,
|
|
||||||
email_verified_at: new Date(),
|
|
||||||
})
|
|
||||||
.returning(["id", "email"])
|
|
||||||
.execute();
|
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
console.error("Error: Failed to create user");
|
|
||||||
await db.destroy();
|
|
||||||
this.process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Created superuser: ${user.email}`);
|
|
||||||
|
|
||||||
// Create "reviq" org
|
|
||||||
const [org] = await db
|
|
||||||
.insertInto("orgs")
|
|
||||||
.values({
|
|
||||||
slug: "reviq",
|
|
||||||
display_name: "RevIQ",
|
|
||||||
})
|
|
||||||
.returning(["id", "slug"])
|
|
||||||
.execute();
|
|
||||||
|
|
||||||
if (!org) {
|
|
||||||
console.error("Error: Failed to create org");
|
|
||||||
await db.destroy();
|
|
||||||
this.process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add user as owner of the org
|
|
||||||
await db
|
|
||||||
.insertInto("org_members")
|
|
||||||
.values({
|
|
||||||
org_id: org.id,
|
|
||||||
user_id: user.id,
|
|
||||||
role: "owner",
|
|
||||||
})
|
|
||||||
.execute();
|
|
||||||
|
|
||||||
console.log(`Created org: ${org.slug}`);
|
|
||||||
|
|
||||||
// Generate API token
|
|
||||||
const token = generateToken();
|
|
||||||
const tokenHashValue = hashToken(token);
|
|
||||||
|
|
||||||
await db
|
|
||||||
.insertInto("api_tokens")
|
|
||||||
.values({
|
|
||||||
user_id: user.id,
|
|
||||||
token_hash: tokenHashValue,
|
|
||||||
name: "CLI bootstrap token",
|
|
||||||
expires_at: new Date(Date.now() + 365 * 24 * 60 * 60 * 1000), // 1 year
|
|
||||||
})
|
|
||||||
.execute();
|
|
||||||
|
|
||||||
// Save to config
|
// Save to config
|
||||||
await writeConfig({
|
await writeConfig({
|
||||||
apiUrl: Bun.env.API_URL ?? "http://localhost:9861",
|
apiUrl: Bun.env.API_URL ?? "http://localhost:9861",
|
||||||
token,
|
token: result.token,
|
||||||
email: user.email,
|
email: result.user.email,
|
||||||
});
|
});
|
||||||
|
|
||||||
console.log("Saved credentials to ~/.config/reviq/credentials.json");
|
console.log("Saved credentials to ~/.config/reviq/credentials.json");
|
||||||
|
|||||||
170
apps/cli/src/utils/token.test.ts
Normal file
170
apps/cli/src/utils/token.test.ts
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
/**
|
||||||
|
* Tests for token generation and parsing utilities
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { describe, expect, test } from "bun:test";
|
||||||
|
import { generateToken, hashToken, parseToken, TOKEN_PREFIX } from "./token.js";
|
||||||
|
|
||||||
|
describe("TOKEN_PREFIX", () => {
|
||||||
|
test("should be 'reviq_'", () => {
|
||||||
|
expect(TOKEN_PREFIX).toBe("reviq_");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("generateToken", () => {
|
||||||
|
test("should generate a token with the reviq_ prefix", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
expect(token.startsWith(TOKEN_PREFIX)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should generate tokens of consistent length", () => {
|
||||||
|
const token1 = generateToken();
|
||||||
|
const token2 = generateToken();
|
||||||
|
const token3 = generateToken();
|
||||||
|
|
||||||
|
// Base58 encoding of 32 bytes produces 43-44 characters
|
||||||
|
// Plus 6 chars for "reviq_" prefix = 49-50 total
|
||||||
|
expect(token1.length).toBeGreaterThanOrEqual(49);
|
||||||
|
expect(token1.length).toBeLessThanOrEqual(50);
|
||||||
|
expect(token2.length).toBeGreaterThanOrEqual(49);
|
||||||
|
expect(token2.length).toBeLessThanOrEqual(50);
|
||||||
|
expect(token3.length).toBeGreaterThanOrEqual(49);
|
||||||
|
expect(token3.length).toBeLessThanOrEqual(50);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should generate unique tokens", () => {
|
||||||
|
const tokens = new Set<string>();
|
||||||
|
for (let i = 0; i < 100; i++) {
|
||||||
|
tokens.add(generateToken());
|
||||||
|
}
|
||||||
|
expect(tokens.size).toBe(100);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should only contain valid base58 characters after prefix", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const base58Part = token.slice(TOKEN_PREFIX.length);
|
||||||
|
// Base58 alphabet: 123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz
|
||||||
|
// (excludes 0, O, I, l)
|
||||||
|
const base58Regex =
|
||||||
|
/^[123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]+$/;
|
||||||
|
expect(base58Regex.test(base58Part)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("parseToken", () => {
|
||||||
|
test("should parse a valid token and return 32 bytes", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const bytes = parseToken(token);
|
||||||
|
expect(bytes).not.toBeNull();
|
||||||
|
expect(bytes?.length).toBe(32);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return null for tokens without the prefix", () => {
|
||||||
|
const bytes = parseToken("invalid_token");
|
||||||
|
expect(bytes).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return null for tokens with wrong prefix", () => {
|
||||||
|
const bytes = parseToken(
|
||||||
|
"wrong_5HueCGU8rMjxEXxiPuD5BDku4MkFqeZyd4dZ1jvhTVqvbTLvyTJ",
|
||||||
|
);
|
||||||
|
expect(bytes).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return null for tokens with invalid base58 characters", () => {
|
||||||
|
// 'O', '0', 'I', 'l' are not valid base58 characters
|
||||||
|
const bytes = parseToken("reviq_0InvalidBase58");
|
||||||
|
expect(bytes).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return null for tokens with incorrect byte length", () => {
|
||||||
|
// A shorter base58 string (less than 32 bytes)
|
||||||
|
const bytes = parseToken("reviq_abc123");
|
||||||
|
expect(bytes).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should round-trip correctly", () => {
|
||||||
|
const originalToken = generateToken();
|
||||||
|
const bytes = parseToken(originalToken);
|
||||||
|
expect(bytes).not.toBeNull();
|
||||||
|
|
||||||
|
// Verify the bytes are the same entropy
|
||||||
|
expect(bytes?.length).toBe(32);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("hashToken", () => {
|
||||||
|
test("should produce a 64-character hex string", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const hash = hashToken(token);
|
||||||
|
expect(hash.length).toBe(64);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should produce consistent hashes for the same token", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const hash1 = hashToken(token);
|
||||||
|
const hash2 = hashToken(token);
|
||||||
|
expect(hash1).toBe(hash2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should produce different hashes for different tokens", () => {
|
||||||
|
const token1 = generateToken();
|
||||||
|
const token2 = generateToken();
|
||||||
|
const hash1 = hashToken(token1);
|
||||||
|
const hash2 = hashToken(token2);
|
||||||
|
expect(hash1).not.toBe(hash2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should only contain hex characters", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const hash = hashToken(token);
|
||||||
|
const hexRegex = /^[0-9a-f]+$/;
|
||||||
|
expect(hexRegex.test(hash)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should hash legacy tokens correctly", () => {
|
||||||
|
// Test with a legacy hex token (pre-reviq_ format)
|
||||||
|
const legacyToken = "a".repeat(64);
|
||||||
|
const hash = hashToken(legacyToken);
|
||||||
|
expect(hash.length).toBe(64);
|
||||||
|
expect(hash).not.toBe(legacyToken);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("token security properties", () => {
|
||||||
|
test("should have sufficient entropy (32 bytes = 256 bits)", () => {
|
||||||
|
const token = generateToken();
|
||||||
|
const bytes = parseToken(token);
|
||||||
|
expect(bytes).not.toBeNull();
|
||||||
|
expect(bytes?.length).toBe(32);
|
||||||
|
// 32 bytes * 8 bits = 256 bits of entropy
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should be cryptographically random (statistical test)", () => {
|
||||||
|
// Generate many tokens and check that byte distribution is roughly uniform
|
||||||
|
const byteCounts: number[] = new Array<number>(256).fill(0);
|
||||||
|
const numTokens = 1000;
|
||||||
|
|
||||||
|
for (let i = 0; i < numTokens; i++) {
|
||||||
|
const token = generateToken();
|
||||||
|
const bytes = parseToken(token);
|
||||||
|
if (bytes) {
|
||||||
|
for (const byte of bytes) {
|
||||||
|
// byte is always 0-255, and byteCounts has 256 elements initialized to 0
|
||||||
|
byteCounts[byte] = (byteCounts[byte] ?? 0) + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Each byte value should appear roughly (numTokens * 32) / 256 times
|
||||||
|
const expectedCount = (numTokens * 32) / 256;
|
||||||
|
const tolerance = expectedCount * 0.5; // Allow 50% variance
|
||||||
|
|
||||||
|
// Check that no byte value is extremely over- or under-represented
|
||||||
|
for (let i = 0; i < 256; i++) {
|
||||||
|
const count: number = byteCounts[i] ?? 0;
|
||||||
|
expect(count).toBeGreaterThan(expectedCount - tolerance);
|
||||||
|
expect(count).toBeLessThan(expectedCount + tolerance);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,22 +1,12 @@
|
|||||||
/**
|
/**
|
||||||
* Token generation and hashing utilities
|
* Token generation and hashing utilities
|
||||||
|
*
|
||||||
|
* Re-exports from @reviq/db for convenience
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { sha256 } from "@noble/hashes/sha2.js";
|
export {
|
||||||
import { randomBytes } from "@noble/hashes/utils.js";
|
generateToken,
|
||||||
|
hashToken,
|
||||||
/**
|
parseToken,
|
||||||
* Generate a cryptographically secure random token
|
TOKEN_PREFIX,
|
||||||
* Returns a 32-byte hex string (64 characters)
|
} from "@reviq/db";
|
||||||
*/
|
|
||||||
export const generateToken = (): string => {
|
|
||||||
return Buffer.from(randomBytes(32)).toString("hex");
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Hash a token using SHA-256
|
|
||||||
* Returns a hex string
|
|
||||||
*/
|
|
||||||
export const hashToken = (token: string): string => {
|
|
||||||
return Buffer.from(sha256(Buffer.from(token))).toString("hex");
|
|
||||||
};
|
|
||||||
|
|||||||
12
bun.lock
12
bun.lock
@@ -23,6 +23,7 @@
|
|||||||
"@reviq/db": "workspace:*",
|
"@reviq/db": "workspace:*",
|
||||||
"@reviq/db-schema": "workspace:*",
|
"@reviq/db-schema": "workspace:*",
|
||||||
"@reviq/utils": "workspace:*",
|
"@reviq/utils": "workspace:*",
|
||||||
|
"@scure/base": "^2.0.0",
|
||||||
"@simplewebauthn/server": "^13.2.2",
|
"@simplewebauthn/server": "^13.2.2",
|
||||||
"@simplewebauthn/types": "^12.0.0",
|
"@simplewebauthn/types": "^12.0.0",
|
||||||
"kysely": "^0.28.2",
|
"kysely": "^0.28.2",
|
||||||
@@ -47,13 +48,13 @@
|
|||||||
"name": "@reviq/cli",
|
"name": "@reviq/cli",
|
||||||
"version": "0.0.0",
|
"version": "0.0.0",
|
||||||
"bin": {
|
"bin": {
|
||||||
"reviq": "./dist/reviq",
|
"reviq": "./dist/index.js",
|
||||||
"__reviq_bash_complete": "./dist/bash-complete",
|
"__reviq_bash_complete": "./dist/bash-complete.js",
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@noble/hashes": "^2.0.1",
|
"@noble/hashes": "^2.0.1",
|
||||||
"@reviq/db": "workspace:*",
|
"@reviq/db": "workspace:*",
|
||||||
"@reviq/utils": "workspace:*",
|
"@scure/base": "^2.0.0",
|
||||||
"@stricli/auto-complete": "^1.0.0",
|
"@stricli/auto-complete": "^1.0.0",
|
||||||
"@stricli/core": "^1.2.5",
|
"@stricli/core": "^1.2.5",
|
||||||
},
|
},
|
||||||
@@ -125,7 +126,10 @@
|
|||||||
"name": "@reviq/db",
|
"name": "@reviq/db",
|
||||||
"version": "0.0.1",
|
"version": "0.0.1",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@noble/hashes": "^2.0.1",
|
||||||
"@reviq/db-schema": "workspace:*",
|
"@reviq/db-schema": "workspace:*",
|
||||||
|
"@reviq/utils": "workspace:*",
|
||||||
|
"@scure/base": "^2.0.0",
|
||||||
"kysely": "^0.28.9",
|
"kysely": "^0.28.9",
|
||||||
"pg": "^8.13.1",
|
"pg": "^8.13.1",
|
||||||
},
|
},
|
||||||
@@ -453,6 +457,8 @@
|
|||||||
|
|
||||||
"@rollup/rollup-win32-x64-msvc": ["@rollup/rollup-win32-x64-msvc@4.55.1", "", { "os": "win32", "cpu": "x64" }, "sha512-SPEpaL6DX4rmcXtnhdrQYgzQ5W2uW3SCJch88lB2zImhJRhIIK44fkUrgIV/Q8yUNfw5oyZ5vkeQsZLhCb06lw=="],
|
"@rollup/rollup-win32-x64-msvc": ["@rollup/rollup-win32-x64-msvc@4.55.1", "", { "os": "win32", "cpu": "x64" }, "sha512-SPEpaL6DX4rmcXtnhdrQYgzQ5W2uW3SCJch88lB2zImhJRhIIK44fkUrgIV/Q8yUNfw5oyZ5vkeQsZLhCb06lw=="],
|
||||||
|
|
||||||
|
"@scure/base": ["@scure/base@2.0.0", "", {}, "sha512-3E1kpuZginKkek01ovG8krQ0Z44E3DHPjc5S2rjJw9lZn3KSQOs8S7wqikF/AH7iRanHypj85uGyxk0XAyC37w=="],
|
||||||
|
|
||||||
"@simplewebauthn/browser": ["@simplewebauthn/browser@13.2.2", "", {}, "sha512-FNW1oLQpTJyqG5kkDg5ZsotvWgmBaC6jCHR7Ej0qUNep36Wl9tj2eZu7J5rP+uhXgHaLk+QQ3lqcw2vS5MX1IA=="],
|
"@simplewebauthn/browser": ["@simplewebauthn/browser@13.2.2", "", {}, "sha512-FNW1oLQpTJyqG5kkDg5ZsotvWgmBaC6jCHR7Ej0qUNep36Wl9tj2eZu7J5rP+uhXgHaLk+QQ3lqcw2vS5MX1IA=="],
|
||||||
|
|
||||||
"@simplewebauthn/server": ["@simplewebauthn/server@13.2.2", "", { "dependencies": { "@hexagon/base64": "^1.1.27", "@levischuck/tiny-cbor": "^0.2.2", "@peculiar/asn1-android": "^2.3.10", "@peculiar/asn1-ecc": "^2.3.8", "@peculiar/asn1-rsa": "^2.3.8", "@peculiar/asn1-schema": "^2.3.8", "@peculiar/asn1-x509": "^2.3.8", "@peculiar/x509": "^1.13.0" } }, "sha512-HcWLW28yTMGXpwE9VLx9J+N2KEUaELadLrkPEEI9tpI5la70xNEVEsu/C+m3u7uoq4FulLqZQhgBCzR9IZhFpA=="],
|
"@simplewebauthn/server": ["@simplewebauthn/server@13.2.2", "", { "dependencies": { "@hexagon/base64": "^1.1.27", "@levischuck/tiny-cbor": "^0.2.2", "@peculiar/asn1-android": "^2.3.10", "@peculiar/asn1-ecc": "^2.3.8", "@peculiar/asn1-rsa": "^2.3.8", "@peculiar/asn1-schema": "^2.3.8", "@peculiar/asn1-x509": "^2.3.8", "@peculiar/x509": "^1.13.0" } }, "sha512-HcWLW28yTMGXpwE9VLx9J+N2KEUaELadLrkPEEI9tpI5la70xNEVEsu/C+m3u7uoq4FulLqZQhgBCzR9IZhFpA=="],
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ import {
|
|||||||
updateMemberRoleInputSchema,
|
updateMemberRoleInputSchema,
|
||||||
} from "./schemas/org.js";
|
} from "./schemas/org.js";
|
||||||
import {
|
import {
|
||||||
|
authStatusOutputSchema,
|
||||||
deviceOutputSchema,
|
deviceOutputSchema,
|
||||||
passkeyOutputSchema,
|
passkeyOutputSchema,
|
||||||
sessionOutputSchema,
|
sessionOutputSchema,
|
||||||
@@ -113,6 +114,9 @@ export const contract = oc.router({
|
|||||||
updateProfile: oc.input(updateProfileInputSchema).output(z.void()),
|
updateProfile: oc.input(updateProfileInputSchema).output(z.void()),
|
||||||
delete: oc.input(z.object({ password: z.string() })).output(z.void()),
|
delete: oc.input(z.object({ password: z.string() })).output(z.void()),
|
||||||
|
|
||||||
|
// Auth status (for CLI and debugging)
|
||||||
|
authStatus: oc.output(authStatusOutputSchema),
|
||||||
|
|
||||||
// Authentication settings
|
// Authentication settings
|
||||||
setPassword: oc.input(setPasswordInputSchema).output(z.void()),
|
setPassword: oc.input(setPasswordInputSchema).output(z.void()),
|
||||||
|
|
||||||
|
|||||||
@@ -98,3 +98,37 @@ export const deviceOutputSchema = z.object({
|
|||||||
export const trustDeviceInputSchema = z.object({
|
export const trustDeviceInputSchema = z.object({
|
||||||
name: nonEmptyString(100),
|
name: nonEmptyString(100),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auth status output schema for API token authentication
|
||||||
|
*/
|
||||||
|
export const apiTokenAuthStatusSchema = z.object({
|
||||||
|
method: z.literal("api_token"),
|
||||||
|
tokenId: z.string(),
|
||||||
|
tokenName: z.string(),
|
||||||
|
expiresAt: z.date(),
|
||||||
|
lastUsedAt: z.date().nullable(),
|
||||||
|
createdAt: z.date(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auth status output schema for session authentication
|
||||||
|
*/
|
||||||
|
export const sessionAuthStatusSchema = z.object({
|
||||||
|
method: z.literal("session"),
|
||||||
|
sessionId: z.string(),
|
||||||
|
expiresAt: z.date(),
|
||||||
|
createdAt: z.date(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Auth status output schema
|
||||||
|
* Returns information about the current authentication method
|
||||||
|
*/
|
||||||
|
export const authStatusOutputSchema = z.object({
|
||||||
|
user: userProfileSchema,
|
||||||
|
auth: z.discriminatedUnion("method", [
|
||||||
|
apiTokenAuthStatusSchema,
|
||||||
|
sessionAuthStatusSchema,
|
||||||
|
]),
|
||||||
|
});
|
||||||
|
|||||||
@@ -12,7 +12,10 @@
|
|||||||
"lint": "eslint . --cache"
|
"lint": "eslint . --cache"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@noble/hashes": "^2.0.1",
|
||||||
"@reviq/db-schema": "workspace:*",
|
"@reviq/db-schema": "workspace:*",
|
||||||
|
"@reviq/utils": "workspace:*",
|
||||||
|
"@scure/base": "^2.0.0",
|
||||||
"kysely": "^0.28.9",
|
"kysely": "^0.28.9",
|
||||||
"pg": "^8.13.1"
|
"pg": "^8.13.1"
|
||||||
},
|
},
|
||||||
|
|||||||
168
packages/db/src/helpers/execute-bootstrap.ts
Normal file
168
packages/db/src/helpers/execute-bootstrap.ts
Normal file
@@ -0,0 +1,168 @@
|
|||||||
|
/**
|
||||||
|
* Core bootstrap logic for creating a superuser and initial organization
|
||||||
|
*
|
||||||
|
* This is extracted from the CLI bootstrap command to make it reusable
|
||||||
|
* and testable. It operates on a database transaction.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type { Database } from "@reviq/db-schema";
|
||||||
|
import type { Kysely, Transaction } from "kysely";
|
||||||
|
import { hashPassword } from "@reviq/utils";
|
||||||
|
import { generateToken, hashToken } from "./token.js";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Input for the bootstrap operation
|
||||||
|
*/
|
||||||
|
export interface BootstrapInput {
|
||||||
|
/** Email address for the superuser */
|
||||||
|
email: string;
|
||||||
|
/** Password for the superuser */
|
||||||
|
password: string;
|
||||||
|
/** Optional organization slug (defaults to "reviq") */
|
||||||
|
orgSlug?: string;
|
||||||
|
/** Optional organization display name (defaults to "RevIQ") */
|
||||||
|
orgDisplayName?: string;
|
||||||
|
/** Optional token name (defaults to "CLI bootstrap token") */
|
||||||
|
tokenName?: string;
|
||||||
|
/** Optional token expiration in days (defaults to 365) */
|
||||||
|
tokenExpirationDays?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Result of the bootstrap operation
|
||||||
|
*/
|
||||||
|
export interface BootstrapResult {
|
||||||
|
/** The created user */
|
||||||
|
user: {
|
||||||
|
id: number;
|
||||||
|
email: string;
|
||||||
|
};
|
||||||
|
/** The created organization */
|
||||||
|
org: {
|
||||||
|
id: number;
|
||||||
|
slug: string;
|
||||||
|
};
|
||||||
|
/** The created API token (raw token, not hashed) */
|
||||||
|
token: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the bootstrap operation within a transaction
|
||||||
|
*
|
||||||
|
* Creates:
|
||||||
|
* - A superuser with the given email and password
|
||||||
|
* - An organization with the superuser as owner
|
||||||
|
* - An API token for the superuser
|
||||||
|
*
|
||||||
|
* @param trx - Database transaction (use db.transaction() or pass a Transaction)
|
||||||
|
* @param input - Bootstrap configuration
|
||||||
|
* @returns The created user, org, and API token
|
||||||
|
* @throws Error if user already exists or validation fails
|
||||||
|
*/
|
||||||
|
export const executeBootstrap = async (
|
||||||
|
trx: Kysely<Database> | Transaction<Database>,
|
||||||
|
input: BootstrapInput,
|
||||||
|
): Promise<BootstrapResult> => {
|
||||||
|
const {
|
||||||
|
email,
|
||||||
|
password,
|
||||||
|
orgSlug = "reviq",
|
||||||
|
orgDisplayName = "RevIQ",
|
||||||
|
tokenName = "CLI bootstrap token",
|
||||||
|
tokenExpirationDays = 365,
|
||||||
|
} = input;
|
||||||
|
|
||||||
|
// Validate password length
|
||||||
|
if (password.length < 8) {
|
||||||
|
throw new Error("Password must be at least 8 characters");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate email format (basic check)
|
||||||
|
if (!email.includes("@")) {
|
||||||
|
throw new Error("Invalid email address");
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalizedEmail = email.toLowerCase();
|
||||||
|
|
||||||
|
// Check if user already exists
|
||||||
|
const existing = await trx
|
||||||
|
.selectFrom("users")
|
||||||
|
.where("email", "=", normalizedEmail)
|
||||||
|
.select("id")
|
||||||
|
.executeTakeFirst();
|
||||||
|
|
||||||
|
if (existing) {
|
||||||
|
throw new Error(`User with email ${email} already exists`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash the password
|
||||||
|
const passwordHash = await hashPassword(password);
|
||||||
|
|
||||||
|
// Create superuser
|
||||||
|
const [user] = await trx
|
||||||
|
.insertInto("users")
|
||||||
|
.values({
|
||||||
|
email: normalizedEmail,
|
||||||
|
password_hash: passwordHash,
|
||||||
|
is_superuser: true,
|
||||||
|
email_verified_at: new Date(),
|
||||||
|
})
|
||||||
|
.returning(["id", "email"])
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new Error("Failed to create user");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create organization
|
||||||
|
const [org] = await trx
|
||||||
|
.insertInto("orgs")
|
||||||
|
.values({
|
||||||
|
slug: orgSlug,
|
||||||
|
display_name: orgDisplayName,
|
||||||
|
})
|
||||||
|
.returning(["id", "slug"])
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
if (!org) {
|
||||||
|
throw new Error("Failed to create organization");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add user as owner of the org
|
||||||
|
await trx
|
||||||
|
.insertInto("org_members")
|
||||||
|
.values({
|
||||||
|
org_id: org.id,
|
||||||
|
user_id: user.id,
|
||||||
|
role: "owner",
|
||||||
|
})
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Generate API token
|
||||||
|
const token = generateToken();
|
||||||
|
const tokenHashValue = hashToken(token);
|
||||||
|
|
||||||
|
await trx
|
||||||
|
.insertInto("api_tokens")
|
||||||
|
.values({
|
||||||
|
user_id: user.id,
|
||||||
|
token_hash: tokenHashValue,
|
||||||
|
name: tokenName,
|
||||||
|
expires_at: new Date(
|
||||||
|
Date.now() + tokenExpirationDays * 24 * 60 * 60 * 1000,
|
||||||
|
),
|
||||||
|
})
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return {
|
||||||
|
user: {
|
||||||
|
id: user.id,
|
||||||
|
email: user.email,
|
||||||
|
},
|
||||||
|
org: {
|
||||||
|
id: org.id,
|
||||||
|
slug: org.slug,
|
||||||
|
},
|
||||||
|
token,
|
||||||
|
};
|
||||||
|
};
|
||||||
51
packages/db/src/helpers/token.ts
Normal file
51
packages/db/src/helpers/token.ts
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
/**
|
||||||
|
* Token generation and hashing utilities
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { sha256 } from "@noble/hashes/sha2.js";
|
||||||
|
import { randomBytes } from "@noble/hashes/utils.js";
|
||||||
|
import { base58 } from "@scure/base";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Token prefix for all RevIQ API tokens
|
||||||
|
*/
|
||||||
|
export const TOKEN_PREFIX = "reviq_";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generate a cryptographically secure random token
|
||||||
|
* Format: reviq_<base58-encoded-32-bytes>
|
||||||
|
* Example: reviq_5HueCGU8rMjxEXxiPuD5BDku4MkFqeZyd4dZ1jvhTVqvbTLvyTJ
|
||||||
|
*/
|
||||||
|
export const generateToken = (): string => {
|
||||||
|
const bytes = randomBytes(32);
|
||||||
|
return TOKEN_PREFIX + base58.encode(bytes);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate that a token has the correct format
|
||||||
|
* Returns the raw bytes if valid, null if invalid
|
||||||
|
*/
|
||||||
|
export const parseToken = (token: string): Uint8Array | null => {
|
||||||
|
if (!token.startsWith(TOKEN_PREFIX)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
const encoded = token.slice(TOKEN_PREFIX.length);
|
||||||
|
try {
|
||||||
|
const bytes = base58.decode(encoded);
|
||||||
|
// Expect 32 bytes of entropy
|
||||||
|
if (bytes.length !== 32) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return bytes;
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hash a token using SHA-256
|
||||||
|
* Returns a hex string
|
||||||
|
*/
|
||||||
|
export const hashToken = (token: string): string => {
|
||||||
|
return Buffer.from(sha256(Buffer.from(token))).toString("hex");
|
||||||
|
};
|
||||||
@@ -2,24 +2,34 @@
|
|||||||
* Database client for RevIQ Publisher Dashboard
|
* Database client for RevIQ Publisher Dashboard
|
||||||
*
|
*
|
||||||
* @module @reviq/db
|
* @module @reviq/db
|
||||||
*/
|
|
||||||
|
|
||||||
import type { Database } from "@reviq/db-schema";
|
|
||||||
import type { Kysely } from "kysely";
|
|
||||||
import { createDb } from "./client.js";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Default database instance
|
|
||||||
*
|
*
|
||||||
* Uses DATABASE_URL environment variable for connection
|
* Usage:
|
||||||
|
* import { createDb } from "@reviq/db";
|
||||||
|
* const db = createDb(); // Uses DATABASE_URL env var
|
||||||
|
* // ... use db ...
|
||||||
|
* await db.destroy();
|
||||||
*/
|
*/
|
||||||
export const db: Kysely<Database> = createDb();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Re-export database types from db-schema
|
* Re-export database types from db-schema
|
||||||
*/
|
*/
|
||||||
export type { Database } from "@reviq/db-schema";
|
export type { Database } from "@reviq/db-schema";
|
||||||
/**
|
/**
|
||||||
* Export createDb for creating custom database instances
|
* Export createDb for creating database instances
|
||||||
|
* Callers should create their own database instance and pass it to functions
|
||||||
*/
|
*/
|
||||||
export { createDb } from "./client.js";
|
export { createDb } from "./client.js";
|
||||||
|
/**
|
||||||
|
* Export helper functions
|
||||||
|
*/
|
||||||
|
export {
|
||||||
|
type BootstrapInput,
|
||||||
|
type BootstrapResult,
|
||||||
|
executeBootstrap,
|
||||||
|
} from "./helpers/execute-bootstrap.js";
|
||||||
|
export {
|
||||||
|
generateToken,
|
||||||
|
hashToken,
|
||||||
|
parseToken,
|
||||||
|
TOKEN_PREFIX,
|
||||||
|
} from "./helpers/token.js";
|
||||||
|
|||||||
Reference in New Issue
Block a user