@fluidframework/local-driver 1.4.0-121020 → 2.0.0-dev-rc.1.0.0.225277

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 (124) hide show
  1. package/.eslintrc.js +19 -22
  2. package/.mocharc.js +12 -0
  3. package/CHANGELOG.md +127 -0
  4. package/README.md +38 -0
  5. package/api-extractor-esm.json +4 -0
  6. package/api-extractor-lint.json +4 -0
  7. package/api-extractor.json +4 -0
  8. package/api-report/local-driver.api.md +129 -0
  9. package/dist/{auth.js → auth.cjs} +8 -6
  10. package/dist/auth.cjs.map +1 -0
  11. package/dist/auth.d.ts.map +1 -1
  12. package/dist/index.cjs +24 -0
  13. package/dist/index.cjs.map +1 -0
  14. package/dist/index.d.ts +7 -6
  15. package/dist/index.d.ts.map +1 -1
  16. package/dist/local-driver-alpha.d.ts +110 -0
  17. package/dist/local-driver-beta.d.ts +75 -0
  18. package/dist/local-driver-public.d.ts +75 -0
  19. package/dist/local-driver-untrimmed.d.ts +226 -0
  20. package/dist/localCreateDocument.cjs +28 -0
  21. package/dist/localCreateDocument.cjs.map +1 -0
  22. package/dist/localCreateDocument.d.ts +8 -0
  23. package/dist/localCreateDocument.d.ts.map +1 -0
  24. package/dist/{localDeltaStorageService.js → localDeltaStorageService.cjs} +3 -2
  25. package/dist/localDeltaStorageService.cjs.map +1 -0
  26. package/dist/localDeltaStorageService.d.ts +4 -3
  27. package/dist/localDeltaStorageService.d.ts.map +1 -1
  28. package/dist/{localDocumentDeltaConnection.js → localDocumentDeltaConnection.cjs} +9 -11
  29. package/dist/localDocumentDeltaConnection.cjs.map +1 -0
  30. package/dist/localDocumentDeltaConnection.d.ts +5 -4
  31. package/dist/localDocumentDeltaConnection.d.ts.map +1 -1
  32. package/dist/{localDocumentService.js → localDocumentService.cjs} +15 -30
  33. package/dist/localDocumentService.cjs.map +1 -0
  34. package/dist/localDocumentService.d.ts +15 -11
  35. package/dist/localDocumentService.d.ts.map +1 -1
  36. package/dist/{localDocumentServiceFactory.js → localDocumentServiceFactory.cjs} +14 -31
  37. package/dist/localDocumentServiceFactory.cjs.map +1 -0
  38. package/dist/localDocumentServiceFactory.d.ts +3 -2
  39. package/dist/localDocumentServiceFactory.d.ts.map +1 -1
  40. package/dist/localDocumentStorageService.cjs +88 -0
  41. package/dist/localDocumentStorageService.cjs.map +1 -0
  42. package/dist/localDocumentStorageService.d.ts +30 -0
  43. package/dist/localDocumentStorageService.d.ts.map +1 -0
  44. package/dist/{localResolver.js → localResolver.cjs} +10 -6
  45. package/dist/localResolver.cjs.map +1 -0
  46. package/dist/localResolver.d.ts +5 -0
  47. package/dist/localResolver.d.ts.map +1 -1
  48. package/dist/{localSessionStorageDb.js → localSessionStorageDb.cjs} +26 -18
  49. package/dist/localSessionStorageDb.cjs.map +1 -0
  50. package/dist/localSessionStorageDb.d.ts +1 -1
  51. package/dist/localSessionStorageDb.d.ts.map +1 -1
  52. package/dist/tsdoc-metadata.json +11 -0
  53. package/lib/auth.d.mts +12 -0
  54. package/lib/auth.d.mts.map +1 -0
  55. package/lib/auth.mjs +40 -0
  56. package/lib/auth.mjs.map +1 -0
  57. package/lib/index.d.mts +12 -0
  58. package/lib/index.d.mts.map +1 -0
  59. package/lib/index.mjs +12 -0
  60. package/lib/index.mjs.map +1 -0
  61. package/lib/local-driver-alpha.d.mts +110 -0
  62. package/lib/local-driver-beta.d.mts +75 -0
  63. package/lib/local-driver-public.d.mts +75 -0
  64. package/lib/local-driver-untrimmed.d.mts +226 -0
  65. package/lib/localCreateDocument.d.mts +8 -0
  66. package/lib/localCreateDocument.d.mts.map +1 -0
  67. package/lib/localCreateDocument.mjs +24 -0
  68. package/lib/localCreateDocument.mjs.map +1 -0
  69. package/lib/localDeltaStorageService.d.mts +20 -0
  70. package/lib/localDeltaStorageService.d.mts.map +1 -0
  71. package/lib/localDeltaStorageService.mjs +32 -0
  72. package/lib/localDeltaStorageService.mjs.map +1 -0
  73. package/lib/localDocumentDeltaConnection.d.mts +48 -0
  74. package/lib/localDocumentDeltaConnection.d.mts.map +1 -0
  75. package/lib/localDocumentDeltaConnection.mjs +86 -0
  76. package/lib/localDocumentDeltaConnection.mjs.map +1 -0
  77. package/lib/localDocumentService.d.mts +56 -0
  78. package/lib/localDocumentService.d.mts.map +1 -0
  79. package/lib/localDocumentService.mjs +84 -0
  80. package/lib/localDocumentService.mjs.map +1 -0
  81. package/lib/localDocumentServiceFactory.d.mts +45 -0
  82. package/lib/localDocumentServiceFactory.d.mts.map +1 -0
  83. package/lib/localDocumentServiceFactory.mjs +80 -0
  84. package/lib/localDocumentServiceFactory.mjs.map +1 -0
  85. package/lib/localDocumentStorageService.d.mts +30 -0
  86. package/lib/localDocumentStorageService.d.mts.map +1 -0
  87. package/lib/localDocumentStorageService.mjs +84 -0
  88. package/lib/localDocumentStorageService.mjs.map +1 -0
  89. package/lib/localResolver.d.mts +31 -0
  90. package/lib/localResolver.d.mts.map +1 -0
  91. package/lib/localResolver.mjs +74 -0
  92. package/lib/localResolver.mjs.map +1 -0
  93. package/lib/localSessionStorageDb.d.mts +11 -0
  94. package/lib/localSessionStorageDb.d.mts.map +1 -0
  95. package/lib/localSessionStorageDb.mjs +292 -0
  96. package/lib/localSessionStorageDb.mjs.map +1 -0
  97. package/package.json +140 -59
  98. package/prettier.config.cjs +8 -0
  99. package/src/auth.ts +42 -33
  100. package/src/index.ts +7 -6
  101. package/src/localCreateDocument.ts +48 -0
  102. package/src/localDeltaStorageService.ts +28 -27
  103. package/src/localDocumentDeltaConnection.ts +91 -88
  104. package/src/localDocumentService.ts +117 -97
  105. package/src/localDocumentServiceFactory.ts +95 -123
  106. package/src/localDocumentStorageService.ts +131 -0
  107. package/src/localResolver.ts +57 -59
  108. package/src/localSessionStorageDb.ts +272 -265
  109. package/tsc-multi.test.json +4 -0
  110. package/tsconfig.json +11 -13
  111. package/dist/auth.js.map +0 -1
  112. package/dist/index.js +0 -23
  113. package/dist/index.js.map +0 -1
  114. package/dist/localDeltaStorageService.js.map +0 -1
  115. package/dist/localDocumentDeltaConnection.js.map +0 -1
  116. package/dist/localDocumentService.js.map +0 -1
  117. package/dist/localDocumentServiceFactory.js.map +0 -1
  118. package/dist/localResolver.js.map +0 -1
  119. package/dist/localSessionStorageDb.js.map +0 -1
  120. package/dist/packageVersion.d.ts +0 -9
  121. package/dist/packageVersion.d.ts.map +0 -1
  122. package/dist/packageVersion.js +0 -12
  123. package/dist/packageVersion.js.map +0 -1
  124. package/src/packageVersion.ts +0 -9
@@ -3,100 +3,116 @@
3
3
  * Licensed under the MIT License.
4
4
  */
5
5
 
6
- import * as api from "@fluidframework/driver-definitions";
6
+ import {
7
+ IDocumentDeltaConnection,
8
+ IDocumentDeltaStorageService,
9
+ IDocumentService,
10
+ IDocumentServicePolicies,
11
+ IDocumentStorageService,
12
+ IResolvedUrl,
13
+ } from "@fluidframework/driver-definitions";
7
14
  import { IClient } from "@fluidframework/protocol-definitions";
8
- import * as socketStorage from "@fluidframework/routerlicious-driver";
15
+ import { ITokenProvider } from "@fluidframework/routerlicious-driver";
9
16
  import { GitManager } from "@fluidframework/server-services-client";
10
17
  import { TestHistorian } from "@fluidframework/server-test-utils";
11
18
  import { ILocalDeltaConnectionServer } from "@fluidframework/server-local-server";
12
- import { TelemetryNullLogger } from "@fluidframework/common-utils";
13
- import { LocalDeltaStorageService, LocalDocumentDeltaConnection } from ".";
19
+ import { ITelemetryBaseLogger } from "@fluidframework/core-interfaces";
20
+ import { LocalDocumentStorageService } from "./localDocumentStorageService";
21
+ import { LocalDocumentDeltaConnection } from "./localDocumentDeltaConnection";
22
+ import { LocalDeltaStorageService } from "./localDeltaStorageService";
14
23
 
15
24
  /**
16
25
  * Basic implementation of a document service for local use.
26
+ * @internal
17
27
  */
18
- export class LocalDocumentService implements api.IDocumentService {
19
- /**
20
- * @param localDeltaConnectionServer - delta connection server for ops
21
- * @param tokenProvider - token provider
22
- * @param tenantId - ID of tenant
23
- * @param documentId - ID of document
24
- */
25
- constructor(
26
- public readonly resolvedUrl: api.IResolvedUrl,
27
- private readonly localDeltaConnectionServer: ILocalDeltaConnectionServer,
28
- private readonly tokenProvider: socketStorage.ITokenProvider,
29
- private readonly tenantId: string,
30
- private readonly documentId: string,
31
- private readonly documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection>,
32
- public readonly policies: api.IDocumentServicePolicies = {},
33
- private readonly innerDocumentService?: api.IDocumentService,
34
- ) { }
28
+ export class LocalDocumentService implements IDocumentService {
29
+ /**
30
+ * @param localDeltaConnectionServer - delta connection server for ops
31
+ * @param tokenProvider - token provider
32
+ * @param tenantId - ID of tenant
33
+ * @param documentId - ID of document
34
+ */
35
+ constructor(
36
+ public readonly resolvedUrl: IResolvedUrl,
37
+ private readonly localDeltaConnectionServer: ILocalDeltaConnectionServer,
38
+ private readonly tokenProvider: ITokenProvider,
39
+ private readonly tenantId: string,
40
+ private readonly documentId: string,
41
+ private readonly documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection>,
42
+ public readonly policies: IDocumentServicePolicies = {},
43
+ private readonly innerDocumentService?: IDocumentService,
44
+ private readonly logger?: ITelemetryBaseLogger,
45
+ ) {}
35
46
 
36
- public dispose() { }
47
+ public dispose() {}
37
48
 
38
- /**
39
- * Creates and returns a document storage service for local use.
40
- */
41
- public async connectToStorage(): Promise<api.IDocumentStorageService> {
42
- return new socketStorage.DocumentStorageService(
43
- this.documentId,
44
- new GitManager(new TestHistorian(this.localDeltaConnectionServer.testDbFactory.testDatabase)),
45
- new TelemetryNullLogger(),
46
- { minBlobSize: 2048 }, // Test blob aggregation.
47
- undefined,
48
- undefined,
49
- undefined,
50
- new GitManager(new TestHistorian(this.localDeltaConnectionServer.testDbFactory.testDatabase)));
51
- }
49
+ /**
50
+ * Creates and returns a document storage service for local use.
51
+ */
52
+ public async connectToStorage(): Promise<IDocumentStorageService> {
53
+ return new LocalDocumentStorageService(
54
+ this.documentId,
55
+ new GitManager(
56
+ new TestHistorian(this.localDeltaConnectionServer.testDbFactory.testDatabase),
57
+ ),
58
+ {
59
+ maximumCacheDurationMs: 432_000_000, // 5 days in ms. Not actually enforced but shouldn't matter for any local driver scenario
60
+ },
61
+ this.localDeltaConnectionServer,
62
+ this.resolvedUrl,
63
+ );
64
+ }
52
65
 
53
- /**
54
- * Creates and returns a delta storage service for local use.
55
- */
56
- public async connectToDeltaStorage(): Promise<api.IDocumentDeltaStorageService> {
57
- if (this.innerDocumentService) {
58
- return this.innerDocumentService.connectToDeltaStorage();
59
- }
60
- return new LocalDeltaStorageService(
61
- this.tenantId,
62
- this.documentId,
63
- this.localDeltaConnectionServer.databaseManager);
64
- }
66
+ /**
67
+ * Creates and returns a delta storage service for local use.
68
+ */
69
+ public async connectToDeltaStorage(): Promise<IDocumentDeltaStorageService> {
70
+ if (this.innerDocumentService) {
71
+ return this.innerDocumentService.connectToDeltaStorage();
72
+ }
73
+ return new LocalDeltaStorageService(
74
+ this.tenantId,
75
+ this.documentId,
76
+ this.localDeltaConnectionServer.databaseManager,
77
+ );
78
+ }
65
79
 
66
- /**
67
- * Creates and returns a delta stream for local use.
68
- * @param client - client data
69
- */
70
- public async connectToDeltaStream(client: IClient): Promise<api.IDocumentDeltaConnection> {
71
- if (this.policies.storageOnly === true) {
72
- throw new Error("can't connect to delta stream in storage-only mode");
73
- }
74
- if (this.innerDocumentService) {
75
- return this.innerDocumentService.connectToDeltaStream(client);
76
- }
77
- const ordererToken = await this.tokenProvider.fetchOrdererToken(
78
- this.tenantId,
79
- this.documentId,
80
- );
81
- const documentDeltaConnection = await LocalDocumentDeltaConnection.create(
82
- this.tenantId,
83
- this.documentId,
84
- ordererToken.jwt,
85
- client,
86
- this.localDeltaConnectionServer.webSocketServer,
87
- );
88
- const clientId = documentDeltaConnection.clientId;
80
+ /**
81
+ * Creates and returns a delta stream for local use.
82
+ * @param client - client data
83
+ */
84
+ public async connectToDeltaStream(client: IClient): Promise<IDocumentDeltaConnection> {
85
+ if (this.policies.storageOnly === true) {
86
+ throw new Error("can't connect to delta stream in storage-only mode");
87
+ }
88
+ if (this.innerDocumentService) {
89
+ return this.innerDocumentService.connectToDeltaStream(client);
90
+ }
91
+ const ordererToken = await this.tokenProvider.fetchOrdererToken(
92
+ this.tenantId,
93
+ this.documentId,
94
+ );
95
+ const documentDeltaConnection = await LocalDocumentDeltaConnection.create(
96
+ this.tenantId,
97
+ this.documentId,
98
+ ordererToken.jwt,
99
+ client,
100
+ this.localDeltaConnectionServer.webSocketServer,
101
+ undefined,
102
+ this.logger,
103
+ );
104
+ const clientId = documentDeltaConnection.clientId;
89
105
 
90
- // Add this document service for the clientId in the document service factory.
91
- this.documentDeltaConnectionsMap.set(clientId, documentDeltaConnection);
106
+ // Add this document service for the clientId in the document service factory.
107
+ this.documentDeltaConnectionsMap.set(clientId, documentDeltaConnection);
92
108
 
93
- // Add a listener to remove this document service when the client is disconnected.
94
- documentDeltaConnection.on("disconnect", () => {
95
- this.documentDeltaConnectionsMap.delete(clientId);
96
- });
109
+ // Add a listener to remove this document service when the client is disconnected.
110
+ documentDeltaConnection.on("disconnect", () => {
111
+ this.documentDeltaConnectionsMap.delete(clientId);
112
+ });
97
113
 
98
- return documentDeltaConnection;
99
- }
114
+ return documentDeltaConnection;
115
+ }
100
116
  }
101
117
 
102
118
  /**
@@ -105,24 +121,28 @@ export class LocalDocumentService implements api.IDocumentService {
105
121
  * @param tokenProvider - token provider with a single token
106
122
  * @param tenantId - ID of tenant
107
123
  * @param documentId - ID of document
124
+ * @internal
108
125
  */
109
126
  export function createLocalDocumentService(
110
- resolvedUrl: api.IResolvedUrl,
111
- localDeltaConnectionServer: ILocalDeltaConnectionServer,
112
- tokenProvider: socketStorage.ITokenProvider,
113
- tenantId: string,
114
- documentId: string,
115
- documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection>,
116
- policies?: api.IDocumentServicePolicies,
117
- innerDocumentService?: api.IDocumentService): api.IDocumentService {
118
- return new LocalDocumentService(
119
- resolvedUrl,
120
- localDeltaConnectionServer,
121
- tokenProvider,
122
- tenantId,
123
- documentId,
124
- documentDeltaConnectionsMap,
125
- policies,
126
- innerDocumentService,
127
- );
127
+ resolvedUrl: IResolvedUrl,
128
+ localDeltaConnectionServer: ILocalDeltaConnectionServer,
129
+ tokenProvider: ITokenProvider,
130
+ tenantId: string,
131
+ documentId: string,
132
+ documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection>,
133
+ policies?: IDocumentServicePolicies,
134
+ innerDocumentService?: IDocumentService,
135
+ logger?: ITelemetryBaseLogger,
136
+ ): IDocumentService {
137
+ return new LocalDocumentService(
138
+ resolvedUrl,
139
+ localDeltaConnectionServer,
140
+ tokenProvider,
141
+ tenantId,
142
+ documentId,
143
+ documentDeltaConnectionsMap,
144
+ policies,
145
+ innerDocumentService,
146
+ logger,
147
+ );
128
148
  }
@@ -5,145 +5,117 @@
5
5
 
6
6
  import { parse } from "url";
7
7
  import {
8
- IDocumentService,
9
- IDocumentServiceFactory,
10
- IDocumentServicePolicies,
11
- IResolvedUrl,
8
+ IDocumentService,
9
+ IDocumentServiceFactory,
10
+ IDocumentServicePolicies,
11
+ IResolvedUrl,
12
12
  } from "@fluidframework/driver-definitions";
13
- import { ITelemetryBaseLogger } from "@fluidframework/common-definitions";
13
+ import { ITelemetryBaseLogger } from "@fluidframework/core-interfaces";
14
14
  import { DefaultTokenProvider } from "@fluidframework/routerlicious-driver";
15
- import { ILocalDeltaConnectionServer, LocalDeltaConnectionServer } from "@fluidframework/server-local-server";
16
- import {
17
- ensureFluidResolvedUrl,
18
- getDocAttributesFromProtocolSummary,
19
- getQuorumValuesFromProtocolSummary,
20
- } from "@fluidframework/driver-utils";
15
+ import { ILocalDeltaConnectionServer } from "@fluidframework/server-local-server";
21
16
  import { ISummaryTree, NackErrorType } from "@fluidframework/protocol-definitions";
22
- import { defaultHash } from "@fluidframework/server-services-client";
23
17
  import { LocalDocumentDeltaConnection } from "./localDocumentDeltaConnection";
24
18
  import { createLocalDocumentService } from "./localDocumentService";
19
+ import { createDocument } from "./localCreateDocument";
25
20
 
26
21
  /**
27
22
  * Implementation of document service factory for local use.
23
+ * @alpha
28
24
  */
29
25
  export class LocalDocumentServiceFactory implements IDocumentServiceFactory {
30
- public readonly protocolName = "fluid-test:";
31
-
32
- // A map of clientId to LocalDocumentService.
33
- private readonly documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection> = new Map();
34
-
35
- /**
36
- * @param localDeltaConnectionServer - delta connection server for ops
37
- */
38
- constructor(
39
- private readonly localDeltaConnectionServer: ILocalDeltaConnectionServer,
40
- private readonly policies?: IDocumentServicePolicies,
41
- private readonly innerDocumentService?: IDocumentService) { }
42
-
43
- public async createContainer(
44
- createNewSummary: ISummaryTree | undefined,
45
- resolvedUrl: IResolvedUrl,
46
- logger?: ITelemetryBaseLogger,
47
- clientIsSummarizer?: boolean,
48
- ): Promise<IDocumentService> {
49
- ensureFluidResolvedUrl(resolvedUrl);
50
- if (createNewSummary === undefined) {
51
- throw new Error("Empty file summary creation isn't supported in this driver.");
52
- }
53
- const pathName = new URL(resolvedUrl.url).pathname;
54
- const pathArr = pathName.split("/");
55
- const tenantId = pathArr[pathArr.length - 2];
56
- const id = pathArr[pathArr.length - 1];
57
- if (!this.localDeltaConnectionServer) {
58
- throw new Error("Provide the localDeltaConnectionServer!!");
59
- }
60
- const documentStorage = (this.localDeltaConnectionServer as LocalDeltaConnectionServer).documentStorage;
26
+ // A map of clientId to LocalDocumentService.
27
+ private readonly documentDeltaConnectionsMap: Map<string, LocalDocumentDeltaConnection> =
28
+ new Map();
61
29
 
62
- const protocolSummary = createNewSummary.tree[".protocol"] as ISummaryTree;
63
- const appSummary = createNewSummary.tree[".app"] as ISummaryTree;
64
- if (!(protocolSummary && appSummary)) {
65
- throw new Error("Protocol and App Summary required in the full summary");
66
- }
67
- const documentAttributes = getDocAttributesFromProtocolSummary(protocolSummary);
68
- const quorumValues = getQuorumValuesFromProtocolSummary(protocolSummary);
69
- const sequenceNumber = documentAttributes.sequenceNumber;
70
- await documentStorage.createDocument(
71
- tenantId,
72
- id,
73
- appSummary,
74
- sequenceNumber,
75
- documentAttributes.term ?? 1,
76
- defaultHash,
77
- resolvedUrl.endpoints.ordererUrl ?? "",
78
- resolvedUrl.endpoints.storageUrl ?? "",
79
- quorumValues,
80
- false,
81
- );
82
- return this.createDocumentService(resolvedUrl, logger, clientIsSummarizer);
83
- }
30
+ /**
31
+ * @param localDeltaConnectionServer - delta connection server for ops
32
+ * @alpha
33
+ */
34
+ constructor(
35
+ private readonly localDeltaConnectionServer: ILocalDeltaConnectionServer,
36
+ private readonly policies?: IDocumentServicePolicies,
37
+ private readonly innerDocumentService?: IDocumentService,
38
+ ) {}
84
39
 
85
- /**
86
- * Creates and returns a document service for testing using the given resolved
87
- * URL for the tenant ID, document ID, and token.
88
- * @param resolvedUrl - resolved URL of document
89
- */
90
- public async createDocumentService(
91
- resolvedUrl: IResolvedUrl,
92
- logger?: ITelemetryBaseLogger,
93
- clientIsSummarizer?: boolean,
94
- ): Promise<IDocumentService> {
95
- ensureFluidResolvedUrl(resolvedUrl);
40
+ public async createContainer(
41
+ createNewSummary: ISummaryTree | undefined,
42
+ resolvedUrl: IResolvedUrl,
43
+ logger?: ITelemetryBaseLogger,
44
+ clientIsSummarizer?: boolean,
45
+ ): Promise<IDocumentService> {
46
+ if (!this.localDeltaConnectionServer) {
47
+ throw new Error("Provide the localDeltaConnectionServer!!");
48
+ }
49
+ if (createNewSummary !== undefined) {
50
+ await createDocument(this.localDeltaConnectionServer, resolvedUrl, createNewSummary);
51
+ }
52
+ return this.createDocumentService(resolvedUrl, logger, clientIsSummarizer);
53
+ }
96
54
 
97
- const parsedUrl = parse(resolvedUrl.url);
98
- const [, tenantId, documentId] = parsedUrl.path ? parsedUrl.path.split("/") : [];
99
- if (!documentId || !tenantId) {
100
- throw new Error(`Couldn't parse resolved url. [documentId:${documentId}][tenantId:${tenantId}]`);
101
- }
55
+ /**
56
+ * Creates and returns a document service for testing using the given resolved
57
+ * URL for the tenant ID, document ID, and token.
58
+ * @param resolvedUrl - resolved URL of document
59
+ */
60
+ public async createDocumentService(
61
+ resolvedUrl: IResolvedUrl,
62
+ logger?: ITelemetryBaseLogger,
63
+ clientIsSummarizer?: boolean,
64
+ ): Promise<IDocumentService> {
65
+ const parsedUrl = parse(resolvedUrl.url);
66
+ const [, tenantId, documentId] = parsedUrl.path ? parsedUrl.path.split("/") : [];
67
+ if (!documentId || !tenantId) {
68
+ throw new Error(
69
+ `Couldn't parse resolved url. [documentId:${documentId}][tenantId:${tenantId}]`,
70
+ );
71
+ }
102
72
 
103
- const fluidResolvedUrl = resolvedUrl;
104
- const jwtToken = fluidResolvedUrl.tokens.jwt;
105
- if (!jwtToken) {
106
- throw new Error(`Token was not provided.`);
107
- }
73
+ const fluidResolvedUrl = resolvedUrl;
74
+ const jwtToken = fluidResolvedUrl.tokens.jwt;
75
+ if (!jwtToken) {
76
+ throw new Error(`Token was not provided.`);
77
+ }
108
78
 
109
- const tokenProvider = new DefaultTokenProvider(jwtToken);
79
+ const tokenProvider = new DefaultTokenProvider(jwtToken);
110
80
 
111
- return createLocalDocumentService(
112
- resolvedUrl,
113
- this.localDeltaConnectionServer,
114
- tokenProvider,
115
- tenantId,
116
- documentId,
117
- this.documentDeltaConnectionsMap,
118
- this.policies,
119
- this.innerDocumentService);
120
- }
81
+ return createLocalDocumentService(
82
+ resolvedUrl,
83
+ this.localDeltaConnectionServer,
84
+ tokenProvider,
85
+ tenantId,
86
+ documentId,
87
+ this.documentDeltaConnectionsMap,
88
+ this.policies,
89
+ this.innerDocumentService,
90
+ logger,
91
+ );
92
+ }
121
93
 
122
- /**
123
- * Gets the document delta connection for the clientId and asks it to disconnect the client.
124
- * @param clientId - The ID of the client to be disconnected.
125
- * @param disconnectReason - The reason of the disconnection.
126
- */
127
- public disconnectClient(clientId: string, disconnectReason: string) {
128
- if (!this.documentDeltaConnectionsMap.has(clientId)) {
129
- throw new Error(`No client with the id: ${clientId}`);
130
- }
131
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
132
- this.documentDeltaConnectionsMap.get(clientId)!.disconnectClient(disconnectReason);
133
- }
94
+ /**
95
+ * Gets the document delta connection for the clientId and asks it to disconnect the client.
96
+ * @param clientId - The ID of the client to be disconnected.
97
+ * @param disconnectReason - The reason of the disconnection.
98
+ */
99
+ public disconnectClient(clientId: string, disconnectReason: string) {
100
+ const documentDeltaConnection = this.documentDeltaConnectionsMap.get(clientId);
101
+ if (documentDeltaConnection === undefined) {
102
+ throw new Error(`No client with the id: ${clientId}`);
103
+ }
104
+ documentDeltaConnection.disconnectClient(disconnectReason);
105
+ }
134
106
 
135
- /**
136
- * Gets the document delta connection for the clientId and asks it to nack the client.
137
- * @param clientId - The ID of the client to be Nack'd.
138
- * @param code - An error code number that represents the error. It will be a valid HTTP error code.
139
- * @param type - Type of the Nack.
140
- * @param message - A message about the nack for debugging/logging/telemetry purposes.
141
- */
142
- public nackClient(clientId: string, code?: number, type?: NackErrorType, message?: any) {
143
- if (!this.documentDeltaConnectionsMap.has(clientId)) {
144
- throw new Error(`No client with the id: ${clientId}`);
145
- }
146
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
147
- this.documentDeltaConnectionsMap.get(clientId)!.nackClient(code, type, message);
148
- }
107
+ /**
108
+ * Gets the document delta connection for the clientId and asks it to nack the client.
109
+ * @param clientId - The ID of the client to be Nack'd.
110
+ * @param code - An error code number that represents the error. It will be a valid HTTP error code.
111
+ * @param type - Type of the Nack.
112
+ * @param message - A message about the nack for debugging/logging/telemetry purposes.
113
+ */
114
+ public nackClient(clientId: string, code?: number, type?: NackErrorType, message?: any) {
115
+ const documentDeltaConnection = this.documentDeltaConnectionsMap.get(clientId);
116
+ if (documentDeltaConnection === undefined) {
117
+ throw new Error(`No client with the id: ${clientId}`);
118
+ }
119
+ documentDeltaConnection.nackClient(code, type, message);
120
+ }
149
121
  }
@@ -0,0 +1,131 @@
1
+ /*!
2
+ * Copyright (c) Microsoft Corporation and contributors. All rights reserved.
3
+ * Licensed under the MIT License.
4
+ */
5
+
6
+ import { stringToBuffer, Uint8ArrayToString } from "@fluid-internal/client-utils";
7
+ import {
8
+ IDocumentStorageService,
9
+ IDocumentStorageServicePolicies,
10
+ IResolvedUrl,
11
+ ISummaryContext,
12
+ } from "@fluidframework/driver-definitions";
13
+ import {
14
+ ICreateBlobResponse,
15
+ ISnapshotTreeEx,
16
+ ISummaryHandle,
17
+ ISummaryTree,
18
+ IVersion,
19
+ } from "@fluidframework/protocol-definitions";
20
+ import { buildGitTreeHierarchy } from "@fluidframework/protocol-base";
21
+ import {
22
+ GitManager,
23
+ ISummaryUploadManager,
24
+ SummaryTreeUploadManager,
25
+ } from "@fluidframework/server-services-client";
26
+ import { ILocalDeltaConnectionServer } from "@fluidframework/server-local-server";
27
+ import { createDocument } from "./localCreateDocument";
28
+
29
+ const minTTLInSeconds = 24 * 60 * 60; // Same TTL as ODSP
30
+ /**
31
+ * @internal
32
+ */
33
+ export class LocalDocumentStorageService implements IDocumentStorageService {
34
+ // The values of this cache is useless. We only need the keys. So we are always putting
35
+ // empty strings as values.
36
+ protected readonly blobsShaCache = new Map<string, string>();
37
+ private readonly summaryTreeUploadManager: ISummaryUploadManager;
38
+
39
+ public readonly repositoryUrl: string = "";
40
+
41
+ constructor(
42
+ private readonly id: string,
43
+ private readonly manager: GitManager,
44
+ public readonly policies: IDocumentStorageServicePolicies,
45
+ private readonly localDeltaConnectionServer?: ILocalDeltaConnectionServer,
46
+ private readonly resolvedUrl?: IResolvedUrl,
47
+ ) {
48
+ this.summaryTreeUploadManager = new SummaryTreeUploadManager(
49
+ manager,
50
+ this.blobsShaCache,
51
+ this.getPreviousFullSnapshot.bind(this),
52
+ );
53
+ }
54
+
55
+ public async getVersions(versionId: string | null, count: number): Promise<IVersion[]> {
56
+ const id = versionId ? versionId : this.id;
57
+ const commits = await this.manager.getCommits(id, count);
58
+ return commits.map((commit) => ({
59
+ date: commit.commit.author.date,
60
+ id: commit.sha,
61
+ treeId: commit.commit.tree.sha,
62
+ }));
63
+ }
64
+
65
+ public async getSnapshotTree(version?: IVersion): Promise<ISnapshotTreeEx | null> {
66
+ let requestVersion = version;
67
+ if (!requestVersion) {
68
+ const versions = await this.getVersions(this.id, 1);
69
+ if (versions.length === 0) {
70
+ return null;
71
+ }
72
+
73
+ requestVersion = versions[0];
74
+ }
75
+
76
+ const rawTree = await this.manager.getTree(requestVersion.treeId);
77
+ const tree = buildGitTreeHierarchy(rawTree, this.blobsShaCache, true);
78
+ return tree;
79
+ }
80
+
81
+ public async readBlob(blobId: string): Promise<ArrayBufferLike> {
82
+ const blob = await this.manager.getBlob(blobId);
83
+ this.blobsShaCache.set(blob.sha, "");
84
+ const bufferContent = stringToBuffer(blob.content, blob.encoding);
85
+ return bufferContent;
86
+ }
87
+
88
+ public async uploadSummaryWithContext(
89
+ summary: ISummaryTree,
90
+ context: ISummaryContext,
91
+ ): Promise<string> {
92
+ if (context.referenceSequenceNumber === 0) {
93
+ if (this.localDeltaConnectionServer === undefined || this.resolvedUrl === undefined) {
94
+ throw new Error(
95
+ "Insufficient constructor parameters. An ILocalDeltaConnectionServer and IResolvedUrl required",
96
+ );
97
+ }
98
+ await createDocument(this.localDeltaConnectionServer, this.resolvedUrl, summary);
99
+ const version = await this.getVersions(this.id, 1);
100
+ return version[0].id;
101
+ }
102
+ return this.summaryTreeUploadManager.writeSummaryTree(
103
+ summary,
104
+ context.ackHandle ?? "",
105
+ "channel",
106
+ );
107
+ }
108
+
109
+ public async createBlob(file: ArrayBufferLike): Promise<ICreateBlobResponse> {
110
+ const uint8ArrayFile = new Uint8Array(file);
111
+ return this.manager
112
+ .createBlob(Uint8ArrayToString(uint8ArrayFile, "base64"), "base64")
113
+ .then((r) => ({ id: r.sha, url: r.url, minTTLInSeconds }));
114
+ }
115
+
116
+ public async downloadSummary(handle: ISummaryHandle): Promise<ISummaryTree> {
117
+ throw new Error("NOT IMPLEMENTED!");
118
+ }
119
+
120
+ private async getPreviousFullSnapshot(
121
+ parentHandle: string,
122
+ ): Promise<ISnapshotTreeEx | null | undefined> {
123
+ return parentHandle
124
+ ? this.getVersions(parentHandle, 1).then(async (versions) => {
125
+ // Clear the cache as the getSnapshotTree call will fill the cache.
126
+ this.blobsShaCache.clear();
127
+ return this.getSnapshotTree(versions[0]);
128
+ })
129
+ : undefined;
130
+ }
131
+ }