Skip to content

Commit ea992c3

Browse files
committed
Add login feature
1 parent 9b0ca28 commit ea992c3

File tree

5 files changed

+102
-1
lines changed

5 files changed

+102
-1
lines changed

package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,9 @@
1919
"chalk": "^5.2.0",
2020
"indent-string": "^5.0.0",
2121
"inquirer": "^9.2.7",
22-
"minimist": "^1.2.8"
22+
"minimist": "^1.2.8",
23+
"node-fetch": "^3.3.1",
24+
"open": "^9.1.0"
2325
},
2426
"devDependencies": {
2527
"@types/inquirer": "^9.0.3",

src/api.ts

+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
import os from 'node:os'
2+
import {URL} from 'node:url'
3+
import {createHash} from 'node:crypto'
4+
import chalk from 'chalk'
5+
import fetch from 'node-fetch'
6+
import open from 'open'
7+
import {randomString, sleep} from './utils.js'
8+
import {confirm} from './prompt.js'
9+
import fs from 'node:fs'
10+
import {webpodLocalConfig} from './env.js'
11+
12+
export async function login() {
13+
let baseUrl = 'https://app.webpod.dev'
14+
const localDev = createHash('sha1').update(process.env.WEBPOD_LOCAL_DEV ?? '').digest('hex')
15+
if (localDev == 'd4f57dba817bd3ddc1101e3498754ed234f08094') {
16+
baseUrl = 'http://0.0.0.0:8000'
17+
}
18+
const resp = await fetch(`${baseUrl}/api/user`, {
19+
headers: {
20+
Authorization: `Bearer ${process.env.WEBPOD_TOKEN}`,
21+
Accept: 'application/json',
22+
},
23+
})
24+
if (resp.status == 200) {
25+
const {id} = await resp.json() as { id: number }
26+
if (!Number.isFinite(id)) {
27+
throw new Error('Invalid user id')
28+
}
29+
return
30+
}
31+
32+
const randomCode = [randomString(), randomString(), randomString()].join('-')
33+
const url = new URL(`${baseUrl}/api-token`)
34+
url.searchParams.set('code', randomCode)
35+
url.searchParams.set('hostname', os.hostname())
36+
console.log(`Please visit ${chalk.underline(url.href)} to login.`)
37+
if (await confirm(`Open in browser?`)) {
38+
await open(url.href)
39+
}
40+
41+
console.log(chalk.magenta('Waiting for login...'))
42+
console.log('Alternatively, you can:')
43+
console.log('- Create API Token: https://app.webpod.dev/user/api-tokens')
44+
console.log(`- Set the ${chalk.bold('WEBPOD_TOKEN')} environment variable to the token.`)
45+
46+
while (true) {
47+
await sleep(3000)
48+
49+
const resp = await fetch(`${baseUrl}/api-token/${randomCode}`, {
50+
headers: {
51+
Accept: 'application/json',
52+
},
53+
})
54+
if (resp.status != 200) {
55+
continue
56+
}
57+
58+
const json = await resp.json() as { ok: number, token: string }
59+
if (!json.ok) {
60+
continue
61+
}
62+
63+
fs.appendFileSync(webpodLocalConfig(), `\nglobal.WEBPOD_TOKEN = '${json.token}';\n`)
64+
process.env.WEBPOD_TOKEN = json.token
65+
console.log(chalk.green('Login successful!'))
66+
67+
return
68+
}
69+
}

src/cli.ts

+5
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@ import {ask, confirm, skipPrompts} from './prompt.js'
1212
import fs from 'node:fs'
1313
import {handleError, StopError} from './error.js'
1414
import {detectFramework, setupFramework} from './framework.js'
15+
import {login} from './api.js'
16+
import {loadUserConfig} from './env.js'
1517

1618
const {cyan, grey, green, bold} = chalk
1719

@@ -54,6 +56,9 @@ await async function main() {
5456
process.exit(1)
5557
}
5658

59+
await loadUserConfig()
60+
await login()
61+
5762
let task, remoteUser, hostname, become
5863
if (argv._.length == 2) {
5964
task = argv._[0];

src/env.ts

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import path from 'node:path'
2+
import os from 'node:os'
3+
import fs from 'node:fs'
4+
5+
export function webpodLocalConfig() {
6+
return path.join(os.homedir(), '.webpod.js')
7+
}
8+
9+
export async function loadUserConfig() {
10+
if (fs.existsSync(webpodLocalConfig())) {
11+
await import(webpodLocalConfig())
12+
const token = (global as any).WEBPOD_TOKEN
13+
if (token && !process.env.WEBPOD_TOKEN) {
14+
process.env.WEBPOD_TOKEN = token
15+
}
16+
}
17+
}

src/utils.ts

+8
Original file line numberDiff line numberDiff line change
@@ -116,3 +116,11 @@ export function secondsToHumanReadableFormat(seconds: number): string {
116116
export function humanPath(...parts: string[]): string {
117117
return path.resolve(path.join(...parts)).replace(os.homedir(), '~')
118118
}
119+
120+
export function randomString(): string {
121+
return Math.random().toString(36).slice(2)
122+
}
123+
124+
export function sleep(ms: number): Promise<void> {
125+
return new Promise(resolve => setTimeout(resolve, ms))
126+
}

0 commit comments

Comments
 (0)