@gitpod/gitpod-protocol 0.1.5-vn-update-gitpod-json-schema.0 → 0.1.5-vn-jetbrains-backend-plugin-223.6

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 (183) hide show
  1. package/data/gitpod-schema.json +63 -113
  2. package/lib/accounting-protocol.d.ts +8 -0
  3. package/lib/accounting-protocol.d.ts.map +1 -1
  4. package/lib/accounting-protocol.js +12 -1
  5. package/lib/accounting-protocol.js.map +1 -1
  6. package/lib/admin-protocol.d.ts +7 -0
  7. package/lib/admin-protocol.d.ts.map +1 -1
  8. package/lib/admin-protocol.js.map +1 -1
  9. package/lib/attribution.d.ts +20 -0
  10. package/lib/attribution.d.ts.map +1 -0
  11. package/lib/attribution.js +40 -0
  12. package/lib/attribution.js.map +1 -0
  13. package/lib/billing-mode.d.ts +41 -0
  14. package/lib/billing-mode.d.ts.map +1 -0
  15. package/lib/billing-mode.js +44 -0
  16. package/lib/billing-mode.js.map +1 -0
  17. package/lib/blocked-repositories-protocol.d.ts +13 -0
  18. package/lib/blocked-repositories-protocol.d.ts.map +1 -0
  19. package/lib/blocked-repositories-protocol.js +8 -0
  20. package/lib/blocked-repositories-protocol.js.map +1 -0
  21. package/lib/context-url.js +1 -1
  22. package/lib/context-url.js.map +1 -1
  23. package/lib/encryption/encryption-engine.d.ts.map +1 -1
  24. package/lib/encryption/encryption-engine.js +3 -3
  25. package/lib/encryption/encryption-engine.js.map +1 -1
  26. package/lib/encryption/encryption-engine.spec.js +1 -1
  27. package/lib/encryption/encryption-engine.spec.js.map +1 -1
  28. package/lib/encryption/key-provider.js +1 -1
  29. package/lib/encryption/key-provider.js.map +1 -1
  30. package/lib/experiments/always-default.d.ts +8 -0
  31. package/lib/experiments/always-default.d.ts.map +1 -0
  32. package/lib/experiments/always-default.js +20 -0
  33. package/lib/experiments/always-default.js.map +1 -0
  34. package/lib/experiments/configcat-server.d.ts +10 -0
  35. package/lib/experiments/configcat-server.d.ts.map +1 -0
  36. package/lib/experiments/configcat-server.js +36 -0
  37. package/lib/experiments/configcat-server.js.map +1 -0
  38. package/lib/experiments/configcat.d.ts +23 -0
  39. package/lib/experiments/configcat.d.ts.map +1 -0
  40. package/lib/experiments/configcat.js +57 -0
  41. package/lib/experiments/configcat.js.map +1 -0
  42. package/lib/experiments/types.d.ts +24 -0
  43. package/lib/experiments/types.d.ts.map +1 -0
  44. package/lib/experiments/types.js +10 -0
  45. package/lib/experiments/types.js.map +1 -0
  46. package/lib/gitpod-service.d.ts +39 -12
  47. package/lib/gitpod-service.d.ts.map +1 -1
  48. package/lib/gitpod-service.js +13 -1
  49. package/lib/gitpod-service.js.map +1 -1
  50. package/lib/ide-protocol.d.ts +8 -0
  51. package/lib/ide-protocol.d.ts.map +1 -1
  52. package/lib/index.d.ts +1 -1
  53. package/lib/index.d.ts.map +1 -1
  54. package/lib/index.js +1 -1
  55. package/lib/index.js.map +1 -1
  56. package/lib/installation-admin-protocol.d.ts +3 -0
  57. package/lib/installation-admin-protocol.d.ts.map +1 -1
  58. package/lib/installation-admin-protocol.js +1 -0
  59. package/lib/installation-admin-protocol.js.map +1 -1
  60. package/lib/messaging/client-call-metrics.d.ts +1 -17
  61. package/lib/messaging/client-call-metrics.d.ts.map +1 -1
  62. package/lib/messaging/client-call-metrics.js +1 -2
  63. package/lib/messaging/client-call-metrics.js.map +1 -1
  64. package/lib/messaging/error.d.ts +6 -1
  65. package/lib/messaging/error.d.ts.map +1 -1
  66. package/lib/messaging/error.js +13 -3
  67. package/lib/messaging/error.js.map +1 -1
  68. package/lib/plans.d.ts +1 -0
  69. package/lib/plans.d.ts.map +1 -1
  70. package/lib/plans.js +4 -0
  71. package/lib/plans.js.map +1 -1
  72. package/lib/protocol.d.ts +94 -12
  73. package/lib/protocol.d.ts.map +1 -1
  74. package/lib/protocol.js +119 -13
  75. package/lib/protocol.js.map +1 -1
  76. package/lib/protocol.spec.d.ts +7 -0
  77. package/lib/protocol.spec.d.ts.map +1 -0
  78. package/lib/protocol.spec.js +127 -0
  79. package/lib/protocol.spec.js.map +1 -0
  80. package/lib/team-subscription-protocol.d.ts +17 -0
  81. package/lib/team-subscription-protocol.d.ts.map +1 -1
  82. package/lib/team-subscription-protocol.js +16 -1
  83. package/lib/team-subscription-protocol.js.map +1 -1
  84. package/lib/teams-projects-protocol.d.ts +14 -1
  85. package/lib/teams-projects-protocol.d.ts.map +1 -1
  86. package/lib/teams-projects-protocol.js.map +1 -1
  87. package/lib/usage.d.ts +61 -0
  88. package/lib/usage.d.ts.map +1 -0
  89. package/lib/usage.js +14 -0
  90. package/lib/usage.js.map +1 -0
  91. package/lib/util/debug-app.d.ts +25 -0
  92. package/lib/util/debug-app.d.ts.map +1 -0
  93. package/lib/util/debug-app.js +100 -0
  94. package/lib/util/debug-app.js.map +1 -0
  95. package/lib/util/garbage-collected-cache.d.ts.map +1 -1
  96. package/lib/util/garbage-collected-cache.js +5 -0
  97. package/lib/util/garbage-collected-cache.js.map +1 -1
  98. package/lib/util/gitpod-host-url.d.ts +3 -0
  99. package/lib/util/gitpod-host-url.d.ts.map +1 -1
  100. package/lib/util/gitpod-host-url.js +18 -0
  101. package/lib/util/gitpod-host-url.js.map +1 -1
  102. package/lib/util/grpc.d.ts +19 -0
  103. package/lib/util/grpc.d.ts.map +1 -1
  104. package/lib/util/grpc.js +73 -1
  105. package/lib/util/grpc.js.map +1 -1
  106. package/lib/util/logging.d.ts.map +1 -1
  107. package/lib/util/logging.js +8 -1
  108. package/lib/util/logging.js.map +1 -1
  109. package/lib/util/logging.spec.d.ts +7 -0
  110. package/lib/util/logging.spec.d.ts.map +1 -0
  111. package/lib/util/logging.spec.js +52 -0
  112. package/lib/util/logging.spec.js.map +1 -0
  113. package/lib/util/timeutil.d.ts +2 -0
  114. package/lib/util/timeutil.d.ts.map +1 -1
  115. package/lib/util/timeutil.js +13 -1
  116. package/lib/util/timeutil.js.map +1 -1
  117. package/lib/util/timeutil.spec.d.ts +2 -0
  118. package/lib/util/timeutil.spec.d.ts.map +1 -1
  119. package/lib/util/timeutil.spec.js +34 -0
  120. package/lib/util/timeutil.spec.js.map +1 -1
  121. package/lib/webhook-event.d.ts +44 -0
  122. package/lib/webhook-event.d.ts.map +1 -0
  123. package/lib/webhook-event.js +8 -0
  124. package/lib/webhook-event.js.map +1 -0
  125. package/lib/workspace-class.d.ts +14 -0
  126. package/lib/workspace-class.d.ts.map +1 -0
  127. package/lib/workspace-class.js +8 -0
  128. package/lib/workspace-class.js.map +1 -0
  129. package/lib/workspace-cluster.d.ts +5 -7
  130. package/lib/workspace-cluster.d.ts.map +1 -1
  131. package/lib/workspace-cluster.js.map +1 -1
  132. package/lib/workspace-instance.d.ts +12 -0
  133. package/lib/workspace-instance.d.ts.map +1 -1
  134. package/lib/wsready.d.ts +3 -3
  135. package/lib/wsready.d.ts.map +1 -1
  136. package/lib/wsready.js +0 -2
  137. package/lib/wsready.js.map +1 -1
  138. package/package.json +12 -6
  139. package/pkg-yarn.lock +8 -5
  140. package/provenance-bundle.jsonl +1 -1
  141. package/src/accounting-protocol.ts +13 -0
  142. package/src/admin-protocol.ts +10 -0
  143. package/src/attribution.ts +48 -0
  144. package/src/billing-mode.ts +77 -0
  145. package/src/blocked-repositories-protocol.ts +13 -0
  146. package/src/context-url.ts +1 -1
  147. package/src/encryption/encryption-engine.spec.ts +1 -1
  148. package/src/encryption/encryption-engine.ts +7 -3
  149. package/src/encryption/key-provider.ts +1 -1
  150. package/src/experiments/always-default.ts +24 -0
  151. package/src/experiments/configcat-server.ts +41 -0
  152. package/src/experiments/configcat.ts +62 -0
  153. package/src/experiments/types.ts +38 -0
  154. package/src/gitpod-service.ts +67 -11
  155. package/src/ide-protocol.ts +10 -0
  156. package/src/index.ts +1 -1
  157. package/src/installation-admin-protocol.ts +3 -0
  158. package/src/messaging/client-call-metrics.ts +1 -20
  159. package/src/messaging/error.ts +19 -4
  160. package/src/plans.ts +4 -0
  161. package/src/protocol.spec.ts +97 -0
  162. package/src/protocol.ts +205 -20
  163. package/src/team-subscription-protocol.ts +27 -0
  164. package/src/teams-projects-protocol.ts +15 -1
  165. package/src/usage.ts +71 -0
  166. package/src/util/debug-app.ts +81 -0
  167. package/src/util/garbage-collected-cache.ts +5 -0
  168. package/src/util/gitpod-host-url.ts +21 -0
  169. package/src/util/grpc.ts +89 -0
  170. package/src/util/logging.spec.ts +23 -0
  171. package/src/util/logging.ts +11 -2
  172. package/src/util/timeutil.spec.ts +28 -1
  173. package/src/util/timeutil.ts +12 -0
  174. package/src/webhook-event.ts +55 -0
  175. package/src/workspace-class.ts +14 -0
  176. package/src/workspace-cluster.ts +2 -4
  177. package/src/workspace-instance.ts +20 -0
  178. package/src/wsready.ts +5 -4
  179. package/lib/email-protocol.d.ts +0 -49
  180. package/lib/email-protocol.d.ts.map +0 -1
  181. package/lib/email-protocol.js +0 -28
  182. package/lib/email-protocol.js.map +0 -1
  183. package/src/email-protocol.ts +0 -65
package/src/protocol.ts CHANGED
@@ -8,6 +8,7 @@ import { WorkspaceInstance, PortVisibility } from "./workspace-instance";
8
8
  import { RoleOrPermission } from "./permission";
9
9
  import { Project } from "./teams-projects-protocol";
10
10
  import { createHash } from "crypto";
11
+ import { AttributionId } from "./attribution";
11
12
 
12
13
  export interface UserInfo {
13
14
  name?: string;
@@ -45,6 +46,15 @@ export interface User {
45
46
  markedDeleted?: boolean;
46
47
 
47
48
  additionalData?: AdditionalUserData;
49
+
50
+ // Identifies an explicit team or user ID to which all the user's workspace usage should be attributed to (e.g. for billing purposes)
51
+ usageAttributionId?: string;
52
+
53
+ // The last time this user got verified somehow. The user is not verified if this is empty.
54
+ lastVerificationTime?: string;
55
+
56
+ // The phone number used for the last phone verification.
57
+ verificationPhoneNumber?: string;
48
58
  }
49
59
 
50
60
  export namespace User {
@@ -70,11 +80,14 @@ export namespace User {
70
80
  }
71
81
 
72
82
  /**
73
- * Tries to return the primaryEmail of the first identity this user signed up with.
83
+ * Returns the stored email or if it doesn't exist returns the primaryEmail of the first identity this user signed up with.
74
84
  * @param user
75
85
  * @returns A primaryEmail, or undefined if there is none.
76
86
  */
77
87
  export function getPrimaryEmail(user: User): string | undefined {
88
+ if (user.additionalData?.profile?.emailAddress) {
89
+ return user.additionalData?.profile?.emailAddress;
90
+ }
78
91
  const identities = user.identities.filter((i) => !!i.primaryEmail);
79
92
  if (identities.length <= 0) {
80
93
  return undefined;
@@ -136,6 +149,50 @@ export namespace User {
136
149
  }
137
150
  user.additionalData.ideSettings = newIDESettings;
138
151
  }
152
+
153
+ export function getProfile(user: User): Profile {
154
+ return {
155
+ name: User.getName(user!) || "",
156
+ email: User.getPrimaryEmail(user!) || "",
157
+ company: user?.additionalData?.profile?.companyName,
158
+ avatarURL: user?.avatarUrl,
159
+ };
160
+ }
161
+
162
+ export function setProfile(user: User, profile: Profile): User {
163
+ user.fullName = profile.name;
164
+ user.avatarUrl = profile.avatarURL;
165
+
166
+ if (!user.additionalData) {
167
+ user.additionalData = {};
168
+ }
169
+ if (!user.additionalData.profile) {
170
+ user.additionalData.profile = {};
171
+ }
172
+ user.additionalData.profile.emailAddress = profile.email;
173
+ user.additionalData.profile.companyName = profile.company;
174
+ user.additionalData.profile.lastUpdatedDetailsNudge = new Date().toISOString();
175
+
176
+ return user;
177
+ }
178
+
179
+ // The actual Profile of a User
180
+ export interface Profile {
181
+ name: string;
182
+ email: string;
183
+ company?: string;
184
+ avatarURL?: string;
185
+ }
186
+ export namespace Profile {
187
+ export function hasChanges(before: Profile, after: Profile) {
188
+ return (
189
+ before.name !== after.name ||
190
+ before.email !== after.email ||
191
+ before.company !== after.company ||
192
+ before.avatarURL !== after.avatarURL
193
+ );
194
+ }
195
+ }
139
196
  }
140
197
 
141
198
  export interface AdditionalUserData {
@@ -150,9 +207,36 @@ export interface AdditionalUserData {
150
207
  oauthClientsApproved?: { [key: string]: string };
151
208
  // to remember GH Orgs the user installed/updated the GH App for
152
209
  knownGitHubOrgs?: string[];
153
-
154
210
  // Git clone URL pointing to the user's dotfile repo
155
211
  dotfileRepo?: string;
212
+ // preferred workspace classes
213
+ workspaceClasses?: WorkspaceClasses;
214
+ // additional user profile data
215
+ profile?: ProfileDetails;
216
+ }
217
+ export namespace AdditionalUserData {
218
+ export function set(user: User, partialData: Partial<AdditionalUserData>): User {
219
+ if (!user.additionalData) {
220
+ user.additionalData = {
221
+ ...partialData,
222
+ };
223
+ } else {
224
+ user.additionalData = {
225
+ ...user.additionalData,
226
+ ...partialData,
227
+ };
228
+ }
229
+ return user;
230
+ }
231
+ }
232
+ // The format in which we store User Profiles in
233
+ export interface ProfileDetails {
234
+ // when was the last time the user updated their profile information or has been nudged to do so.
235
+ lastUpdatedDetailsNudge?: string;
236
+ // the user's company name
237
+ companyName?: string;
238
+ // the user's email
239
+ emailAddress?: string;
156
240
  }
157
241
 
158
242
  export interface EmailNotificationSettings {
@@ -171,6 +255,11 @@ export type IDESettings = {
171
255
  useLatestVersion?: boolean;
172
256
  };
173
257
 
258
+ export interface WorkspaceClasses {
259
+ regular: string;
260
+ prebuild: string;
261
+ }
262
+
174
263
  export interface UserPlatform {
175
264
  uid: string;
176
265
  userAgent: string;
@@ -197,16 +286,29 @@ export interface UserFeatureSettings {
197
286
  permanentWSFeatureFlags?: NamedWorkspaceFeatureFlag[];
198
287
  }
199
288
 
289
+ export type BillingTier = "paid" | "free";
290
+
200
291
  /**
201
292
  * The values of this type MUST MATCH enum values in WorkspaceFeatureFlag from ws-manager/client/core_pb.d.ts
202
293
  * If they don't we'll break things during workspace startup.
203
294
  */
204
295
  export const WorkspaceFeatureFlags = {
205
296
  full_workspace_backup: undefined,
206
- fixed_resources: undefined,
207
297
  persistent_volume_claim: undefined,
298
+ protected_secrets: undefined,
299
+ workspace_class_limiting: undefined,
300
+ workspace_connection_limiting: undefined,
208
301
  };
209
302
  export type NamedWorkspaceFeatureFlag = keyof typeof WorkspaceFeatureFlags;
303
+ export namespace NamedWorkspaceFeatureFlag {
304
+ export const WORKSPACE_PERSISTED_FEATTURE_FLAGS: NamedWorkspaceFeatureFlag[] = [
305
+ "full_workspace_backup",
306
+ "persistent_volume_claim",
307
+ ];
308
+ export function isWorkspacePersisted(ff: NamedWorkspaceFeatureFlag): boolean {
309
+ return WORKSPACE_PERSISTED_FEATTURE_FLAGS.includes(ff);
310
+ }
311
+ }
210
312
 
211
313
  export interface EnvVarWithValue {
212
314
  name: string;
@@ -243,7 +345,7 @@ export namespace UserEnvVar {
243
345
  return "Name must not be empty.";
244
346
  }
245
347
  if (name.length > 255) {
246
- return 'Name too long. Maximum name length is 255 characters.';
348
+ return "Name too long. Maximum name length is 255 characters.";
247
349
  }
248
350
  if (!/^[a-zA-Z_]+[a-zA-Z0-9_]*$/.test(name)) {
249
351
  return "Name must match /^[a-zA-Z_]+[a-zA-Z0-9_]*$/.";
@@ -252,7 +354,7 @@ export namespace UserEnvVar {
252
354
  return "Value must not be empty.";
253
355
  }
254
356
  if (variable.value.length > 32767) {
255
- return 'Value too long. Maximum value length is 32767 characters.';
357
+ return "Value too long. Maximum value length is 32767 characters.";
256
358
  }
257
359
  if (pattern.trim() === "") {
258
360
  return "Scope must not be empty.";
@@ -356,6 +458,68 @@ export namespace UserEnvVar {
356
458
  }
357
459
  }
358
460
 
461
+ export interface SSHPublicKeyValue {
462
+ name: string;
463
+ key: string;
464
+ }
465
+ export interface UserSSHPublicKey extends SSHPublicKeyValue {
466
+ id: string;
467
+ key: string;
468
+ userId: string;
469
+ fingerprint: string;
470
+ creationTime: string;
471
+ lastUsedTime?: string;
472
+ }
473
+
474
+ export type UserSSHPublicKeyValue = Omit<UserSSHPublicKey, "key" | "userId">;
475
+
476
+ export namespace SSHPublicKeyValue {
477
+ export function validate(value: SSHPublicKeyValue): string | undefined {
478
+ if (value.name.length === 0) {
479
+ return "Title must not be empty.";
480
+ }
481
+ if (value.name.length > 255) {
482
+ return "Title too long. Maximum value length is 255 characters.";
483
+ }
484
+ if (value.key.length === 0) {
485
+ return "Key must not be empty.";
486
+ }
487
+ try {
488
+ getData(value);
489
+ } catch (e) {
490
+ return "Key is invalid. You must supply a key in OpenSSH public key format.";
491
+ }
492
+ return;
493
+ }
494
+
495
+ export function getData(value: SSHPublicKeyValue) {
496
+ // Begins with 'ssh-rsa', 'ecdsa-sha2-nistp256', 'ecdsa-sha2-nistp384', 'ecdsa-sha2-nistp521', 'ssh-ed25519', 'sk-ecdsa-sha2-nistp256@openssh.com', or 'sk-ssh-ed25519@openssh.com'.
497
+ const regex =
498
+ /^(?<type>ssh-rsa|ecdsa-sha2-nistp256|ecdsa-sha2-nistp384|ecdsa-sha2-nistp521|ssh-ed25519|sk-ecdsa-sha2-nistp256@openssh\.com|sk-ssh-ed25519@openssh\.com) (?<key>.*?)( (?<email>.*?))?$/;
499
+ const resultGroup = regex.exec(value.key.trim());
500
+ if (!resultGroup) {
501
+ throw new Error("Key is invalid.");
502
+ }
503
+ return {
504
+ type: resultGroup.groups?.["type"] as string,
505
+ key: resultGroup.groups?.["key"] as string,
506
+ email: resultGroup.groups?.["email"] || undefined,
507
+ };
508
+ }
509
+
510
+ export function getFingerprint(value: SSHPublicKeyValue) {
511
+ const data = getData(value);
512
+ let buf = Buffer.from(data.key, "base64");
513
+ // gitlab style
514
+ // const hash = createHash("md5").update(buf).digest("hex");
515
+ // github style
516
+ const hash = createHash("sha256").update(buf).digest("base64");
517
+ return hash;
518
+ }
519
+
520
+ export const MAXIMUM_KEY_LENGTH = 5;
521
+ }
522
+
359
523
  export interface GitpodToken {
360
524
  /** Hash value (SHA256) of the token (primary key). */
361
525
  tokenHash: string;
@@ -488,6 +652,13 @@ export interface Snapshot {
488
652
  message?: string;
489
653
  }
490
654
 
655
+ export interface VolumeSnapshot {
656
+ id: string;
657
+ workspaceId: string;
658
+ creationTime: string;
659
+ volumeHandle: string;
660
+ }
661
+
491
662
  export type SnapshotState = "pending" | "available" | "error";
492
663
 
493
664
  export interface LayoutData {
@@ -557,7 +728,7 @@ export interface Workspace {
557
728
 
558
729
  export type WorkspaceSoftDeletion = "user" | "gc";
559
730
 
560
- export type WorkspaceType = "regular" | "prebuild" | "probe";
731
+ export type WorkspaceType = "regular" | "prebuild";
561
732
 
562
733
  export namespace Workspace {
563
734
  export function getFullRepositoryName(ws: Workspace): string | undefined {
@@ -633,6 +804,7 @@ export interface JetBrainsConfig {
633
804
  }
634
805
  export interface JetBrainsProductConfig {
635
806
  prebuilds?: JetBrainsPrebuilds;
807
+ vmoptions?: string;
636
808
  }
637
809
  export interface JetBrainsPrebuilds {
638
810
  version?: "stable" | "latest" | "both";
@@ -643,6 +815,12 @@ export interface RepositoryCloneInformation {
643
815
  checkoutLocation?: string;
644
816
  }
645
817
 
818
+ export interface CoreDumpConfig {
819
+ enabled?: boolean;
820
+ softLimit?: number;
821
+ hardLimit?: number;
822
+ }
823
+
646
824
  export interface WorkspaceConfig {
647
825
  mainConfiguration?: string;
648
826
  additionalRepositories?: RepositoryCloneInformation[];
@@ -655,6 +833,7 @@ export interface WorkspaceConfig {
655
833
  github?: GithubAppConfig;
656
834
  vscode?: VSCodeConfig;
657
835
  jetbrains?: JetBrainsConfig;
836
+ coreDump?: CoreDumpConfig;
658
837
 
659
838
  /** deprecated. Enabled by default **/
660
839
  experimentalNetwork?: boolean;
@@ -663,13 +842,12 @@ export interface WorkspaceConfig {
663
842
  * Where the config object originates from.
664
843
  *
665
844
  * repo - from the repository
666
- * project-db - from the "Project" stored in the database
667
845
  * definitly-gp - from github.com/gitpod-io/definitely-gp
668
846
  * derived - computed based on analyzing the repository
669
847
  * additional-content - config comes from additional content, usually provided through the project's configuration
670
848
  * default - our static catch-all default config
671
849
  */
672
- _origin?: "repo" | "project-db" | "definitely-gp" | "derived" | "additional-content" | "default";
850
+ _origin?: "repo" | "definitely-gp" | "derived" | "additional-content" | "default";
673
851
 
674
852
  /**
675
853
  * Set of automatically infered feature flags. That's not something the user can set, but
@@ -748,7 +926,9 @@ export interface PrebuiltWorkspace {
748
926
 
749
927
  export namespace PrebuiltWorkspace {
750
928
  export function isDone(pws: PrebuiltWorkspace) {
751
- return pws.state === "available" || pws.state === "timeout" || pws.state === "aborted";
929
+ return (
930
+ pws.state === "available" || pws.state === "timeout" || pws.state === "aborted" || pws.state === "failed"
931
+ );
752
932
  }
753
933
 
754
934
  export function isAvailable(pws: PrebuiltWorkspace) {
@@ -872,6 +1052,7 @@ export namespace ExternalImageConfigFile {
872
1052
 
873
1053
  export interface WorkspaceContext {
874
1054
  title: string;
1055
+ ref?: string;
875
1056
  /** This contains the URL portion of the contextURL (which might contain other modifiers as well). It's optional because it's not set for older workspaces. */
876
1057
  normalizedContextURL?: string;
877
1058
  forceCreateNewWorkspace?: boolean;
@@ -984,17 +1165,6 @@ export namespace WithEnvvarsContext {
984
1165
  }
985
1166
  }
986
1167
 
987
- export interface WorkspaceProbeContext extends WorkspaceContext {
988
- responseURL: string;
989
- responseToken: string;
990
- }
991
-
992
- export namespace WorkspaceProbeContext {
993
- export function is(context: any): context is WorkspaceProbeContext {
994
- return context && "responseURL" in context && "responseToken" in context;
995
- }
996
- }
997
-
998
1168
  export type RefType = "branch" | "tag" | "revision";
999
1169
  export namespace RefType {
1000
1170
  export const getRefType = (commit: Commit): RefType => {
@@ -1340,3 +1510,18 @@ export interface Terms {
1340
1510
  readonly content: string;
1341
1511
  readonly formElements?: object;
1342
1512
  }
1513
+
1514
+ export interface StripeConfig {
1515
+ individualUsagePriceIds: { [currency: string]: string };
1516
+ teamUsagePriceIds: { [currency: string]: string };
1517
+ }
1518
+
1519
+ export type BillingStrategy = "other" | "stripe";
1520
+ export interface CostCenter {
1521
+ readonly id: AttributionId;
1522
+ /**
1523
+ * Unit: credits
1524
+ */
1525
+ spendingLimit: number;
1526
+ billingStrategy: BillingStrategy;
1527
+ }
@@ -33,6 +33,33 @@ export namespace TeamSubscription {
33
33
  };
34
34
  }
35
35
 
36
+ export interface TeamSubscription2 {
37
+ id: string;
38
+ teamId: string;
39
+ planId: string;
40
+ startDate: string;
41
+ endDate?: string;
42
+ quantity: number;
43
+ /** The Chargebee subscription id */
44
+ paymentReference: string;
45
+ cancellationDate?: string;
46
+ excludeFromMoreResources: boolean;
47
+ }
48
+
49
+ export namespace TeamSubscription2 {
50
+ export const create = (ts2: Omit<TeamSubscription2, "id">): TeamSubscription2 => {
51
+ const withId = ts2 as TeamSubscription2;
52
+ withId.id = uuidv4();
53
+ return withId;
54
+ };
55
+ export const isActive = (ts2: TeamSubscription2, date: string): boolean => {
56
+ return ts2.startDate <= date && (ts2.endDate === undefined || date < ts2.endDate);
57
+ };
58
+ export function isCancelled(s: TeamSubscription2, date: string): boolean {
59
+ return (!!s.cancellationDate && s.cancellationDate < date) || (!!s.endDate && s.endDate < date); // This edge case is meant to handle bad data: If for whatever reason cancellationDate has not been set: treat endDate as such
60
+ }
61
+ }
62
+
36
63
  /**
37
64
  * A slot represents one unit of a TeamSubscription that gets assigned to one user at a time
38
65
  */
@@ -7,6 +7,7 @@
7
7
  import { PrebuiltWorkspaceState } from "./protocol";
8
8
  import { v4 as uuidv4 } from "uuid";
9
9
  import { DeepPartial } from "./util/deep-partial";
10
+ import { WebhookEvent } from "./webhook-event";
10
11
 
11
12
  export interface ProjectConfig {
12
13
  ".gitpod.yml": string;
@@ -14,6 +15,8 @@ export interface ProjectConfig {
14
15
 
15
16
  export interface ProjectSettings {
16
17
  useIncrementalPrebuilds?: boolean;
18
+ usePersistentVolumeClaim?: boolean;
19
+ keepOutdatedPrebuildsRunning?: boolean;
17
20
  }
18
21
 
19
22
  export interface Project {
@@ -24,7 +27,6 @@ export interface Project {
24
27
  teamId?: string;
25
28
  userId?: string;
26
29
  appInstallationId: string;
27
- config?: ProjectConfig;
28
30
  settings?: ProjectSettings;
29
31
  creationTime: string;
30
32
  /** This is a flag that triggers the HARD DELETION of this entity */
@@ -150,3 +152,15 @@ export interface TeamMembershipInvite {
150
152
  /** This is a flag that triggers the HARD DELETION of this entity */
151
153
  deleted?: boolean;
152
154
  }
155
+
156
+ export interface PrebuildEvent {
157
+ id: string;
158
+ creationTime: string;
159
+ status: WebhookEvent.Status | WebhookEvent.PrebuildStatus;
160
+ message?: string;
161
+ prebuildId?: string;
162
+ projectId?: string;
163
+ cloneUrl?: string;
164
+ branch?: string;
165
+ commit?: string;
166
+ }
package/src/usage.ts ADDED
@@ -0,0 +1,71 @@
1
+ /**
2
+ * Copyright (c) 2022 Gitpod GmbH. All rights reserved.
3
+ * Licensed under the GNU Affero General Public License (AGPL).
4
+ * See License-AGPL.txt in the project root for license information.
5
+ */
6
+
7
+ import { WorkspaceType } from "./protocol";
8
+
9
+ // types below are copied over from components/usage-api/typescript/src/usage/v1/usage_pb.d.ts
10
+ export interface ListUsageRequest {
11
+ attributionId: string;
12
+ from?: number;
13
+ to?: number;
14
+ order: Ordering;
15
+ pagination?: PaginationRequest;
16
+ }
17
+
18
+ export enum Ordering {
19
+ ORDERING_DESCENDING = 0,
20
+ ORDERING_ASCENDING = 1,
21
+ }
22
+
23
+ export interface PaginationRequest {
24
+ perPage: number;
25
+ page: number;
26
+ }
27
+
28
+ export interface ListUsageResponse {
29
+ usageEntriesList: Usage[];
30
+ pagination?: PaginationResponse;
31
+ creditsUsed: number;
32
+ }
33
+
34
+ export interface PaginationResponse {
35
+ perPage: number;
36
+ totalPages: number;
37
+ total: number;
38
+ page: number;
39
+ }
40
+
41
+ export type UsageKind = "workspaceinstance" | "invoice";
42
+ export interface Usage {
43
+ id: string;
44
+ attributionId: string;
45
+ description: string;
46
+ credits: number;
47
+ effectiveTime?: number;
48
+ kind: UsageKind;
49
+ workspaceInstanceId: string;
50
+ draft: boolean;
51
+ metadata: WorkspaceInstanceUsageData | InvoiceUsageData;
52
+ }
53
+
54
+ // the equivalent golang shape is maintained in `/workspace/gitpod/`components/usage/pkg/db/usage.go`
55
+ export interface WorkspaceInstanceUsageData {
56
+ workspaceId: string;
57
+ workspaceType: WorkspaceType;
58
+ workspaceClass: string;
59
+ contextURL: string;
60
+ startTime: string;
61
+ endTime?: string;
62
+ userId: string;
63
+ userName: string;
64
+ userAvatarURL: string;
65
+ }
66
+
67
+ export interface InvoiceUsageData {
68
+ invoiceId: string;
69
+ startDate: string;
70
+ endDate: string;
71
+ }
@@ -0,0 +1,81 @@
1
+ /**
2
+ * Copyright (c) 2021 Gitpod GmbH. All rights reserved.
3
+ * Licensed under the GNU Affero General Public License (AGPL).
4
+ * See License-AGPL.txt in the project root for license information.
5
+ */
6
+
7
+ import * as http from "http";
8
+ import * as express from "express";
9
+ import { injectable, postConstruct } from "inversify";
10
+ import { log, LogrusLogLevel } from "./logging";
11
+
12
+ export interface SetLogLevelRequest {
13
+ level: LogrusLogLevel;
14
+ }
15
+ export namespace SetLogLevelRequest {
16
+ export function is(o: any): o is SetLogLevelRequest {
17
+ return typeof o === "object" && "level" in o;
18
+ }
19
+ }
20
+
21
+ @injectable()
22
+ export class DebugApp {
23
+ protected _app: express.Application;
24
+ protected httpServer: http.Server | undefined = undefined;
25
+
26
+ @postConstruct()
27
+ public ctor() {
28
+ this._app = this.create();
29
+ }
30
+
31
+ create(): express.Application {
32
+ const app = express();
33
+
34
+ app.use(express.json());
35
+ app.use(express.urlencoded({ extended: true }));
36
+
37
+ app.post("/debug/logging", (req, res) => {
38
+ try {
39
+ const levelRequest = req.body;
40
+ if (!SetLogLevelRequest.is(levelRequest)) {
41
+ res.status(400).end("not a SetLogLevelRequest");
42
+ return;
43
+ }
44
+
45
+ const newLogLevel = levelRequest.level;
46
+ log.setLogLevel(newLogLevel);
47
+ log.info("set log level", { newLogLevel });
48
+ res.status(200).end(JSON.stringify(levelRequest));
49
+ } catch (err) {
50
+ res.status(500).end(err);
51
+ }
52
+ });
53
+ return app;
54
+ }
55
+
56
+ public start(port: number = 6060) {
57
+ this.httpServer = this._app.listen(port, "localhost", () => {
58
+ log.info(`debug server listening on port: ${port}`);
59
+ });
60
+ }
61
+
62
+ public async stop() {
63
+ const server = this.httpServer;
64
+ if (!server) {
65
+ return;
66
+ }
67
+ return new Promise<void>((resolve) =>
68
+ server.close((err: any) => {
69
+ if (err) {
70
+ log.warn(`error while closing http server`, { err });
71
+ }
72
+ this.httpServer = undefined;
73
+ resolve();
74
+ }),
75
+ );
76
+ }
77
+
78
+ public get app(): express.Application {
79
+ return this._app;
80
+ }
81
+ }
@@ -37,6 +37,11 @@ export class GarbageCollectedCache<T> {
37
37
  if (!entry) {
38
38
  return undefined;
39
39
  }
40
+ // Still valid?
41
+ if (entry.expiryDate < Date.now()) {
42
+ this.store.delete(entry.key);
43
+ return undefined;
44
+ }
40
45
  return entry.value;
41
46
  }
42
47
 
@@ -85,6 +85,10 @@ export class GitpodHostUrl {
85
85
  return this.with((url) => ({ protocol: url.protocol === "https:" ? "wss:" : "ws:" }));
86
86
  }
87
87
 
88
+ asWorkspacePage(): GitpodHostUrl {
89
+ return this.with((url) => ({ pathname: "/workspaces" }));
90
+ }
91
+
88
92
  asDashboard(): GitpodHostUrl {
89
93
  return this.with((url) => ({ pathname: "/" }));
90
94
  }
@@ -171,4 +175,21 @@ export class GitpodHostUrl {
171
175
  asApiLogout(): GitpodHostUrl {
172
176
  return this.withApi((url) => ({ pathname: "/logout/" }));
173
177
  }
178
+
179
+ asIDEProxy(): GitpodHostUrl {
180
+ const hostSegments = this.url.host.split(".");
181
+ if (hostSegments[0] === "ide") {
182
+ return this;
183
+ }
184
+ return this.with((url) => ({ host: "ide." + url.host }));
185
+ }
186
+
187
+ asIDEMetrics(): GitpodHostUrl {
188
+ let newUrl: GitpodHostUrl = this;
189
+ const hostSegments = this.url.host.split(".");
190
+ if (hostSegments[0] !== "ide") {
191
+ newUrl = newUrl.asIDEProxy();
192
+ }
193
+ return newUrl.with((url) => ({ pathname: "/metrics-api" }));
194
+ }
174
195
  }