alpic 0.0.0-dev.fdfada2 → 0.0.0-dev.fe594dc

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.
Files changed (92) hide show
  1. package/dist/__tests__/auth.e2e.test.d.ts +1 -0
  2. package/dist/__tests__/auth.e2e.test.js +142 -0
  3. package/dist/__tests__/auth.e2e.test.js.map +1 -0
  4. package/dist/__tests__/deploy.e2e.test.d.ts +1 -0
  5. package/dist/__tests__/deploy.e2e.test.js +217 -0
  6. package/dist/__tests__/deploy.e2e.test.js.map +1 -0
  7. package/dist/__tests__/fixtures/demo-project/index.d.ts +1 -0
  8. package/dist/__tests__/fixtures/demo-project/index.js +4 -0
  9. package/dist/__tests__/fixtures/demo-project/index.js.map +1 -0
  10. package/dist/__tests__/git.e2e.test.d.ts +1 -0
  11. package/dist/__tests__/git.e2e.test.js +218 -0
  12. package/dist/__tests__/git.e2e.test.js.map +1 -0
  13. package/dist/__tests__/mock-server.d.ts +22 -0
  14. package/dist/__tests__/mock-server.js +492 -0
  15. package/dist/__tests__/mock-server.js.map +1 -0
  16. package/dist/__tests__/utils.d.ts +61 -0
  17. package/dist/__tests__/utils.js +197 -0
  18. package/dist/__tests__/utils.js.map +1 -0
  19. package/dist/api.d.ts +0 -1
  20. package/dist/api.js +5 -9
  21. package/dist/api.js.map +1 -1
  22. package/dist/commands/deploy.d.ts +2 -2
  23. package/dist/commands/deploy.js +6 -10
  24. package/dist/commands/deploy.js.map +1 -1
  25. package/dist/commands/git/connect.d.ts +9 -0
  26. package/dist/commands/git/connect.js +59 -0
  27. package/dist/commands/git/connect.js.map +1 -0
  28. package/dist/commands/git/disconnect.d.ts +9 -0
  29. package/dist/commands/git/disconnect.js +51 -0
  30. package/dist/commands/git/disconnect.js.map +1 -0
  31. package/dist/commands/git.d.ts +6 -0
  32. package/dist/commands/git.js +17 -0
  33. package/dist/commands/git.js.map +1 -0
  34. package/dist/commands/login.d.ts +6 -0
  35. package/dist/commands/login.js +39 -0
  36. package/dist/commands/login.js.map +1 -0
  37. package/dist/commands/logout.d.ts +6 -0
  38. package/dist/commands/logout.js +20 -0
  39. package/dist/commands/logout.js.map +1 -0
  40. package/dist/commands/whoami.d.ts +6 -0
  41. package/dist/commands/whoami.js +25 -0
  42. package/dist/commands/whoami.js.map +1 -0
  43. package/dist/env.d.ts +4 -0
  44. package/dist/env.js +10 -0
  45. package/dist/env.js.map +1 -0
  46. package/dist/lib/alpic-command.d.ts +4 -0
  47. package/dist/lib/alpic-command.js +17 -0
  48. package/dist/lib/alpic-command.js.map +1 -0
  49. package/dist/lib/auth/auth.d.ts +2 -0
  50. package/dist/lib/auth/auth.js +21 -0
  51. package/dist/lib/auth/auth.js.map +1 -0
  52. package/dist/lib/auth/oauth/client.d.ts +29 -0
  53. package/dist/lib/auth/oauth/client.js +112 -0
  54. package/dist/lib/auth/oauth/client.js.map +1 -0
  55. package/dist/lib/auth/oauth/constants.d.ts +2 -0
  56. package/dist/lib/auth/oauth/constants.js +3 -0
  57. package/dist/lib/auth/oauth/constants.js.map +1 -0
  58. package/dist/lib/auth/oauth/pages/CallbackPage.d.ts +7 -0
  59. package/dist/lib/auth/oauth/pages/CallbackPage.js +6 -0
  60. package/dist/lib/auth/oauth/pages/CallbackPage.js.map +1 -0
  61. package/dist/lib/auth/oauth/pages/Layout.d.ts +7 -0
  62. package/dist/lib/auth/oauth/pages/Layout.js +6 -0
  63. package/dist/lib/auth/oauth/pages/Layout.js.map +1 -0
  64. package/dist/lib/auth/oauth/pages/PickerPage.d.ts +5 -0
  65. package/dist/lib/auth/oauth/pages/PickerPage.js +9 -0
  66. package/dist/lib/auth/oauth/pages/PickerPage.js.map +1 -0
  67. package/dist/lib/auth/oauth/pages/styles.d.ts +1 -0
  68. package/dist/lib/auth/oauth/pages/styles.js +145 -0
  69. package/dist/lib/auth/oauth/pages/styles.js.map +1 -0
  70. package/dist/lib/auth/oauth/server.d.ts +8 -0
  71. package/dist/lib/auth/oauth/server.js +76 -0
  72. package/dist/lib/auth/oauth/server.js.map +1 -0
  73. package/dist/lib/auth/whoami.d.ts +28 -0
  74. package/dist/lib/auth/whoami.js +35 -0
  75. package/dist/lib/auth/whoami.js.map +1 -0
  76. package/dist/lib/deployment.d.ts +2 -3
  77. package/dist/lib/deployment.js +13 -8
  78. package/dist/lib/deployment.js.map +1 -1
  79. package/dist/lib/git.d.ts +14 -0
  80. package/dist/lib/git.js +105 -0
  81. package/dist/lib/git.js.map +1 -0
  82. package/dist/lib/global-store.d.ts +28 -0
  83. package/dist/lib/global-store.js +75 -0
  84. package/dist/lib/global-store.js.map +1 -0
  85. package/dist/lib/project.js +6 -2
  86. package/dist/lib/project.js.map +1 -1
  87. package/dist/lib/telemetry.js +6 -6
  88. package/dist/lib/telemetry.js.map +1 -1
  89. package/package.json +25 -24
  90. package/dist/lib/global-config.d.ts +0 -9
  91. package/dist/lib/global-config.js +0 -48
  92. package/dist/lib/global-config.js.map +0 -1
package/dist/env.js ADDED
@@ -0,0 +1,10 @@
1
+ import { createEnv } from "@t3-oss/env-core";
2
+ import { z } from "zod";
3
+ export const env = createEnv({
4
+ server: {
5
+ ALPIC_API_BASE_URL: z.string().default("https://api.alpic.ai"),
6
+ ALPIC_COGNITO_CLIENT_ID: z.string().default("1023dpimab488tfv4sh4bdk2r"),
7
+ },
8
+ runtimeEnv: process.env,
9
+ });
10
+ //# sourceMappingURL=env.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"env.js","sourceRoot":"","sources":["../src/env.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AAC7C,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,MAAM,CAAC,MAAM,GAAG,GAAG,SAAS,CAAC;IAC3B,MAAM,EAAE;QACN,kBAAkB,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,sBAAsB,CAAC;QAC9D,uBAAuB,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,2BAA2B,CAAC;KACzE;IACD,UAAU,EAAE,OAAO,CAAC,GAAG;CACxB,CAAC,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Command } from "@oclif/core";
2
+ export declare abstract class AlpicCommand extends Command {
3
+ catch(error: unknown): Promise<void>;
4
+ }
@@ -0,0 +1,17 @@
1
+ import * as p from "@clack/prompts";
2
+ import { Command } from "@oclif/core";
3
+ import { ORPCError } from "@orpc/client";
4
+ export class AlpicCommand extends Command {
5
+ async catch(error) {
6
+ if (error instanceof ORPCError) {
7
+ p.cancel(`An error occurred while connecting to Alpic: ${error.message}`);
8
+ return;
9
+ }
10
+ if (error instanceof Error) {
11
+ p.cancel(error.message);
12
+ return;
13
+ }
14
+ p.cancel(String(error));
15
+ }
16
+ }
17
+ //# sourceMappingURL=alpic-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"alpic-command.js","sourceRoot":"","sources":["../../src/lib/alpic-command.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,gBAAgB,CAAC;AACpC,OAAO,EAAE,OAAO,EAAE,MAAM,aAAa,CAAC;AACtC,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAEzC,MAAM,OAAgB,YAAa,SAAQ,OAAO;IACvC,KAAK,CAAC,KAAK,CAAC,KAAc;QACjC,IAAI,KAAK,YAAY,SAAS,EAAE,CAAC;YAC/B,CAAC,CAAC,MAAM,CAAC,gDAAgD,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YAC1E,OAAO;QACT,CAAC;QACD,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;YAC3B,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YACxB,OAAO;QACT,CAAC;QACD,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IAC1B,CAAC;CACF"}
@@ -0,0 +1,2 @@
1
+ export declare function getApiToken(): Promise<string | undefined>;
2
+ export declare function isAuthenticated(): Promise<boolean>;
@@ -0,0 +1,21 @@
1
+ import { oAuthClient } from "./oauth/client.js";
2
+ export async function getApiToken() {
3
+ const token = process.env.ALPIC_API_KEY ?? (await oAuthClient.getValidAccessToken())?.access_token;
4
+ return token;
5
+ }
6
+ export async function isAuthenticated() {
7
+ const isAuthenticatedViaApiKey = hasApiKey();
8
+ if (isAuthenticatedViaApiKey)
9
+ return true;
10
+ const isAuthenticatedViaOAuth = await hasValidAccessToken();
11
+ if (isAuthenticatedViaOAuth)
12
+ return true;
13
+ return false;
14
+ }
15
+ function hasApiKey() {
16
+ return process.env.ALPIC_API_KEY !== undefined;
17
+ }
18
+ async function hasValidAccessToken() {
19
+ return (await oAuthClient.getValidAccessToken()) !== null;
20
+ }
21
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../src/lib/auth/auth.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,CAAC,KAAK,UAAU,WAAW;IAC/B,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,CAAC,MAAM,WAAW,CAAC,mBAAmB,EAAE,CAAC,EAAE,YAAY,CAAC;IAEnG,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,eAAe;IACnC,MAAM,wBAAwB,GAAG,SAAS,EAAE,CAAC;IAC7C,IAAI,wBAAwB;QAAE,OAAO,IAAI,CAAC;IAC1C,MAAM,uBAAuB,GAAG,MAAM,mBAAmB,EAAE,CAAC;IAC5D,IAAI,uBAAuB;QAAE,OAAO,IAAI,CAAC;IAEzC,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,SAAS;IAChB,OAAO,OAAO,CAAC,GAAG,CAAC,aAAa,KAAK,SAAS,CAAC;AACjD,CAAC;AAED,KAAK,UAAU,mBAAmB;IAChC,OAAO,CAAC,MAAM,WAAW,CAAC,mBAAmB,EAAE,CAAC,KAAK,IAAI,CAAC;AAC5D,CAAC"}
@@ -0,0 +1,29 @@
1
+ import * as openid from "openid-client";
2
+ import { type Credentials } from "../../global-store.js";
3
+ export declare class OAuthClient {
4
+ private config;
5
+ initialize: Promise<void>;
6
+ constructor();
7
+ getValidAccessToken(): Promise<Credentials | null>;
8
+ fetchUserInfo(credentials: Credentials): Promise<openid.UserInfoResponse>;
9
+ refreshAccessToken(credentials: Credentials): Promise<Credentials>;
10
+ prepareOAuthConfig(): Promise<{
11
+ authorizeUrl: URL;
12
+ state: string;
13
+ nonce: string;
14
+ codeVerifier: string;
15
+ }>;
16
+ exchangeAuthorizationCode({ url, codeVerifier, state, nonce, }: {
17
+ url: URL;
18
+ codeVerifier: string;
19
+ state: string;
20
+ nonce: string;
21
+ }): Promise<openid.TokenEndpointResponse & openid.TokenEndpointResponseHelpers>;
22
+ getExpiresAt(expires_in: number): number;
23
+ private getCallbackUrl;
24
+ private loadConfig;
25
+ private fetchOAuthProtectedResourceConfig;
26
+ private getConfig;
27
+ private isAccessTokenExpired;
28
+ }
29
+ export declare const oAuthClient: OAuthClient;
@@ -0,0 +1,112 @@
1
+ import * as openid from "openid-client";
2
+ import { env } from "../../../env.js";
3
+ import { globalStore } from "../../global-store.js";
4
+ import { LOOPBACK_HOST, LOOPBACK_PORT } from "./constants.js";
5
+ const SCOPES = ["openid", "email", "profile"];
6
+ export class OAuthClient {
7
+ config = null;
8
+ initialize;
9
+ constructor() {
10
+ this.initialize = this.loadConfig();
11
+ }
12
+ async getValidAccessToken() {
13
+ await this.initialize;
14
+ const stored = globalStore.getCredentials();
15
+ if (!stored) {
16
+ return null;
17
+ }
18
+ if (this.isAccessTokenExpired(stored)) {
19
+ try {
20
+ return await this.refreshAccessToken(stored);
21
+ }
22
+ catch {
23
+ return null;
24
+ }
25
+ }
26
+ return stored;
27
+ }
28
+ async fetchUserInfo(credentials) {
29
+ return openid.fetchUserInfo(await this.getConfig(), credentials.access_token, credentials.sub);
30
+ }
31
+ async refreshAccessToken(credentials) {
32
+ if (!credentials.refresh_token) {
33
+ throw new Error("No refresh token available");
34
+ }
35
+ const response = await openid.refreshTokenGrant(await this.getConfig(), credentials.refresh_token);
36
+ const refreshed = {
37
+ access_token: response.access_token,
38
+ refresh_token: response.refresh_token ?? credentials.refresh_token,
39
+ expires_at: this.getExpiresAt(response.expires_in ?? 0),
40
+ sub: credentials.sub,
41
+ };
42
+ globalStore.saveCredentials(refreshed);
43
+ return refreshed;
44
+ }
45
+ async prepareOAuthConfig() {
46
+ await this.initialize;
47
+ if (!this.config) {
48
+ throw new Error("Config not loaded");
49
+ }
50
+ const codeVerifier = openid.randomPKCECodeVerifier();
51
+ const codeChallenge = await openid.calculatePKCECodeChallenge(codeVerifier);
52
+ const state = openid.randomState();
53
+ const nonce = openid.randomNonce();
54
+ const authorizeUrl = openid.buildAuthorizationUrl(this.config, {
55
+ redirect_uri: this.getCallbackUrl().toString(),
56
+ scope: SCOPES.join(" "),
57
+ code_challenge: codeChallenge,
58
+ code_challenge_method: "S256",
59
+ state,
60
+ nonce,
61
+ });
62
+ return { authorizeUrl, state, nonce, codeVerifier };
63
+ }
64
+ async exchangeAuthorizationCode({ url, codeVerifier, state, nonce, }) {
65
+ return await openid.authorizationCodeGrant(await this.getConfig(), url, {
66
+ pkceCodeVerifier: codeVerifier,
67
+ expectedState: state,
68
+ expectedNonce: nonce,
69
+ });
70
+ }
71
+ getExpiresAt(expires_in) {
72
+ return expires_in !== undefined ? Math.floor(Date.now() / 1000) + expires_in : Date.now() / 1000 + 3600;
73
+ }
74
+ getCallbackUrl() {
75
+ return new URL(`http://${LOOPBACK_HOST}:${LOOPBACK_PORT}/callback`);
76
+ }
77
+ async loadConfig() {
78
+ const protectedResourceConfig = await this.fetchOAuthProtectedResourceConfig();
79
+ const issuer = protectedResourceConfig.authorization_servers[0];
80
+ if (!issuer) {
81
+ throw new Error("No authorization server in OAuth protected resource config");
82
+ }
83
+ const issuerUrl = new URL(issuer);
84
+ try {
85
+ this.config = await openid.discovery(issuerUrl, env.ALPIC_COGNITO_CLIENT_ID);
86
+ }
87
+ catch {
88
+ throw new Error("Failed to discover OAuth config");
89
+ }
90
+ }
91
+ async fetchOAuthProtectedResourceConfig() {
92
+ const baseUrl = env.ALPIC_API_BASE_URL;
93
+ const response = await fetch(`${baseUrl}/.well-known/oauth-protected-resource`);
94
+ if (!response.ok) {
95
+ throw new Error(`Failed to load service config from ${baseUrl} (${response.status} ${response.statusText})`);
96
+ }
97
+ return (await response.json());
98
+ }
99
+ async getConfig() {
100
+ await this.initialize;
101
+ if (!this.config) {
102
+ throw new Error("Config not loaded");
103
+ }
104
+ return this.config;
105
+ }
106
+ isAccessTokenExpired(credentials) {
107
+ const EXPIRATION_WINDOW_IN_SECONDS = 300;
108
+ return Date.now() / 1000 + EXPIRATION_WINDOW_IN_SECONDS >= credentials.expires_at;
109
+ }
110
+ }
111
+ export const oAuthClient = new OAuthClient();
112
+ //# sourceMappingURL=client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.js","sourceRoot":"","sources":["../../../../src/lib/auth/oauth/client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare const LOOPBACK_HOST = "127.0.0.1";
2
+ export declare const LOOPBACK_PORT = 38472;
@@ -0,0 +1,3 @@
1
+ export const LOOPBACK_HOST = "127.0.0.1";
2
+ export const LOOPBACK_PORT = 38472;
3
+ //# sourceMappingURL=constants.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.js","sourceRoot":"","sources":["../../../../src/lib/auth/oauth/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,aAAa,GAAG,WAAW,CAAC;AACzC,MAAM,CAAC,MAAM,aAAa,GAAG,KAAK,CAAC"}
@@ -0,0 +1,7 @@
1
+ interface CallbackPageProps {
2
+ title: string;
3
+ body: string;
4
+ success: boolean;
5
+ }
6
+ export declare function CallbackPage({ title, body, success }: CallbackPageProps): import("preact").JSX.Element;
7
+ export {};
@@ -0,0 +1,6 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ import { Layout } from "./Layout.jsx";
3
+ export function CallbackPage({ title, body, success }) {
4
+ return (_jsx(Layout, { pageTitle: title, children: _jsxs("div", { class: success ? "callback callback--success" : "callback callback--error", children: [_jsx("h1", { children: title }), _jsx("p", { class: "subtitle", children: body })] }) }));
5
+ }
6
+ //# sourceMappingURL=CallbackPage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"CallbackPage.js","sourceRoot":"","sources":["../../../../../src/lib/auth/oauth/pages/CallbackPage.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAQtC,MAAM,UAAU,YAAY,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,OAAO,EAAqB;IACtE,OAAO,CACL,KAAC,MAAM,IAAC,SAAS,EAAE,KAAK,YACtB,eAAK,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC,4BAA4B,CAAC,CAAC,CAAC,0BAA0B,aAC7E,uBAAK,KAAK,GAAM,EAChB,YAAG,KAAK,EAAC,UAAU,YAAE,IAAI,GAAK,IAC1B,GACC,CACV,CAAC;AACJ,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { ComponentChildren } from "preact";
2
+ interface LayoutProps {
3
+ pageTitle: string;
4
+ children: ComponentChildren;
5
+ }
6
+ export declare function Layout({ pageTitle, children }: LayoutProps): import("preact").JSX.Element;
7
+ export {};
@@ -0,0 +1,6 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ import { STYLES } from "./styles.js";
3
+ export function Layout({ pageTitle, children }) {
4
+ return (_jsxs("html", { lang: "en", children: [_jsxs("head", { children: [_jsx("meta", { charset: "UTF-8" }), _jsx("meta", { name: "viewport", content: "width=device-width, initial-scale=1.0" }), _jsx("title", { children: pageTitle }), _jsx("link", { href: "https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600&family=Mozilla+Text:wght@200..700&display=swap", rel: "stylesheet" }), _jsx("style", { dangerouslySetInnerHTML: { __html: STYLES } })] }), _jsx("body", { children: _jsx("div", { class: "card", children: children }) })] }));
5
+ }
6
+ //# sourceMappingURL=Layout.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Layout.js","sourceRoot":"","sources":["../../../../../src/lib/auth/oauth/pages/Layout.tsx"],"names":[],"mappings":";AAEA,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAOrC,MAAM,UAAU,MAAM,CAAC,EAAE,SAAS,EAAE,QAAQ,EAAe;IACzD,OAAO,CACL,gBAAM,IAAI,EAAC,IAAI,aACb,2BACE,eAAM,OAAO,EAAC,OAAO,GAAG,EACxB,eAAM,IAAI,EAAC,UAAU,EAAC,OAAO,EAAC,uCAAuC,GAAG,EACxE,0BAAQ,SAAS,GAAS,EAC1B,eACE,IAAI,EAAC,gHAAgH,EACrH,GAAG,EAAC,YAAY,GAChB,EACF,gBAAO,uBAAuB,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,GAAI,IACjD,EACP,yBACE,cAAK,KAAK,EAAC,MAAM,YAAE,QAAQ,GAAO,GAC7B,IACF,CACR,CAAC;AACJ,CAAC"}
@@ -0,0 +1,5 @@
1
+ interface PickerPageProps {
2
+ authorizeUrlEscaped: string;
3
+ }
4
+ export declare function PickerPage({ authorizeUrlEscaped }: PickerPageProps): import("preact").JSX.Element;
5
+ export {};
@@ -0,0 +1,9 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ import { Layout } from "./Layout.jsx";
3
+ function providerHref(authorizeUrlEscaped, provider) {
4
+ return `${authorizeUrlEscaped}&identity_provider=${encodeURIComponent(provider)}`;
5
+ }
6
+ export function PickerPage({ authorizeUrlEscaped }) {
7
+ return (_jsxs(Layout, { pageTitle: "Sign in to Alpic", children: [_jsx("h1", { children: "Welcome to Alpic" }), _jsx("p", { class: "subtitle", children: "Sign in to manage your MCP Servers" }), _jsxs("div", { class: "buttons", children: [_jsxs("a", { href: providerHref(authorizeUrlEscaped, "Github"), class: "btn btn-github", children: [_jsxs("svg", { viewBox: "0 0 24 24", xmlns: "http://www.w3.org/2000/svg", children: [_jsx("title", { children: "GitHub" }), _jsx("path", { d: "M12 .297c-6.63 0-12 5.373-12 12 0 5.303 3.438 9.8 8.205 11.385.6.113.82-.258.82-.577 0-.285-.01-1.04-.015-2.04-3.338.724-4.042-1.61-4.042-1.61C4.422 18.07 3.633 17.7 3.633 17.7c-1.087-.744.084-.729.084-.729 1.205.084 1.838 1.236 1.838 1.236 1.07 1.835 2.809 1.305 3.495.998.108-.776.417-1.305.76-1.605-2.665-.3-5.466-1.332-5.466-5.93 0-1.31.465-2.38 1.235-3.22-.135-.303-.54-1.523.105-3.176 0 0 1.005-.322 3.3 1.23.96-.267 1.98-.399 3-.405 1.02.006 2.04.138 3 .405 2.28-1.552 3.285-1.23 3.285-1.23.645 1.653.24 2.873.12 3.176.765.84 1.23 1.91 1.23 3.22 0 4.61-2.805 5.625-5.475 5.92.42.36.81 1.096.81 2.22 0 1.606-.015 2.896-.015 3.286 0 .315.21.69.825.57C20.565 22.092 24 17.592 24 12.297c0-6.627-5.373-12-12-12" })] }), "Sign in with GitHub"] }), _jsxs("a", { href: providerHref(authorizeUrlEscaped, "Google"), class: "btn btn-google", children: [_jsxs("svg", { viewBox: "0 0 24 24", xmlns: "http://www.w3.org/2000/svg", children: [_jsx("title", { children: "Google" }), _jsx("path", { d: "M12.48 10.92v3.28h7.84c-.24 1.84-.853 3.187-1.8 4.133-1.147 1.147-2.933 2.4-6.04 2.4-4.827 0-8.6-3.893-8.6-8.72s3.773-8.72 8.6-8.72c2.6 0 4.507 1.027 5.907 2.347l2.307-2.307C18.747 1.44 16.133 0 12.48 0 5.867 0 .307 5.387.307 12s5.56 12 12.173 12c3.573 0 6.267-1.173 8.373-3.36 2.16-2.16 2.84-5.213 2.84-7.667 0-.76-.053-1.467-.173-2.053H12.48z" })] }), "Sign in with Google"] })] })] }));
8
+ }
9
+ //# sourceMappingURL=PickerPage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PickerPage.js","sourceRoot":"","sources":["../../../../../src/lib/auth/oauth/pages/PickerPage.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAMtC,SAAS,YAAY,CAAC,mBAA2B,EAAE,QAAgB;IACjE,OAAO,GAAG,mBAAmB,sBAAsB,kBAAkB,CAAC,QAAQ,CAAC,EAAE,CAAC;AACpF,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,EAAE,mBAAmB,EAAmB;IACjE,OAAO,CACL,MAAC,MAAM,IAAC,SAAS,EAAC,kBAAkB,aAClC,4CAAyB,EACzB,YAAG,KAAK,EAAC,UAAU,mDAAuC,EAC1D,eAAK,KAAK,EAAC,SAAS,aAClB,aAAG,IAAI,EAAE,YAAY,CAAC,mBAAmB,EAAE,QAAQ,CAAC,EAAE,KAAK,EAAC,gBAAgB,aAC1E,eAAK,OAAO,EAAC,WAAW,EAAC,KAAK,EAAC,4BAA4B,aACzD,qCAAqB,EACrB,eAAM,CAAC,EAAC,0sBAA0sB,GAAG,IACjtB,2BAEJ,EACJ,aAAG,IAAI,EAAE,YAAY,CAAC,mBAAmB,EAAE,QAAQ,CAAC,EAAE,KAAK,EAAC,gBAAgB,aAC1E,eAAK,OAAO,EAAC,WAAW,EAAC,KAAK,EAAC,4BAA4B,aACzD,qCAAqB,EACrB,eAAM,CAAC,EAAC,0VAA0V,GAAG,IACjW,2BAEJ,IACA,IACC,CACV,CAAC;AACJ,CAAC"}
@@ -0,0 +1 @@
1
+ export declare const STYLES = "\n *,\n *::before,\n *::after {\n margin: 0;\n padding: 0;\n box-sizing: border-box;\n }\n\n a {\n color: inherit;\n text-decoration: none;\n }\n\n :root {\n --midnight: #051413;\n }\n\n body {\n font-family: \"Inter\", system-ui, sans-serif;\n background-color: var(--midnight);\n min-height: 100vh;\n display: flex;\n padding: 64px;\n }\n\n .card {\n background: #fff;\n border-radius: 2px;\n padding: 48px;\n width: 50%;\n min-height: calc(100vh - 128px);\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n }\n\n h1 {\n font-family: \"Mozilla Text\", system-ui, sans-serif;\n font-weight: 400;\n font-size: 48px;\n color: var(--midnight);\n margin-bottom: 8px;\n }\n\n .subtitle {\n font-size: 20px;\n color: var(--midnight);\n margin-bottom: 32px;\n }\n\n .buttons {\n display: flex;\n flex-direction: column;\n gap: 12px;\n }\n\n .btn {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 8px;\n height: 48px;\n width: 256px;\n padding: 0 16px;\n border-radius: 2px;\n border: 1px solid var(--midnight);\n font-family: \"Inter\", system-ui, sans-serif;\n font-size: 16px;\n font-weight: 500;\n cursor: pointer;\n transition: background 0.15s;\n }\n\n .btn:focus-visible {\n outline: none;\n box-shadow: 0 0 0 2px var(--midnight);\n }\n\n .btn svg {\n width: 20px;\n height: 20px;\n flex-shrink: 0;\n }\n\n .btn-github {\n background: var(--midnight);\n color: #fff;\n }\n .btn-github svg {\n fill: #fff;\n }\n .btn-github:hover {\n background: rgba(5, 20, 19, 0.9);\n }\n\n .btn-google {\n background: #fff;\n color: var(--midnight);\n }\n .btn-google svg {\n fill: var(--midnight);\n }\n .btn-google:hover {\n background: rgba(5, 20, 19, 0.1);\n }\n\n .callback {\n padding-left: 24px;\n border-left: 4px solid var(--midnight);\n border-radius: 2px;\n }\n .callback--success {\n border-left-color: #10b981;\n }\n .callback--error {\n border-left-color: #ef4444;\n }\n .callback h1 {\n font-size: 36px;\n margin-bottom: 8px;\n }\n .callback .subtitle {\n margin-bottom: 0;\n }\n\n @media (max-width: 768px) {\n body {\n padding: 16px;\n }\n .card {\n width: 100%;\n height: auto;\n padding: 24px;\n }\n h1 {\n font-size: 36px;\n }\n .subtitle {\n font-size: 18px;\n }\n }\n";
@@ -0,0 +1,145 @@
1
+ export const STYLES = `
2
+ *,
3
+ *::before,
4
+ *::after {
5
+ margin: 0;
6
+ padding: 0;
7
+ box-sizing: border-box;
8
+ }
9
+
10
+ a {
11
+ color: inherit;
12
+ text-decoration: none;
13
+ }
14
+
15
+ :root {
16
+ --midnight: #051413;
17
+ }
18
+
19
+ body {
20
+ font-family: "Inter", system-ui, sans-serif;
21
+ background-color: var(--midnight);
22
+ min-height: 100vh;
23
+ display: flex;
24
+ padding: 64px;
25
+ }
26
+
27
+ .card {
28
+ background: #fff;
29
+ border-radius: 2px;
30
+ padding: 48px;
31
+ width: 50%;
32
+ min-height: calc(100vh - 128px);
33
+ display: flex;
34
+ flex-direction: column;
35
+ align-items: center;
36
+ justify-content: center;
37
+ }
38
+
39
+ h1 {
40
+ font-family: "Mozilla Text", system-ui, sans-serif;
41
+ font-weight: 400;
42
+ font-size: 48px;
43
+ color: var(--midnight);
44
+ margin-bottom: 8px;
45
+ }
46
+
47
+ .subtitle {
48
+ font-size: 20px;
49
+ color: var(--midnight);
50
+ margin-bottom: 32px;
51
+ }
52
+
53
+ .buttons {
54
+ display: flex;
55
+ flex-direction: column;
56
+ gap: 12px;
57
+ }
58
+
59
+ .btn {
60
+ display: flex;
61
+ align-items: center;
62
+ justify-content: center;
63
+ gap: 8px;
64
+ height: 48px;
65
+ width: 256px;
66
+ padding: 0 16px;
67
+ border-radius: 2px;
68
+ border: 1px solid var(--midnight);
69
+ font-family: "Inter", system-ui, sans-serif;
70
+ font-size: 16px;
71
+ font-weight: 500;
72
+ cursor: pointer;
73
+ transition: background 0.15s;
74
+ }
75
+
76
+ .btn:focus-visible {
77
+ outline: none;
78
+ box-shadow: 0 0 0 2px var(--midnight);
79
+ }
80
+
81
+ .btn svg {
82
+ width: 20px;
83
+ height: 20px;
84
+ flex-shrink: 0;
85
+ }
86
+
87
+ .btn-github {
88
+ background: var(--midnight);
89
+ color: #fff;
90
+ }
91
+ .btn-github svg {
92
+ fill: #fff;
93
+ }
94
+ .btn-github:hover {
95
+ background: rgba(5, 20, 19, 0.9);
96
+ }
97
+
98
+ .btn-google {
99
+ background: #fff;
100
+ color: var(--midnight);
101
+ }
102
+ .btn-google svg {
103
+ fill: var(--midnight);
104
+ }
105
+ .btn-google:hover {
106
+ background: rgba(5, 20, 19, 0.1);
107
+ }
108
+
109
+ .callback {
110
+ padding-left: 24px;
111
+ border-left: 4px solid var(--midnight);
112
+ border-radius: 2px;
113
+ }
114
+ .callback--success {
115
+ border-left-color: #10b981;
116
+ }
117
+ .callback--error {
118
+ border-left-color: #ef4444;
119
+ }
120
+ .callback h1 {
121
+ font-size: 36px;
122
+ margin-bottom: 8px;
123
+ }
124
+ .callback .subtitle {
125
+ margin-bottom: 0;
126
+ }
127
+
128
+ @media (max-width: 768px) {
129
+ body {
130
+ padding: 16px;
131
+ }
132
+ .card {
133
+ width: 100%;
134
+ height: auto;
135
+ padding: 24px;
136
+ }
137
+ h1 {
138
+ font-size: 36px;
139
+ }
140
+ .subtitle {
141
+ font-size: 18px;
142
+ }
143
+ }
144
+ `;
145
+ //# sourceMappingURL=styles.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"styles.js","sourceRoot":"","sources":["../../../../../src/lib/auth/oauth/pages/styles.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,MAAM,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA+IrB,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { Credentials } from "../../global-store.js";
2
+ export declare const getLoginPageUrl: () => string;
3
+ export declare const listenToOAuthCallback: ({ state, nonce, codeVerifier, authorizeUrl, }: {
4
+ state: string;
5
+ nonce: string;
6
+ codeVerifier: string;
7
+ authorizeUrl?: string;
8
+ }) => Promise<Credentials>;
@@ -0,0 +1,76 @@
1
+ import { jsx as _jsx } from "preact/jsx-runtime";
2
+ import * as p from "@clack/prompts";
3
+ import { createServer } from "node:http";
4
+ import { renderToString } from "preact-render-to-string";
5
+ import { oAuthClient } from "./client.js";
6
+ import { LOOPBACK_HOST, LOOPBACK_PORT } from "./constants.js";
7
+ import { CallbackPage } from "./pages/CallbackPage.js";
8
+ import { PickerPage } from "./pages/PickerPage.js";
9
+ const createCallbackServer = ({ state, nonce, codeVerifier, authorizeUrl, onSuccess, onError, }) => createServer(async (req, res) => {
10
+ const url = req.url ?? "/";
11
+ const path = url.split("?")[0];
12
+ if (path === "/" || path === "/login") {
13
+ if (authorizeUrl) {
14
+ res.writeHead(200, { "Content-Type": "text/html" });
15
+ res.end(renderToString(_jsx(PickerPage, { authorizeUrlEscaped: authorizeUrl })));
16
+ return;
17
+ }
18
+ p.log.message(`Ignoring request to ${url}`);
19
+ res.writeHead(404).end();
20
+ return;
21
+ }
22
+ if (!url.startsWith("/callback")) {
23
+ p.log.message(`Ignoring request to ${url}`);
24
+ res.writeHead(404).end();
25
+ return;
26
+ }
27
+ try {
28
+ const tokens = await oAuthClient.exchangeAuthorizationCode({
29
+ url: new URL(url, `http://${req.headers.host}`),
30
+ codeVerifier,
31
+ state,
32
+ nonce,
33
+ });
34
+ const sub = tokens.claims()?.sub;
35
+ if (!sub || !tokens.refresh_token) {
36
+ throw new Error("ID token did not contain sub");
37
+ }
38
+ const stored = {
39
+ access_token: tokens.access_token,
40
+ refresh_token: tokens.refresh_token,
41
+ expires_at: oAuthClient.getExpiresAt(tokens.expires_in ?? 0),
42
+ sub,
43
+ };
44
+ p.log.success("✅ Authentication successful");
45
+ res.writeHead(200, { "Content-Type": "text/html" });
46
+ res.end(renderToString(_jsx(CallbackPage, { title: "Logged in", body: "You can close this window.", success: true })));
47
+ onSuccess(stored);
48
+ }
49
+ catch (e) {
50
+ p.log.error(e instanceof Error ? e.message : String(e));
51
+ res.writeHead(500, { "Content-Type": "text/html" });
52
+ res.end(renderToString(_jsx(CallbackPage, { title: "Login failed", body: "Token exchange error. Try again.", success: false })));
53
+ onError(e instanceof Error ? e : new Error(String(e)));
54
+ }
55
+ });
56
+ export const getLoginPageUrl = () => `http://${LOOPBACK_HOST}:${LOOPBACK_PORT}/`;
57
+ export const listenToOAuthCallback = ({ state, nonce, codeVerifier, authorizeUrl, }) => {
58
+ return new Promise((resolve, reject) => {
59
+ const callbackServer = createCallbackServer({
60
+ state,
61
+ nonce,
62
+ codeVerifier,
63
+ authorizeUrl,
64
+ onSuccess: (storedToken) => {
65
+ callbackServer.close();
66
+ resolve(storedToken);
67
+ },
68
+ onError: (err) => {
69
+ callbackServer.close();
70
+ reject(err);
71
+ },
72
+ });
73
+ callbackServer.listen(LOOPBACK_PORT, LOOPBACK_HOST);
74
+ });
75
+ };
76
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../../../src/lib/auth/oauth/server.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,28 @@
1
+ export type WhoamiInfo = {
2
+ method: "api_key";
3
+ team: {
4
+ id: string;
5
+ name: string;
6
+ };
7
+ } | {
8
+ method: "oauth";
9
+ email: string;
10
+ name: string;
11
+ } | null;
12
+ export declare function getWhoamiInfo(): Promise<{
13
+ method: string;
14
+ team: {
15
+ id: string;
16
+ name: string;
17
+ createdAt: Date;
18
+ hasStripeAccount: boolean;
19
+ hasActiveSubscription: boolean;
20
+ } | undefined;
21
+ email?: undefined;
22
+ name?: undefined;
23
+ } | {
24
+ method: string;
25
+ email: string | undefined;
26
+ name: string | undefined;
27
+ team?: undefined;
28
+ } | null>;
@@ -0,0 +1,35 @@
1
+ import { api } from "../../api.js";
2
+ import { oAuthClient } from "./oauth/client.js";
3
+ export async function getWhoamiInfo() {
4
+ if (process.env.ALPIC_API_KEY !== undefined) {
5
+ const team = await getApiKeyTeam();
6
+ return {
7
+ method: "api_key",
8
+ team,
9
+ };
10
+ }
11
+ const token = await oAuthClient.getValidAccessToken();
12
+ if (!token)
13
+ return null;
14
+ const userInfo = await fetchOAuthUserInfo(token);
15
+ if (!userInfo)
16
+ return null;
17
+ return {
18
+ method: "oauth",
19
+ email: userInfo.email,
20
+ name: userInfo.name,
21
+ };
22
+ }
23
+ async function fetchOAuthUserInfo(credentials) {
24
+ try {
25
+ return await oAuthClient.fetchUserInfo(credentials);
26
+ }
27
+ catch {
28
+ return null;
29
+ }
30
+ }
31
+ async function getApiKeyTeam() {
32
+ const teams = await api.teams.list.v1();
33
+ return teams[0];
34
+ }
35
+ //# sourceMappingURL=whoami.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"whoami.js","sourceRoot":"","sources":["../../../src/lib/auth/whoami.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,cAAc,CAAC;AAEnC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAOhD,MAAM,CAAC,KAAK,UAAU,aAAa;IACjC,IAAI,OAAO,CAAC,GAAG,CAAC,aAAa,KAAK,SAAS,EAAE,CAAC;QAC5C,MAAM,IAAI,GAAG,MAAM,aAAa,EAAE,CAAC;QACnC,OAAO;YACL,MAAM,EAAE,SAAS;YACjB,IAAI;SACL,CAAC;IACJ,CAAC;IAED,MAAM,KAAK,GAAG,MAAM,WAAW,CAAC,mBAAmB,EAAE,CAAC;IACtD,IAAI,CAAC,KAAK;QAAE,OAAO,IAAI,CAAC;IAExB,MAAM,QAAQ,GAAG,MAAM,kBAAkB,CAAC,KAAK,CAAC,CAAC;IACjD,IAAI,CAAC,QAAQ;QAAE,OAAO,IAAI,CAAC;IAE3B,OAAO;QACL,MAAM,EAAE,OAAO;QACf,KAAK,EAAE,QAAQ,CAAC,KAAK;QACrB,IAAI,EAAE,QAAQ,CAAC,IAAI;KACpB,CAAC;AACJ,CAAC;AAED,KAAK,UAAU,kBAAkB,CAAC,WAAwB;IACxD,IAAI,CAAC;QACH,OAAO,MAAM,WAAW,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;IACtD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,KAAK,UAAU,aAAa;IAC1B,MAAM,KAAK,GAAG,MAAM,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC;IACxC,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC"}
@@ -1,10 +1,8 @@
1
1
  import type { RouterOutput } from "@alpic-ai/api";
2
2
  export declare function formatElapsed(ms: number): string;
3
- export declare function deployAndWait({ initial, startedAt, teamId, projectId, }: {
3
+ export declare function deployAndWait({ initial, startedAt, }: {
4
4
  initial: RouterOutput["deployments"]["get"]["v1"];
5
5
  startedAt: number;
6
- teamId: string;
7
- projectId: string;
8
6
  }): Promise<{
9
7
  deployment: {
10
8
  id: string;
@@ -16,6 +14,7 @@ export declare function deployAndWait({ initial, startedAt, teamId, projectId, }
16
14
  authorAvatarUrl: string | null;
17
15
  startedAt: Date | null;
18
16
  completedAt: Date | null;
17
+ deploymentPageUrl: string | null;
19
18
  };
20
19
  elapsedMs: number;
21
20
  }>;