Skip to content

Commit 0e3856d

Browse files
feat: add regions api (#9)
* add regions * Update src/lib/regions.ts Co-authored-by: Han Qiao <[email protected]> * Update src/lib/types.ts Co-authored-by: Han Qiao <[email protected]> * dep * dep * more work on types * rerun npm gen:types * Revert "rerun npm gen:types" This reverts commit 0980180. * chore: update mock response for unit tests --------- Co-authored-by: Han Qiao <[email protected]> Co-authored-by: Qiao Han <[email protected]>
1 parent abf4f53 commit 0e3856d

File tree

4 files changed

+168
-17
lines changed

4 files changed

+168
-17
lines changed

__tests__/regions.test.ts

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import nock from 'nock'
2+
import { describe, it, expect } from '@jest/globals'
3+
import { FLY_API_GRAPHQL } from '../src/client'
4+
import { createClient } from '../src/main'
5+
6+
const fly = createClient(process.env.FLY_API_TOKEN || 'test-token')
7+
8+
describe('regions', () => {
9+
it('get regions', async () => {
10+
const mockResponse = {
11+
data: {
12+
platform: {
13+
requestRegion: 'sin',
14+
regions: [
15+
{
16+
name: 'Amsterdam, Netherlands',
17+
code: 'ams',
18+
latitude: 52.374342,
19+
longitude: 4.895439,
20+
gatewayAvailable: true,
21+
requiresPaidPlan: false,
22+
},
23+
// Add more mock regions as needed
24+
],
25+
},
26+
},
27+
}
28+
29+
nock(FLY_API_GRAPHQL).post('/graphql').reply(200, mockResponse)
30+
31+
const data = await fly.Regions.getRegions()
32+
console.dir(data, { depth: 5 })
33+
34+
// Optionally, add assertions to verify the response
35+
expect(data).toBeDefined()
36+
expect(data.platform).toBeDefined()
37+
expect(data.platform.regions).toBeInstanceOf(Array)
38+
// Add more assertions as needed
39+
})
40+
})

src/client.ts

+3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import { Network } from './lib/network'
55
import { Organization } from './lib/organization'
66
import { Secret } from './lib/secret'
77
import { Volume } from './lib/volume'
8+
import { Regions } from './lib/regions'
89

910
export const FLY_API_GRAPHQL = 'https://api.fly.io'
1011
export const FLY_API_HOSTNAME = 'https://api.machines.dev'
@@ -36,6 +37,7 @@ class Client {
3637
private apiKey: string
3738
App: App
3839
Machine: Machine
40+
Regions: Regions
3941
Network: Network
4042
Organization: Organization
4143
Secret: Secret
@@ -51,6 +53,7 @@ class Client {
5153
this.App = new App(this)
5254
this.Machine = new Machine(this)
5355
this.Network = new Network(this)
56+
this.Regions = new Regions(this)
5457
this.Organization = new Organization(this)
5558
this.Secret = new Secret(this)
5659
this.Volume = new Volume(this)

src/lib/regions.ts

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
import Client from '../client'
2+
3+
interface RegionResponse {
4+
name: string
5+
code: string
6+
latitude: number
7+
longitude: number
8+
gatewayAvailable: boolean
9+
requiresPaidPlan: boolean
10+
}
11+
12+
interface PlatformResponse {
13+
requestRegion: string
14+
regions: RegionResponse[]
15+
}
16+
17+
export interface GetRegionsOutput {
18+
platform: PlatformResponse
19+
}
20+
21+
// Ref: https://github.com/superfly/flyctl/blob/master/api/resource_platform.go
22+
const getRegionsQuery = `query {
23+
platform {
24+
requestRegion
25+
regions {
26+
name
27+
code
28+
latitude
29+
longitude
30+
gatewayAvailable
31+
requiresPaidPlan
32+
}
33+
}
34+
}`
35+
36+
export class Regions {
37+
private client: Client
38+
39+
constructor(client: Client) {
40+
this.client = client
41+
}
42+
43+
async getRegions(): Promise<GetRegionsOutput> {
44+
return this.client.gqlPostOrThrow({
45+
query: getRegionsQuery,
46+
variables: {},
47+
})
48+
}
49+
}

src/lib/types.ts

+76-17
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010
*/
1111

1212
export interface App {
13+
id?: string
1314
name?: string
1415
organization?: Organization
1516
status?: string
@@ -24,6 +25,7 @@ export interface CheckStatus {
2425

2526
export interface CreateAppRequest {
2627
app_name?: string
28+
app_role_id?: string
2729
network?: string
2830
org_slug?: string
2931
}
@@ -34,9 +36,13 @@ export interface CreateLeaseRequest {
3436
}
3537

3638
export interface CreateMachineRequest {
39+
/** An object defining the Machine configuration */
3740
config?: ApiMachineConfig
3841
lease_ttl?: number
42+
lsvd?: boolean
43+
/** Unique name for this Machine. If omitted, one is generated for you */
3944
name?: string
45+
/** The target region. Omitting this param launches in the same region as your WireGuard peer connection (somewhere near you). */
4046
region?: string
4147
skip_launch?: boolean
4248
skip_service_registration?: boolean
@@ -53,12 +59,16 @@ export interface CreateVolumeRequest {
5359
size_gb?: number
5460
/** restore from snapshot */
5561
snapshot_id?: string
62+
snapshot_retention?: number
5663
/** fork from remote volume */
5764
source_volume_id?: string
5865
}
5966

6067
export interface ErrorResponse {
68+
/** Deprecated */
69+
details?: any
6170
error?: string
71+
status?: MainStatusCode
6272
}
6373

6474
export interface ExtendVolumeRequest {
@@ -87,9 +97,12 @@ export interface Lease {
8797
nonce?: string
8898
/** Owner is the user identifier which acquired the Lease. */
8999
owner?: string
100+
/** Machine version */
101+
version?: string
90102
}
91103

92104
export interface ListApp {
105+
id?: string
93106
machine_count?: number
94107
name?: string
95108
network?: any
@@ -171,15 +184,23 @@ export interface StopRequest {
171184
}
172185

173186
export interface UpdateMachineRequest {
187+
/** An object defining the Machine configuration */
174188
config?: ApiMachineConfig
175189
current_version?: string
176190
lease_ttl?: number
191+
lsvd?: boolean
192+
/** Unique name for this Machine. If omitted, one is generated for you */
177193
name?: string
194+
/** The target region. Omitting this param launches in the same region as your WireGuard peer connection (somewhere near you). */
178195
region?: string
179196
skip_launch?: boolean
180197
skip_service_registration?: boolean
181198
}
182199

200+
export interface UpdateVolumeRequest {
201+
snapshot_retention?: number
202+
}
203+
183204
export interface Volume {
184205
attached_alloc_id?: string
185206
attached_machine_id?: string
@@ -194,6 +215,7 @@ export interface Volume {
194215
name?: string
195216
region?: string
196217
size_gb?: number
218+
snapshot_retention?: number
197219
state?: string
198220
zone?: string
199221
}
@@ -203,73 +225,81 @@ export interface VolumeSnapshot {
203225
digest?: string
204226
id?: string
205227
size?: number
228+
status?: string
206229
}
207230

208231
export interface ApiDNSConfig {
209232
skip_registration?: boolean
210233
}
211234

235+
/** A file that will be written to the Machine. One of RawValue or SecretName must be set. */
212236
export interface ApiFile {
213237
/**
214238
* GuestPath is the path on the machine where the file will be written and must be an absolute path.
215-
* i.e. /full/path/to/file.json
239+
* For example: /full/path/to/file.json
216240
*/
217241
guest_path?: string
218-
/** RawValue containts the base64 encoded string of the file contents. */
242+
/** The base64 encoded string of the file contents. */
219243
raw_value?: string
220-
/** SecretName is the name of the secret that contains the base64 encoded file contents. */
244+
/** The name of the secret that contains the base64 encoded file contents. */
221245
secret_name?: string
222246
}
223247

224248
export interface ApiHTTPOptions {
225249
compress?: boolean
250+
h2_backend?: boolean
226251
response?: ApiHTTPResponseOptions
227252
}
228253

229254
export interface ApiHTTPResponseOptions {
230255
headers?: Record<string, any>
231256
}
232257

258+
/** An optional object that defines one or more named checks. The key for each check is the check name. */
233259
export interface ApiMachineCheck {
260+
/** The time to wait after a VM starts before checking its health */
234261
grace_period?: string
235262
headers?: ApiMachineHTTPHeader[]
263+
/** The time between connectivity checks */
236264
interval?: string
265+
/** For http checks, the HTTP method to use to when making the request */
237266
method?: string
267+
/** For http checks, the path to send the request to */
238268
path?: string
269+
/** The port to connect to, often the same as internal_port */
239270
port?: number
271+
/** For http checks, whether to use http or https */
240272
protocol?: string
273+
/** The maximum time a connection can take before being reported as failing its health check */
241274
timeout?: string
275+
/** If the protocol is https, the hostname to use for TLS certificate validation */
242276
tls_server_name?: string
277+
/** For http checks with https protocol, whether or not to verify the TLS certificate */
243278
tls_skip_verify?: boolean
279+
/** tcp or http */
244280
type?: string
245281
}
246282

247283
export interface ApiMachineConfig {
284+
/** Optional boolean telling the Machine to destroy itself once it’s complete (default false) */
248285
auto_destroy?: boolean
249286
checks?: Record<string, ApiMachineCheck>
250287
/** Deprecated: use Service.Autostart instead */
251288
disable_machine_autostart?: boolean
252289
dns?: ApiDNSConfig
253-
/**
254-
* Fields managed from fly.toml
255-
* If you add anything here, ensure appconfig.Config.ToMachine() is updated
256-
*/
290+
/** An object filled with key/value pairs to be set as environment variables */
257291
env?: Record<string, string>
258292
files?: ApiFile[]
259293
guest?: ApiMachineGuest
260-
host_dedication_id?: string
261-
/** Set by fly deploy or fly machines commands */
294+
/** The docker image to run */
262295
image?: string
263296
init?: ApiMachineInit
264297
metadata?: Record<string, string>
265298
metrics?: ApiMachineMetrics
266299
mounts?: ApiMachineMount[]
267300
processes?: ApiMachineProcess[]
301+
/** The Machine restart policy defines whether and how flyd restarts a Machine after its main process exits. See https://fly.io/docs/machines/guides-examples/machine-restart-policy/. */
268302
restart?: ApiMachineRestart
269-
/**
270-
* The following fields can only be set or updated by `fly machines run|update` commands
271-
* "fly deploy" must preserve them, if you add anything here, ensure it is propagated on deploys
272-
*/
273303
schedule?: string
274304
services?: ApiMachineService[]
275305
/** Deprecated: use Guest instead */
@@ -286,13 +316,17 @@ export interface ApiMachineConfig {
286316
export interface ApiMachineGuest {
287317
cpu_kind?: string
288318
cpus?: number
289-
gpus?: number
319+
gpu_kind?: string
320+
host_dedication_id?: string
290321
kernel_args?: string[]
291322
memory_mb?: number
292323
}
293324

325+
/** For http checks, an array of objects with string field Name and array of strings field Values. The key/value pairs specify header and header values that will get passed with the check call. */
294326
export interface ApiMachineHTTPHeader {
327+
/** The header name */
295328
name?: string
329+
/** The header value */
296330
values?: string[]
297331
}
298332

@@ -311,10 +345,13 @@ export interface ApiMachineMetrics {
311345
}
312346

313347
export interface ApiMachineMount {
348+
add_size_gb?: number
314349
encrypted?: boolean
350+
extend_threshold_percent?: number
315351
name?: string
316352
path?: string
317353
size_gb?: number
354+
size_gb_limit?: number
318355
volume?: string
319356
}
320357

@@ -337,10 +374,16 @@ export interface ApiMachineProcess {
337374
user?: string
338375
}
339376

377+
/** The Machine restart policy defines whether and how flyd restarts a Machine after its main process exits. See https://fly.io/docs/machines/guides-examples/machine-restart-policy/. */
340378
export interface ApiMachineRestart {
341-
/** MaxRetries is only relevant with the on-failure policy. */
379+
/** When policy is on-failure, the maximum number of times to attempt to restart the Machine before letting it stop. */
342380
max_retries?: number
343-
policy?: string
381+
/**
382+
* * no - Never try to restart a Machine automatically when its main process exits, whether that’s on purpose or on a crash.
383+
* * always - Always restart a Machine automatically and never let it enter a stopped state, even when the main process exits cleanly.
384+
* * on-failure - Try up to MaxRetries times to automatically restart the Machine if it exits with a non-zero exit code. Default when no explicit policy is set, and for Machines with schedules.
385+
*/
386+
policy?: ApiMachineRestartPolicyEnum
344387
}
345388

346389
export interface ApiMachineService {
@@ -382,6 +425,11 @@ export interface ApiTLSOptions {
382425
versions?: string[]
383426
}
384427

428+
export enum MainStatusCode {
429+
Unknown = 'unknown',
430+
CapacityErr = 'insufficient_capacity',
431+
}
432+
385433
export enum SignalRequestSignalEnum {
386434
SIGABRT = 'SIGABRT',
387435
SIGALRM = 'SIGALRM',
@@ -398,9 +446,20 @@ export enum SignalRequestSignalEnum {
398446
SIGUSR1 = 'SIGUSR1',
399447
}
400448

449+
/**
450+
* * no - Never try to restart a Machine automatically when its main process exits, whether that’s on purpose or on a crash.
451+
* * always - Always restart a Machine automatically and never let it enter a stopped state, even when the main process exits cleanly.
452+
* * on-failure - Try up to MaxRetries times to automatically restart the Machine if it exits with a non-zero exit code. Default when no explicit policy is set, and for Machines with schedules.
453+
*/
454+
export enum ApiMachineRestartPolicyEnum {
455+
No = 'no',
456+
Always = 'always',
457+
OnFailure = 'on-failure',
458+
}
459+
401460
export interface AppsListParams {
402461
/** The org slug, or 'personal', to filter apps */
403-
org_slug?: string
462+
org_slug: string
404463
}
405464

406465
export interface MachinesListParams {

0 commit comments

Comments
 (0)