@fluidframework/replay-driver 1.4.0-115997 → 2.0.0-dev-rc.1.0.0.224419
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.
- package/.eslintrc.js +5 -7
- package/CHANGELOG.md +144 -0
- package/README.md +38 -0
- package/api-extractor-lint.json +4 -0
- package/api-extractor.json +2 -2
- package/api-report/replay-driver.api.md +150 -0
- package/dist/{emptyDeltaStorageService.js → emptyDeltaStorageService.cjs} +1 -1
- package/dist/emptyDeltaStorageService.cjs.map +1 -0
- package/dist/emptyDeltaStorageService.d.ts.map +1 -1
- package/dist/index.cjs +20 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.ts +4 -4
- package/dist/index.d.ts.map +1 -1
- package/dist/replay-driver-alpha.d.ts +34 -0
- package/dist/replay-driver-beta.d.ts +44 -0
- package/dist/replay-driver-public.d.ts +44 -0
- package/dist/replay-driver-untrimmed.d.ts +187 -0
- package/dist/{replayController.js → replayController.cjs} +6 -4
- package/dist/replayController.cjs.map +1 -0
- package/dist/replayController.d.ts +4 -2
- package/dist/replayController.d.ts.map +1 -1
- package/dist/{replayDocumentDeltaConnection.js → replayDocumentDeltaConnection.cjs} +25 -27
- package/dist/replayDocumentDeltaConnection.cjs.map +1 -0
- package/dist/replayDocumentDeltaConnection.d.ts +2 -2
- package/dist/replayDocumentDeltaConnection.d.ts.map +1 -1
- package/dist/{replayDocumentService.js → replayDocumentService.cjs} +9 -7
- package/dist/replayDocumentService.cjs.map +1 -0
- package/dist/replayDocumentService.d.ts +1 -0
- package/dist/replayDocumentService.d.ts.map +1 -1
- package/dist/{replayDocumentServiceFactory.js → replayDocumentServiceFactory.cjs} +13 -8
- package/dist/replayDocumentServiceFactory.cjs.map +1 -0
- package/dist/replayDocumentServiceFactory.d.ts +4 -2
- package/dist/replayDocumentServiceFactory.d.ts.map +1 -1
- package/dist/{storageImplementations.js → storageImplementations.cjs} +21 -12
- package/dist/storageImplementations.cjs.map +1 -0
- package/dist/storageImplementations.d.ts +18 -6
- package/dist/storageImplementations.d.ts.map +1 -1
- package/dist/tsdoc-metadata.json +11 -0
- package/lib/emptyDeltaStorageService.d.mts.map +1 -0
- package/lib/{emptyDeltaStorageService.js → emptyDeltaStorageService.mjs} +1 -1
- package/lib/emptyDeltaStorageService.mjs.map +1 -0
- package/lib/index.d.mts +9 -0
- package/lib/index.d.mts.map +1 -0
- package/lib/index.mjs +9 -0
- package/lib/index.mjs.map +1 -0
- package/lib/replay-driver-alpha.d.mts +34 -0
- package/lib/replay-driver-beta.d.mts +44 -0
- package/lib/replay-driver-public.d.mts +44 -0
- package/lib/replay-driver-untrimmed.d.mts +187 -0
- package/lib/{replayController.d.ts → replayController.d.mts} +4 -2
- package/lib/replayController.d.mts.map +1 -0
- package/lib/{replayController.js → replayController.mjs} +6 -4
- package/lib/replayController.mjs.map +1 -0
- package/lib/{replayDocumentDeltaConnection.d.ts → replayDocumentDeltaConnection.d.mts} +3 -3
- package/lib/replayDocumentDeltaConnection.d.mts.map +1 -0
- package/lib/{replayDocumentDeltaConnection.js → replayDocumentDeltaConnection.mjs} +23 -25
- package/lib/replayDocumentDeltaConnection.mjs.map +1 -0
- package/lib/{replayDocumentService.d.ts → replayDocumentService.d.mts} +2 -1
- package/lib/replayDocumentService.d.mts.map +1 -0
- package/lib/{replayDocumentService.js → replayDocumentService.mjs} +9 -7
- package/lib/replayDocumentService.mjs.map +1 -0
- package/lib/{replayDocumentServiceFactory.d.ts → replayDocumentServiceFactory.d.mts} +5 -3
- package/lib/replayDocumentServiceFactory.d.mts.map +1 -0
- package/lib/{replayDocumentServiceFactory.js → replayDocumentServiceFactory.mjs} +14 -9
- package/lib/replayDocumentServiceFactory.mjs.map +1 -0
- package/lib/{storageImplementations.d.ts → storageImplementations.d.mts} +19 -7
- package/lib/storageImplementations.d.mts.map +1 -0
- package/lib/{storageImplementations.js → storageImplementations.mjs} +20 -11
- package/lib/storageImplementations.mjs.map +1 -0
- package/lib/test/types/validateReplayDriverPrevious.generated.d.mts +2 -0
- package/lib/test/types/validateReplayDriverPrevious.generated.d.mts.map +1 -0
- package/lib/test/types/{validateReplayDriverPrevious.js → validateReplayDriverPrevious.generated.mjs} +1 -3
- package/lib/test/types/validateReplayDriverPrevious.generated.mjs.map +1 -0
- package/package.json +79 -47
- package/prettier.config.cjs +8 -0
- package/src/emptyDeltaStorageService.ts +14 -13
- package/src/index.ts +10 -4
- package/src/replayController.ts +61 -55
- package/src/replayDocumentDeltaConnection.ts +326 -318
- package/src/replayDocumentService.ts +48 -44
- package/src/replayDocumentServiceFactory.ts +59 -47
- package/src/storageImplementations.ts +168 -153
- package/tsconfig.json +11 -13
- package/dist/emptyDeltaStorageService.js.map +0 -1
- package/dist/index.js +0 -21
- package/dist/index.js.map +0 -1
- package/dist/packageVersion.d.ts +0 -9
- package/dist/packageVersion.d.ts.map +0 -1
- package/dist/packageVersion.js +0 -12
- package/dist/packageVersion.js.map +0 -1
- package/dist/replayController.js.map +0 -1
- package/dist/replayDocumentDeltaConnection.js.map +0 -1
- package/dist/replayDocumentService.js.map +0 -1
- package/dist/replayDocumentServiceFactory.js.map +0 -1
- package/dist/storageImplementations.js.map +0 -1
- package/lib/emptyDeltaStorageService.d.ts.map +0 -1
- package/lib/emptyDeltaStorageService.js.map +0 -1
- package/lib/index.d.ts +0 -9
- package/lib/index.d.ts.map +0 -1
- package/lib/index.js +0 -9
- package/lib/index.js.map +0 -1
- package/lib/packageVersion.d.ts +0 -9
- package/lib/packageVersion.d.ts.map +0 -1
- package/lib/packageVersion.js +0 -9
- package/lib/packageVersion.js.map +0 -1
- package/lib/replayController.d.ts.map +0 -1
- package/lib/replayController.js.map +0 -1
- package/lib/replayDocumentDeltaConnection.d.ts.map +0 -1
- package/lib/replayDocumentDeltaConnection.js.map +0 -1
- package/lib/replayDocumentService.d.ts.map +0 -1
- package/lib/replayDocumentService.js.map +0 -1
- package/lib/replayDocumentServiceFactory.d.ts.map +0 -1
- package/lib/replayDocumentServiceFactory.js.map +0 -1
- package/lib/storageImplementations.d.ts.map +0 -1
- package/lib/storageImplementations.js.map +0 -1
- package/lib/test/types/validateReplayDriverPrevious.d.ts +0 -2
- package/lib/test/types/validateReplayDriverPrevious.d.ts.map +0 -1
- package/lib/test/types/validateReplayDriverPrevious.js.map +0 -1
- package/src/packageVersion.ts +0 -9
- package/tsconfig.esnext.json +0 -7
- /package/lib/{emptyDeltaStorageService.d.ts → emptyDeltaStorageService.d.mts} +0 -0
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
import * as api from '@fluidframework/protocol-definitions';
|
|
2
|
+
import * as api_2 from '@fluidframework/driver-definitions';
|
|
3
|
+
import { IClient } from '@fluidframework/protocol-definitions';
|
|
4
|
+
import { IDocumentService } from '@fluidframework/driver-definitions';
|
|
5
|
+
import { IDocumentServiceFactory } from '@fluidframework/driver-definitions';
|
|
6
|
+
import { IDocumentStorageService } from '@fluidframework/driver-definitions';
|
|
7
|
+
import { IResolvedUrl } from '@fluidframework/driver-definitions';
|
|
8
|
+
import { ISnapshotTree } from '@fluidframework/protocol-definitions';
|
|
9
|
+
import { ISummaryContext } from '@fluidframework/driver-definitions';
|
|
10
|
+
import { ISummaryTree } from '@fluidframework/protocol-definitions';
|
|
11
|
+
import { ITelemetryBaseLogger } from '@fluidframework/core-interfaces';
|
|
12
|
+
import { ITelemetryLoggerExt } from '@fluidframework/telemetry-utils';
|
|
13
|
+
import { ITree } from '@fluidframework/protocol-definitions';
|
|
14
|
+
import { IVersion } from '@fluidframework/protocol-definitions';
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* @internal
|
|
18
|
+
*/
|
|
19
|
+
export declare class FileSnapshotReader extends ReadDocumentStorageServiceBase implements IDocumentStorageService {
|
|
20
|
+
protected static readonly FileStorageVersionTreeId = "FileStorageTreeId";
|
|
21
|
+
protected docId?: string;
|
|
22
|
+
protected docTree: ISnapshotTree;
|
|
23
|
+
protected blobs: Map<string, ArrayBufferLike>;
|
|
24
|
+
protected readonly commits: {
|
|
25
|
+
[key: string]: ITree;
|
|
26
|
+
};
|
|
27
|
+
protected readonly trees: {
|
|
28
|
+
[key: string]: ISnapshotTree;
|
|
29
|
+
};
|
|
30
|
+
constructor(json: IFileSnapshot);
|
|
31
|
+
getVersions(versionId: string | null, count: number): Promise<IVersion[]>;
|
|
32
|
+
getSnapshotTree(versionRequested?: IVersion): Promise<ISnapshotTree | null>;
|
|
33
|
+
readBlob(blobId: string): Promise<ArrayBufferLike>;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Structure of snapshot on disk, when we store snapshot as single file
|
|
38
|
+
* @internal
|
|
39
|
+
*/
|
|
40
|
+
export declare interface IFileSnapshot {
|
|
41
|
+
tree: ITree;
|
|
42
|
+
commits: {
|
|
43
|
+
[key: string]: ITree;
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* @internal
|
|
49
|
+
*/
|
|
50
|
+
export declare class OpStorage extends ReadDocumentStorageServiceBase {
|
|
51
|
+
getVersions(versionId: string | null, count: number): Promise<IVersion[]>;
|
|
52
|
+
getSnapshotTree(version?: IVersion): Promise<ISnapshotTree | null>;
|
|
53
|
+
readBlob(blobId: string): Promise<ArrayBufferLike>;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Partial implementation of IDocumentStorageService
|
|
58
|
+
* @internal
|
|
59
|
+
*/
|
|
60
|
+
export declare abstract class ReadDocumentStorageServiceBase implements IDocumentStorageService {
|
|
61
|
+
abstract getVersions(versionId: string | null, count: number): Promise<api.IVersion[]>;
|
|
62
|
+
abstract getSnapshotTree(version?: api.IVersion): Promise<api.ISnapshotTree | null>;
|
|
63
|
+
abstract readBlob(blobId: string): Promise<ArrayBufferLike>;
|
|
64
|
+
uploadSummaryWithContext(summary: api.ISummaryTree, context: ISummaryContext): Promise<string>;
|
|
65
|
+
createBlob(file: ArrayBufferLike): Promise<api.ICreateBlobResponse>;
|
|
66
|
+
downloadSummary(handle: api.ISummaryHandle): Promise<api.ISummaryTree>;
|
|
67
|
+
get repositoryUrl(): string;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Replay controller object
|
|
72
|
+
* It controls where we start (snapshot, local file, no snapshots)
|
|
73
|
+
* As well as dispatch of ops
|
|
74
|
+
* @internal
|
|
75
|
+
*/
|
|
76
|
+
export declare abstract class ReplayController extends ReadDocumentStorageServiceBase {
|
|
77
|
+
/**
|
|
78
|
+
* Initialize reply controller
|
|
79
|
+
* @param documentService - the real document service
|
|
80
|
+
* @returns Whether or not the controller should be used.
|
|
81
|
+
* If `false` is returned, caller should fallback to original storage.
|
|
82
|
+
*/
|
|
83
|
+
abstract initStorage(documentService: IDocumentService): Promise<boolean>;
|
|
84
|
+
/**
|
|
85
|
+
* Returns sequence number to start processing ops
|
|
86
|
+
* Should be zero if not using snapshot, and snapshot seq# otherwise
|
|
87
|
+
*/
|
|
88
|
+
abstract getStartingOpSequence(): Promise<number>;
|
|
89
|
+
/**
|
|
90
|
+
* Returns last op number to fetch from current op
|
|
91
|
+
* Note: this API is called while replay() is in progress - next batch of ops is downloaded in parallel
|
|
92
|
+
* @param currentOp - current op
|
|
93
|
+
*/
|
|
94
|
+
abstract fetchTo(currentOp: number): number | undefined;
|
|
95
|
+
/**
|
|
96
|
+
* Returns true if no more ops should be processed (or downloaded for future processing).
|
|
97
|
+
* It's called at end of each batch with latest op timestamp.
|
|
98
|
+
* Also it's called when there are no more ops available (lastTimeStamp === undefined).
|
|
99
|
+
* If false is returned and there are no more ops, request for more ops is made every 2 seconds.
|
|
100
|
+
* Note: this API is called while replay() is in progress - next batch of ops is downloaded in parallel
|
|
101
|
+
* @param currentOp - current op
|
|
102
|
+
* @param lastTimeStamp - timestamp of last op (if more ops are available). Undefined otherwise.
|
|
103
|
+
*/
|
|
104
|
+
abstract isDoneFetch(currentOp: number, lastTimeStamp?: number): boolean;
|
|
105
|
+
/**
|
|
106
|
+
* Replay batch of ops
|
|
107
|
+
* NOTE: new batch of ops is fetched (fetchTo() & isDoneFetch() APIs are called) while this call is in flights
|
|
108
|
+
* @param emitter - callback to emit ops
|
|
109
|
+
* @param fetchedOps - ops to process
|
|
110
|
+
*/
|
|
111
|
+
abstract replay(emitter: (op: api.ISequencedDocumentMessage[]) => void, fetchedOps: api.ISequencedDocumentMessage[]): Promise<void>;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* The Replay document service dummies out the snapshot and the delta storage.
|
|
116
|
+
* Delta connection simulates the socket by fetching the ops from delta storage
|
|
117
|
+
* and emitting them with a pre determined delay
|
|
118
|
+
* @internal
|
|
119
|
+
*/
|
|
120
|
+
export declare class ReplayDocumentService implements api_2.IDocumentService {
|
|
121
|
+
private readonly controller;
|
|
122
|
+
private readonly deltaStorage;
|
|
123
|
+
static create(documentService: api_2.IDocumentService, controller: ReplayController): Promise<api_2.IDocumentService>;
|
|
124
|
+
constructor(controller: api_2.IDocumentStorageService, deltaStorage: api_2.IDocumentDeltaConnection);
|
|
125
|
+
dispose(): void;
|
|
126
|
+
get resolvedUrl(): api_2.IResolvedUrl;
|
|
127
|
+
/**
|
|
128
|
+
* Connects to a storage endpoint for snapshot service and blobs.
|
|
129
|
+
* @returns returns the dummy document storage service for replay driver.
|
|
130
|
+
*/
|
|
131
|
+
connectToStorage(): Promise<api_2.IDocumentStorageService>;
|
|
132
|
+
/**
|
|
133
|
+
* Connects to a delta storage endpoint for getting ops between a range.
|
|
134
|
+
* @returns returns the dummy document delta storage service for replay driver.
|
|
135
|
+
*/
|
|
136
|
+
connectToDeltaStorage(): Promise<api_2.IDocumentDeltaStorageService>;
|
|
137
|
+
/**
|
|
138
|
+
* Connects to a delta storage endpoint of provided documentService to get ops and then replaying
|
|
139
|
+
* them so as to mimic a delta stream endpoint.
|
|
140
|
+
* @param client - Client that connects to socket.
|
|
141
|
+
* @returns returns the delta stream service which replay ops from --from to --to arguments.
|
|
142
|
+
*/
|
|
143
|
+
connectToDeltaStream(client: IClient): Promise<api_2.IDocumentDeltaConnection>;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* @internal
|
|
148
|
+
*/
|
|
149
|
+
export declare class ReplayDocumentServiceFactory implements IDocumentServiceFactory {
|
|
150
|
+
private readonly documentServiceFactory;
|
|
151
|
+
private readonly controller;
|
|
152
|
+
static create(from: number, to: number, documentServiceFactory: IDocumentServiceFactory): ReplayDocumentServiceFactory;
|
|
153
|
+
constructor(documentServiceFactory: IDocumentServiceFactory, controller: ReplayController);
|
|
154
|
+
/**
|
|
155
|
+
* Creates a replay document service which uses the document service of provided
|
|
156
|
+
* documentServiceFactory for connecting to delta stream endpoint.
|
|
157
|
+
* @param resolvedUrl - URL to be used for connecting to endpoints.
|
|
158
|
+
* @returns returns the requested document service
|
|
159
|
+
*/
|
|
160
|
+
createDocumentService(resolvedUrl: IResolvedUrl, logger?: ITelemetryBaseLogger, clientIsSummarizer?: boolean): Promise<IDocumentService>;
|
|
161
|
+
createContainer(createNewSummary: ISummaryTree, resolvedUrl: IResolvedUrl, logger?: ITelemetryBaseLogger, clientIsSummarizer?: boolean): Promise<IDocumentService>;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* @internal
|
|
166
|
+
*/
|
|
167
|
+
export declare class SnapshotStorage extends ReadDocumentStorageServiceBase {
|
|
168
|
+
protected readonly storage: IDocumentStorageService;
|
|
169
|
+
protected readonly docTree: ISnapshotTree | null;
|
|
170
|
+
protected docId?: string;
|
|
171
|
+
constructor(storage: IDocumentStorageService, docTree: ISnapshotTree | null);
|
|
172
|
+
getVersions(versionId: string | null, count: number): Promise<IVersion[]>;
|
|
173
|
+
getSnapshotTree(versionRequested?: IVersion): Promise<ISnapshotTree | null>;
|
|
174
|
+
readBlob(blobId: string): Promise<ArrayBufferLike>;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* @internal
|
|
179
|
+
*/
|
|
180
|
+
export declare class StaticStorageDocumentServiceFactory implements IDocumentServiceFactory {
|
|
181
|
+
protected readonly storage: IDocumentStorageService;
|
|
182
|
+
constructor(storage: IDocumentStorageService);
|
|
183
|
+
createDocumentService(fileURL: IResolvedUrl, logger?: ITelemetryLoggerExt, clientIsSummarizer?: boolean): Promise<IDocumentService>;
|
|
184
|
+
createContainer(createNewSummary: ISummaryTree, resolvedUrl: IResolvedUrl, logger: ITelemetryLoggerExt, clientIsSummarizer?: boolean): Promise<IDocumentService>;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
export { }
|
|
@@ -7,16 +7,17 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
7
7
|
exports.ReplayController = exports.ReadDocumentStorageServiceBase = void 0;
|
|
8
8
|
/**
|
|
9
9
|
* Partial implementation of IDocumentStorageService
|
|
10
|
+
* @internal
|
|
10
11
|
*/
|
|
11
12
|
class ReadDocumentStorageServiceBase {
|
|
12
13
|
async uploadSummaryWithContext(summary, context) {
|
|
13
|
-
|
|
14
|
+
throw new Error("Invalid operation");
|
|
14
15
|
}
|
|
15
16
|
async createBlob(file) {
|
|
16
|
-
|
|
17
|
+
throw new Error("Invalid operation");
|
|
17
18
|
}
|
|
18
19
|
async downloadSummary(handle) {
|
|
19
|
-
|
|
20
|
+
throw new Error("Invalid operation");
|
|
20
21
|
}
|
|
21
22
|
get repositoryUrl() {
|
|
22
23
|
throw new Error("Invalid operation");
|
|
@@ -27,8 +28,9 @@ exports.ReadDocumentStorageServiceBase = ReadDocumentStorageServiceBase;
|
|
|
27
28
|
* Replay controller object
|
|
28
29
|
* It controls where we start (snapshot, local file, no snapshots)
|
|
29
30
|
* As well as dispatch of ops
|
|
31
|
+
* @internal
|
|
30
32
|
*/
|
|
31
33
|
class ReplayController extends ReadDocumentStorageServiceBase {
|
|
32
34
|
}
|
|
33
35
|
exports.ReplayController = ReplayController;
|
|
34
|
-
//# sourceMappingURL=replayController.
|
|
36
|
+
//# sourceMappingURL=replayController.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"replayController.cjs","sourceRoot":"","sources":["../src/replayController.ts"],"names":[],"mappings":";AAAA;;;GAGG;;;AASH;;;GAGG;AACH,MAAsB,8BAA8B;IAK5C,KAAK,CAAC,wBAAwB,CACpC,OAAyB,EACzB,OAAwB;QAExB,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;IACtC,CAAC;IAEM,KAAK,CAAC,UAAU,CAAC,IAAqB;QAC5C,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;IACtC,CAAC;IAEM,KAAK,CAAC,eAAe,CAAC,MAA0B;QACtD,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;IACtC,CAAC;IAED,IAAW,aAAa;QACvB,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;IACtC,CAAC;CACD;AAvBD,wEAuBC;AAED;;;;;GAKG;AACH,MAAsB,gBAAiB,SAAQ,8BAA8B;CA2C5E;AA3CD,4CA2CC","sourcesContent":["/*!\n * Copyright (c) Microsoft Corporation and contributors. All rights reserved.\n * Licensed under the MIT License.\n */\n\nimport {\n\tIDocumentService,\n\tIDocumentStorageService,\n\tISummaryContext,\n} from \"@fluidframework/driver-definitions\";\nimport * as api from \"@fluidframework/protocol-definitions\";\n\n/**\n * Partial implementation of IDocumentStorageService\n * @internal\n */\nexport abstract class ReadDocumentStorageServiceBase implements IDocumentStorageService {\n\tpublic abstract getVersions(versionId: string | null, count: number): Promise<api.IVersion[]>;\n\tpublic abstract getSnapshotTree(version?: api.IVersion): Promise<api.ISnapshotTree | null>;\n\tpublic abstract readBlob(blobId: string): Promise<ArrayBufferLike>;\n\n\tpublic async uploadSummaryWithContext(\n\t\tsummary: api.ISummaryTree,\n\t\tcontext: ISummaryContext,\n\t): Promise<string> {\n\t\tthrow new Error(\"Invalid operation\");\n\t}\n\n\tpublic async createBlob(file: ArrayBufferLike): Promise<api.ICreateBlobResponse> {\n\t\tthrow new Error(\"Invalid operation\");\n\t}\n\n\tpublic async downloadSummary(handle: api.ISummaryHandle): Promise<api.ISummaryTree> {\n\t\tthrow new Error(\"Invalid operation\");\n\t}\n\n\tpublic get repositoryUrl(): string {\n\t\tthrow new Error(\"Invalid operation\");\n\t}\n}\n\n/**\n * Replay controller object\n * It controls where we start (snapshot, local file, no snapshots)\n * As well as dispatch of ops\n * @internal\n */\nexport abstract class ReplayController extends ReadDocumentStorageServiceBase {\n\t/**\n\t * Initialize reply controller\n\t * @param documentService - the real document service\n\t * @returns Whether or not the controller should be used.\n\t * If `false` is returned, caller should fallback to original storage.\n\t */\n\tpublic abstract initStorage(documentService: IDocumentService): Promise<boolean>;\n\n\t/**\n\t * Returns sequence number to start processing ops\n\t * Should be zero if not using snapshot, and snapshot seq# otherwise\n\t */\n\tpublic abstract getStartingOpSequence(): Promise<number>;\n\n\t/**\n\t * Returns last op number to fetch from current op\n\t * Note: this API is called while replay() is in progress - next batch of ops is downloaded in parallel\n\t * @param currentOp - current op\n\t */\n\tpublic abstract fetchTo(currentOp: number): number | undefined;\n\n\t/**\n\t * Returns true if no more ops should be processed (or downloaded for future processing).\n\t * It's called at end of each batch with latest op timestamp.\n\t * Also it's called when there are no more ops available (lastTimeStamp === undefined).\n\t * If false is returned and there are no more ops, request for more ops is made every 2 seconds.\n\t * Note: this API is called while replay() is in progress - next batch of ops is downloaded in parallel\n\t * @param currentOp - current op\n\t * @param lastTimeStamp - timestamp of last op (if more ops are available). Undefined otherwise.\n\t */\n\tpublic abstract isDoneFetch(currentOp: number, lastTimeStamp?: number): boolean;\n\n\t/**\n\t * Replay batch of ops\n\t * NOTE: new batch of ops is fetched (fetchTo() & isDoneFetch() APIs are called) while this call is in flights\n\t * @param emitter - callback to emit ops\n\t * @param fetchedOps - ops to process\n\t */\n\tpublic abstract replay(\n\t\temitter: (op: api.ISequencedDocumentMessage[]) => void,\n\t\tfetchedOps: api.ISequencedDocumentMessage[],\n\t): Promise<void>;\n}\n"]}
|
|
@@ -6,6 +6,7 @@ import { IDocumentService, IDocumentStorageService, ISummaryContext } from "@flu
|
|
|
6
6
|
import * as api from "@fluidframework/protocol-definitions";
|
|
7
7
|
/**
|
|
8
8
|
* Partial implementation of IDocumentStorageService
|
|
9
|
+
* @internal
|
|
9
10
|
*/
|
|
10
11
|
export declare abstract class ReadDocumentStorageServiceBase implements IDocumentStorageService {
|
|
11
12
|
abstract getVersions(versionId: string | null, count: number): Promise<api.IVersion[]>;
|
|
@@ -20,13 +21,14 @@ export declare abstract class ReadDocumentStorageServiceBase implements IDocumen
|
|
|
20
21
|
* Replay controller object
|
|
21
22
|
* It controls where we start (snapshot, local file, no snapshots)
|
|
22
23
|
* As well as dispatch of ops
|
|
24
|
+
* @internal
|
|
23
25
|
*/
|
|
24
26
|
export declare abstract class ReplayController extends ReadDocumentStorageServiceBase {
|
|
25
27
|
/**
|
|
26
28
|
* Initialize reply controller
|
|
27
29
|
* @param documentService - the real document service
|
|
28
|
-
* @returns
|
|
29
|
-
* If false is returned, caller should fallback to original storage.
|
|
30
|
+
* @returns Whether or not the controller should be used.
|
|
31
|
+
* If `false` is returned, caller should fallback to original storage.
|
|
30
32
|
*/
|
|
31
33
|
abstract initStorage(documentService: IDocumentService): Promise<boolean>;
|
|
32
34
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"replayController.d.ts","sourceRoot":"","sources":["../src/replayController.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,
|
|
1
|
+
{"version":3,"file":"replayController.d.ts","sourceRoot":"","sources":["../src/replayController.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EACN,gBAAgB,EAChB,uBAAuB,EACvB,eAAe,EACf,MAAM,oCAAoC,CAAC;AAC5C,OAAO,KAAK,GAAG,MAAM,sCAAsC,CAAC;AAE5D;;;GAGG;AACH,8BAAsB,8BAA+B,YAAW,uBAAuB;aACtE,WAAW,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC;aAC7E,eAAe,CAAC,OAAO,CAAC,EAAE,GAAG,CAAC,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,aAAa,GAAG,IAAI,CAAC;aAC1E,QAAQ,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,eAAe,CAAC;IAErD,wBAAwB,CACpC,OAAO,EAAE,GAAG,CAAC,YAAY,EACzB,OAAO,EAAE,eAAe,GACtB,OAAO,CAAC,MAAM,CAAC;IAIL,UAAU,CAAC,IAAI,EAAE,eAAe,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC;IAInE,eAAe,CAAC,MAAM,EAAE,GAAG,CAAC,cAAc,GAAG,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC;IAInF,IAAW,aAAa,IAAI,MAAM,CAEjC;CACD;AAED;;;;;GAKG;AACH,8BAAsB,gBAAiB,SAAQ,8BAA8B;IAC5E;;;;;OAKG;aACa,WAAW,CAAC,eAAe,EAAE,gBAAgB,GAAG,OAAO,CAAC,OAAO,CAAC;IAEhF;;;OAGG;aACa,qBAAqB,IAAI,OAAO,CAAC,MAAM,CAAC;IAExD;;;;OAIG;aACa,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAE9D;;;;;;;;OAQG;aACa,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,MAAM,GAAG,OAAO;IAE/E;;;;;OAKG;aACa,MAAM,CACrB,OAAO,EAAE,CAAC,EAAE,EAAE,GAAG,CAAC,yBAAyB,EAAE,KAAK,IAAI,EACtD,UAAU,EAAE,GAAG,CAAC,yBAAyB,EAAE,GACzC,OAAO,CAAC,IAAI,CAAC;CAChB"}
|
|
@@ -6,8 +6,9 @@
|
|
|
6
6
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
7
|
exports.ReplayDocumentDeltaConnection = exports.ReplayControllerStatic = void 0;
|
|
8
8
|
const protocol_definitions_1 = require("@fluidframework/protocol-definitions");
|
|
9
|
-
const
|
|
10
|
-
const
|
|
9
|
+
const client_utils_1 = require("@fluid-internal/client-utils");
|
|
10
|
+
const core_utils_1 = require("@fluidframework/core-utils");
|
|
11
|
+
const replayController_1 = require("./replayController.cjs");
|
|
11
12
|
const ReplayDocumentId = "documentId";
|
|
12
13
|
class ReplayControllerStatic extends replayController_1.ReplayController {
|
|
13
14
|
// Simulated delay interval for emitting the ops
|
|
@@ -39,7 +40,7 @@ class ReplayControllerStatic extends replayController_1.ReplayController {
|
|
|
39
40
|
return version ? Promise.reject(new Error("Invalid operation")) : null;
|
|
40
41
|
}
|
|
41
42
|
async readBlob(blobId) {
|
|
42
|
-
|
|
43
|
+
throw new Error("Invalid operation");
|
|
43
44
|
}
|
|
44
45
|
async getStartingOpSequence() {
|
|
45
46
|
return 0;
|
|
@@ -53,9 +54,9 @@ class ReplayControllerStatic extends replayController_1.ReplayController {
|
|
|
53
54
|
isDoneFetch(currentOp, lastTimeStamp) {
|
|
54
55
|
if (this.replayTo >= 0) {
|
|
55
56
|
if (this.unitIsTime === true) {
|
|
56
|
-
return (lastTimeStamp !== undefined
|
|
57
|
-
|
|
58
|
-
|
|
57
|
+
return (lastTimeStamp !== undefined &&
|
|
58
|
+
this.firstTimeStamp !== undefined &&
|
|
59
|
+
lastTimeStamp - this.firstTimeStamp >= this.replayTo);
|
|
59
60
|
}
|
|
60
61
|
return currentOp >= this.replayTo;
|
|
61
62
|
}
|
|
@@ -118,9 +119,9 @@ class ReplayControllerStatic extends replayController_1.ReplayController {
|
|
|
118
119
|
playbackOps.push(op);
|
|
119
120
|
current += 1;
|
|
120
121
|
}
|
|
121
|
-
if (this.firstTimeStamp !== undefined
|
|
122
|
-
|
|
123
|
-
|
|
122
|
+
if (this.firstTimeStamp !== undefined &&
|
|
123
|
+
this.replayTo >= 0 &&
|
|
124
|
+
currentTimeStamp + nextInterval - this.firstTimeStamp > this.replayTo) {
|
|
124
125
|
nextInterval = -1;
|
|
125
126
|
}
|
|
126
127
|
}
|
|
@@ -144,13 +145,7 @@ class ReplayControllerStatic extends replayController_1.ReplayController {
|
|
|
144
145
|
exports.ReplayControllerStatic = ReplayControllerStatic;
|
|
145
146
|
ReplayControllerStatic.DelayInterval = 50;
|
|
146
147
|
ReplayControllerStatic.ReplayResolution = 15;
|
|
147
|
-
class ReplayDocumentDeltaConnection extends
|
|
148
|
-
constructor(details) {
|
|
149
|
-
super();
|
|
150
|
-
this.details = details;
|
|
151
|
-
this.maxMessageSize = ReplayDocumentDeltaConnection.ReplayMaxMessageSize;
|
|
152
|
-
this._disposed = false;
|
|
153
|
-
}
|
|
148
|
+
class ReplayDocumentDeltaConnection extends client_utils_1.TypedEventEmitter {
|
|
154
149
|
/**
|
|
155
150
|
* Creates a new delta connection and mimics the delta connection to replay ops on it.
|
|
156
151
|
* @param documentService - The document service to be used to get underlying endpoints.
|
|
@@ -168,12 +163,6 @@ class ReplayDocumentDeltaConnection extends common_utils_1.TypedEventEmitter {
|
|
|
168
163
|
serviceConfiguration: {
|
|
169
164
|
blockSize: 64436,
|
|
170
165
|
maxMessageSize: ReplayDocumentDeltaConnection.ReplayMaxMessageSize,
|
|
171
|
-
summary: {
|
|
172
|
-
idleTime: 5000,
|
|
173
|
-
maxTime: 5000 * 12,
|
|
174
|
-
maxOps: 1000,
|
|
175
|
-
maxAckWaitTime: 600000,
|
|
176
|
-
},
|
|
177
166
|
},
|
|
178
167
|
supportedVersions: [ReplayDocumentDeltaConnection.replayProtocolVersion],
|
|
179
168
|
version: ReplayDocumentDeltaConnection.replayProtocolVersion,
|
|
@@ -210,15 +199,24 @@ class ReplayDocumentDeltaConnection extends common_utils_1.TypedEventEmitter {
|
|
|
210
199
|
get serviceConfiguration() {
|
|
211
200
|
return this.details.serviceConfiguration;
|
|
212
201
|
}
|
|
202
|
+
constructor(details) {
|
|
203
|
+
super();
|
|
204
|
+
this.details = details;
|
|
205
|
+
this.maxMessageSize = ReplayDocumentDeltaConnection.ReplayMaxMessageSize;
|
|
206
|
+
this._disposed = false;
|
|
207
|
+
}
|
|
213
208
|
submit(documentMessage) {
|
|
214
209
|
// ReplayDocumentDeltaConnection.submit() can't be called - client never sees its own join on,
|
|
215
210
|
// and thus can never move to sending ops.
|
|
216
211
|
throw new Error("ReplayDocumentDeltaConnection.submit() can't be called");
|
|
217
212
|
}
|
|
218
|
-
async submitSignal(message) {
|
|
213
|
+
async submitSignal(message) { }
|
|
214
|
+
get disposed() {
|
|
215
|
+
return this._disposed;
|
|
216
|
+
}
|
|
217
|
+
dispose() {
|
|
218
|
+
this._disposed = true;
|
|
219
219
|
}
|
|
220
|
-
get disposed() { return this._disposed; }
|
|
221
|
-
dispose() { this._disposed = true; }
|
|
222
220
|
/**
|
|
223
221
|
* This gets the specified ops from the delta storage endpoint and replays them in the replayer.
|
|
224
222
|
*/
|
|
@@ -237,7 +235,7 @@ class ReplayDocumentDeltaConnection extends common_utils_1.TypedEventEmitter {
|
|
|
237
235
|
// or because another client keeps submitting new ops.
|
|
238
236
|
done = controller.isDoneFetch(currentOp, undefined);
|
|
239
237
|
if (!done) {
|
|
240
|
-
await (0,
|
|
238
|
+
await (0, core_utils_1.delay)(2000);
|
|
241
239
|
}
|
|
242
240
|
break;
|
|
243
241
|
}
|
|
@@ -266,4 +264,4 @@ ReplayDocumentDeltaConnection.claims = {
|
|
|
266
264
|
exp: Math.round(new Date().getTime() / 1000) + 60 * 60,
|
|
267
265
|
ver: "1.0",
|
|
268
266
|
};
|
|
269
|
-
//# sourceMappingURL=replayDocumentDeltaConnection.
|
|
267
|
+
//# sourceMappingURL=replayDocumentDeltaConnection.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"replayDocumentDeltaConnection.cjs","sourceRoot":"","sources":["../src/replayDocumentDeltaConnection.ts"],"names":[],"mappings":"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","sourcesContent":["/*!\n * Copyright (c) Microsoft Corporation and contributors. All rights reserved.\n * Licensed under the MIT License.\n */\n\nimport { IDisposable } from \"@fluidframework/core-interfaces\";\nimport {\n\tIDocumentDeltaConnection,\n\tIDocumentDeltaStorageService,\n\tIDocumentDeltaConnectionEvents,\n\tIDocumentService,\n} from \"@fluidframework/driver-definitions\";\nimport {\n\tConnectionMode,\n\tIClientConfiguration,\n\tIConnected,\n\tIDocumentMessage,\n\tISequencedDocumentMessage,\n\tISignalClient,\n\tISignalMessage,\n\tITokenClaims,\n\tIVersion,\n\tScopeType,\n} from \"@fluidframework/protocol-definitions\";\nimport { TypedEventEmitter } from \"@fluid-internal/client-utils\";\nimport { delay } from \"@fluidframework/core-utils\";\nimport { ReplayController } from \"./replayController\";\n\nconst ReplayDocumentId = \"documentId\";\n\nexport class ReplayControllerStatic extends ReplayController {\n\tprivate static readonly DelayInterval = 50;\n\tprivate static readonly ReplayResolution = 15;\n\n\tprivate firstTimeStamp: number | undefined;\n\tprivate replayCurrent = 0;\n\t// Simulated delay interval for emitting the ops\n\n\t/**\n\t * Helper class\n\t *\n\t * @param replayFrom - First op to be played on socket.\n\t * @param replayTo - Last op number to be played on socket.\n\t * @param unitIsTime - True is user want to play ops that are within a replay resolution window.\n\t */\n\tpublic constructor(\n\t\tpublic readonly replayFrom: number,\n\t\tpublic readonly replayTo: number,\n\t\tpublic readonly unitIsTime?: boolean,\n\t) {\n\t\tsuper();\n\t\tif (unitIsTime !== true) {\n\t\t\t// There is no code in here to start with snapshot, thus we have to start with op #0.\n\t\t\tthis.replayTo = 0;\n\t\t}\n\t}\n\n\tpublic async initStorage(documentService: IDocumentService) {\n\t\treturn true;\n\t}\n\n\tpublic async getVersions(versionId: string | null, count: number): Promise<IVersion[]> {\n\t\treturn [];\n\t}\n\n\tpublic async getSnapshotTree(version?: IVersion) {\n\t\treturn version ? Promise.reject(new Error(\"Invalid operation\")) : null;\n\t}\n\n\tpublic async readBlob(blobId: string): Promise<ArrayBufferLike> {\n\t\tthrow new Error(\"Invalid operation\");\n\t}\n\n\tpublic async getStartingOpSequence(): Promise<number> {\n\t\treturn 0;\n\t}\n\n\tpublic fetchTo(currentOp: number) {\n\t\tif (!(this.unitIsTime !== true && this.replayTo >= 0)) {\n\t\t\treturn undefined;\n\t\t}\n\t\treturn this.replayTo;\n\t}\n\n\tpublic isDoneFetch(currentOp: number, lastTimeStamp?: number) {\n\t\tif (this.replayTo >= 0) {\n\t\t\tif (this.unitIsTime === true) {\n\t\t\t\treturn (\n\t\t\t\t\tlastTimeStamp !== undefined &&\n\t\t\t\t\tthis.firstTimeStamp !== undefined &&\n\t\t\t\t\tlastTimeStamp - this.firstTimeStamp >= this.replayTo\n\t\t\t\t);\n\t\t\t}\n\t\t\treturn currentOp >= this.replayTo;\n\t\t}\n\t\treturn lastTimeStamp === undefined; // No more ops\n\t}\n\n\tpublic skipToIndex(fetchedOps: ISequencedDocumentMessage[]) {\n\t\tif (this.replayFrom <= 0) {\n\t\t\treturn 0;\n\t\t}\n\t\tif (this.unitIsTime === true) {\n\t\t\tfor (let i = 0; i < fetchedOps.length; i += 1) {\n\t\t\t\tconst timeStamp = fetchedOps[i].timestamp;\n\t\t\t\tif (timeStamp !== undefined) {\n\t\t\t\t\tif (this.firstTimeStamp === undefined) {\n\t\t\t\t\t\tthis.firstTimeStamp = timeStamp;\n\t\t\t\t\t}\n\t\t\t\t\tif (timeStamp - this.firstTimeStamp >= this.replayFrom) {\n\t\t\t\t\t\treturn i;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} else if (this.replayFrom > this.replayCurrent) {\n\t\t\treturn this.replayFrom - this.replayCurrent;\n\t\t}\n\t\treturn 0;\n\t}\n\n\tpublic async replay(\n\t\temitter: (op: ISequencedDocumentMessage[]) => void,\n\t\tfetchedOps: ISequencedDocumentMessage[],\n\t): Promise<void> {\n\t\tlet current = this.skipToIndex(fetchedOps);\n\n\t\treturn new Promise((resolve) => {\n\t\t\tconst replayNextOps = () => {\n\t\t\t\t// Emit the ops from replay to the end every \"deltainterval\" milliseconds\n\t\t\t\t// to simulate the socket stream\n\t\t\t\tconst currentOp = fetchedOps[current];\n\t\t\t\tconst playbackOps = [currentOp];\n\t\t\t\tlet nextInterval = ReplayControllerStatic.DelayInterval;\n\t\t\t\tcurrent += 1;\n\n\t\t\t\tif (this.unitIsTime === true) {\n\t\t\t\t\tconst currentTimeStamp = currentOp.timestamp;\n\t\t\t\t\tif (currentTimeStamp !== undefined) {\n\t\t\t\t\t\t// Emit more ops that is in the ReplayResolution window\n\n\t\t\t\t\t\twhile (current < fetchedOps.length) {\n\t\t\t\t\t\t\tconst op = fetchedOps[current];\n\t\t\t\t\t\t\tif (op.timestamp === undefined) {\n\t\t\t\t\t\t\t\t// Missing timestamp, just delay the standard amount of time\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst timeDiff = op.timestamp - currentTimeStamp;\n\t\t\t\t\t\t\tif (timeDiff >= ReplayControllerStatic.ReplayResolution) {\n\t\t\t\t\t\t\t\t// Time exceeded the resolution window, break out the loop\n\t\t\t\t\t\t\t\t// and delay for the time difference.\n\t\t\t\t\t\t\t\tnextInterval = timeDiff;\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (timeDiff < 0) {\n\t\t\t\t\t\t\t\t// Time have regressed, just delay the standard amount of time\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t// The op is within the ReplayResolution emit it now\n\t\t\t\t\t\t\tplaybackOps.push(op);\n\t\t\t\t\t\t\tcurrent += 1;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (\n\t\t\t\t\t\t\tthis.firstTimeStamp !== undefined &&\n\t\t\t\t\t\t\tthis.replayTo >= 0 &&\n\t\t\t\t\t\t\tcurrentTimeStamp + nextInterval - this.firstTimeStamp > this.replayTo\n\t\t\t\t\t\t) {\n\t\t\t\t\t\t\tnextInterval = -1;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tscheduleNext(nextInterval);\n\t\t\t\temitter(playbackOps);\n\t\t\t};\n\t\t\tconst scheduleNext = (nextInterval: number) => {\n\t\t\t\tif (nextInterval >= 0 && current < fetchedOps.length) {\n\t\t\t\t\tsetTimeout(replayNextOps, nextInterval);\n\t\t\t\t} else {\n\t\t\t\t\tthis.replayCurrent += current;\n\t\t\t\t\tresolve();\n\t\t\t\t}\n\t\t\t};\n\t\t\tscheduleNext(ReplayControllerStatic.DelayInterval);\n\t\t});\n\t}\n}\n\nexport class ReplayDocumentDeltaConnection\n\textends TypedEventEmitter<IDocumentDeltaConnectionEvents>\n\timplements IDocumentDeltaConnection, IDisposable\n{\n\t/**\n\t * Creates a new delta connection and mimics the delta connection to replay ops on it.\n\t * @param documentService - The document service to be used to get underlying endpoints.\n\t */\n\tpublic static create(\n\t\tdocumentStorageService: IDocumentDeltaStorageService,\n\t\tcontroller: ReplayController,\n\t): IDocumentDeltaConnection {\n\t\tconst connection: IConnected = {\n\t\t\tclaims: ReplayDocumentDeltaConnection.claims,\n\t\t\tclientId: \"PseudoClientId\",\n\t\t\texisting: true,\n\t\t\tinitialMessages: [],\n\t\t\tinitialSignals: [],\n\t\t\tinitialClients: [],\n\t\t\tmaxMessageSize: ReplayDocumentDeltaConnection.ReplayMaxMessageSize,\n\t\t\tmode: \"read\",\n\t\t\tserviceConfiguration: {\n\t\t\t\tblockSize: 64436,\n\t\t\t\tmaxMessageSize: ReplayDocumentDeltaConnection.ReplayMaxMessageSize,\n\t\t\t},\n\t\t\tsupportedVersions: [ReplayDocumentDeltaConnection.replayProtocolVersion],\n\t\t\tversion: ReplayDocumentDeltaConnection.replayProtocolVersion,\n\t\t};\n\t\tconst deltaConnection = new ReplayDocumentDeltaConnection(connection);\n\t\t// eslint-disable-next-line @typescript-eslint/no-floating-promises\n\t\tdeltaConnection.fetchAndEmitOps(documentStorageService, controller);\n\n\t\treturn deltaConnection;\n\t}\n\n\tprivate static readonly replayProtocolVersion = \"^0.1.0\";\n\t// Since the replay service never actually sends messages the size below is arbitrary\n\tprivate static readonly ReplayMaxMessageSize = 16 * 1024;\n\n\tprivate static readonly claims: ITokenClaims = {\n\t\tdocumentId: ReplayDocumentId,\n\t\tscopes: [ScopeType.DocRead],\n\t\ttenantId: \"\",\n\t\tuser: {\n\t\t\tid: \"\",\n\t\t},\n\t\tiat: Math.round(new Date().getTime() / 1000),\n\t\texp: Math.round(new Date().getTime() / 1000) + 60 * 60, // 1 hour expiration\n\t\tver: \"1.0\",\n\t};\n\n\tpublic get clientId(): string {\n\t\treturn this.details.clientId;\n\t}\n\n\tpublic get mode(): ConnectionMode {\n\t\treturn this.details.mode;\n\t}\n\n\tpublic get claims(): ITokenClaims {\n\t\treturn this.details.claims;\n\t}\n\n\tpublic get existing(): boolean {\n\t\treturn this.details.existing;\n\t}\n\n\tpublic get version(): string {\n\t\treturn this.details.version;\n\t}\n\n\tpublic get initialMessages(): ISequencedDocumentMessage[] {\n\t\treturn this.details.initialMessages;\n\t}\n\n\tpublic get initialSignals(): ISignalMessage[] {\n\t\treturn this.details.initialSignals;\n\t}\n\n\tpublic get initialClients(): ISignalClient[] {\n\t\treturn this.details.initialClients;\n\t}\n\n\tpublic get serviceConfiguration(): IClientConfiguration {\n\t\treturn this.details.serviceConfiguration;\n\t}\n\n\tpublic readonly maxMessageSize = ReplayDocumentDeltaConnection.ReplayMaxMessageSize;\n\n\tconstructor(public details: IConnected) {\n\t\tsuper();\n\t}\n\n\tpublic submit(documentMessage: IDocumentMessage[]): void {\n\t\t// ReplayDocumentDeltaConnection.submit() can't be called - client never sees its own join on,\n\t\t// and thus can never move to sending ops.\n\t\tthrow new Error(\"ReplayDocumentDeltaConnection.submit() can't be called\");\n\t}\n\n\tpublic async submitSignal(message: any) {}\n\n\tprivate _disposed = false;\n\tpublic get disposed() {\n\t\treturn this._disposed;\n\t}\n\tpublic dispose() {\n\t\tthis._disposed = true;\n\t}\n\n\t/**\n\t * This gets the specified ops from the delta storage endpoint and replays them in the replayer.\n\t */\n\tprivate async fetchAndEmitOps(\n\t\tdocumentStorageService: IDocumentDeltaStorageService,\n\t\tcontroller: ReplayController,\n\t): Promise<void> {\n\t\tlet done;\n\t\tlet replayPromiseChain = Promise.resolve();\n\n\t\tlet currentOp = await controller.getStartingOpSequence();\n\n\t\tdo {\n\t\t\tconst fetchTo = controller.fetchTo(currentOp);\n\n\t\t\tconst abortController = new AbortController();\n\t\t\tconst stream = documentStorageService.fetchMessages(\n\t\t\t\tcurrentOp + 1,\n\t\t\t\tfetchTo,\n\t\t\t\tabortController.signal,\n\t\t\t);\n\t\t\tdo {\n\t\t\t\tconst result = await stream.read();\n\n\t\t\t\tif (result.done) {\n\t\t\t\t\t// No more ops. But, they can show up later, either because document was just created,\n\t\t\t\t\t// or because another client keeps submitting new ops.\n\t\t\t\t\tdone = controller.isDoneFetch(currentOp, undefined);\n\t\t\t\t\tif (!done) {\n\t\t\t\t\t\tawait delay(2000);\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t\treplayPromiseChain = replayPromiseChain.then(async () =>\n\t\t\t\t\tcontroller.replay((ops) => this.emit(\"op\", ReplayDocumentId, ops), messages),\n\t\t\t\t);\n\n\t\t\t\tconst messages = result.value;\n\t\t\t\tcurrentOp += messages.length;\n\t\t\t\tdone = controller.isDoneFetch(currentOp, messages[messages.length - 1].timestamp);\n\t\t\t} while (!done);\n\n\t\t\tabortController.abort();\n\t\t} while (!done);\n\t\treturn replayPromiseChain;\n\t}\n}\n"]}
|
|
@@ -2,10 +2,10 @@
|
|
|
2
2
|
* Copyright (c) Microsoft Corporation and contributors. All rights reserved.
|
|
3
3
|
* Licensed under the MIT License.
|
|
4
4
|
*/
|
|
5
|
-
import { IDisposable } from "@fluidframework/
|
|
5
|
+
import { IDisposable } from "@fluidframework/core-interfaces";
|
|
6
6
|
import { IDocumentDeltaConnection, IDocumentDeltaStorageService, IDocumentDeltaConnectionEvents, IDocumentService } from "@fluidframework/driver-definitions";
|
|
7
7
|
import { ConnectionMode, IClientConfiguration, IConnected, IDocumentMessage, ISequencedDocumentMessage, ISignalClient, ISignalMessage, ITokenClaims, IVersion } from "@fluidframework/protocol-definitions";
|
|
8
|
-
import { TypedEventEmitter } from "@
|
|
8
|
+
import { TypedEventEmitter } from "@fluid-internal/client-utils";
|
|
9
9
|
import { ReplayController } from "./replayController";
|
|
10
10
|
export declare class ReplayControllerStatic extends ReplayController {
|
|
11
11
|
readonly replayFrom: number;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"replayDocumentDeltaConnection.d.ts","sourceRoot":"","sources":["../src/replayDocumentDeltaConnection.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,WAAW,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"replayDocumentDeltaConnection.d.ts","sourceRoot":"","sources":["../src/replayDocumentDeltaConnection.ts"],"names":[],"mappings":"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"}
|
|
@@ -5,18 +5,16 @@
|
|
|
5
5
|
*/
|
|
6
6
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
7
|
exports.ReplayDocumentService = void 0;
|
|
8
|
-
const emptyDeltaStorageService_1 = require("./emptyDeltaStorageService");
|
|
9
|
-
const replayDocumentDeltaConnection_1 = require("./replayDocumentDeltaConnection");
|
|
8
|
+
const emptyDeltaStorageService_1 = require("./emptyDeltaStorageService.cjs");
|
|
9
|
+
const replayDocumentDeltaConnection_1 = require("./replayDocumentDeltaConnection.cjs");
|
|
10
10
|
/**
|
|
11
11
|
* The Replay document service dummies out the snapshot and the delta storage.
|
|
12
12
|
* Delta connection simulates the socket by fetching the ops from delta storage
|
|
13
13
|
* and emitting them with a pre determined delay
|
|
14
|
+
* @internal
|
|
14
15
|
*/
|
|
16
|
+
// eslint-disable-next-line import/namespace
|
|
15
17
|
class ReplayDocumentService {
|
|
16
|
-
constructor(controller, deltaStorage) {
|
|
17
|
-
this.controller = controller;
|
|
18
|
-
this.deltaStorage = deltaStorage;
|
|
19
|
-
}
|
|
20
18
|
static async create(documentService, controller) {
|
|
21
19
|
const useController = await controller.initStorage(documentService);
|
|
22
20
|
if (!useController) {
|
|
@@ -25,6 +23,10 @@ class ReplayDocumentService {
|
|
|
25
23
|
const deltaConnection = replayDocumentDeltaConnection_1.ReplayDocumentDeltaConnection.create(await documentService.connectToDeltaStorage(), controller);
|
|
26
24
|
return new ReplayDocumentService(controller, deltaConnection);
|
|
27
25
|
}
|
|
26
|
+
constructor(controller, deltaStorage) {
|
|
27
|
+
this.controller = controller;
|
|
28
|
+
this.deltaStorage = deltaStorage;
|
|
29
|
+
}
|
|
28
30
|
dispose() { }
|
|
29
31
|
// TODO: Issue-2109 Implement detach container api or put appropriate comment.
|
|
30
32
|
get resolvedUrl() {
|
|
@@ -55,4 +57,4 @@ class ReplayDocumentService {
|
|
|
55
57
|
}
|
|
56
58
|
}
|
|
57
59
|
exports.ReplayDocumentService = ReplayDocumentService;
|
|
58
|
-
//# sourceMappingURL=replayDocumentService.
|
|
60
|
+
//# sourceMappingURL=replayDocumentService.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"replayDocumentService.cjs","sourceRoot":"","sources":["../src/replayDocumentService.ts"],"names":[],"mappings":";AAAA;;;GAGG;;;AAIH,6EAAsE;AAEtE,uFAAgF;AAEhF;;;;;GAKG;AACH,4CAA4C;AAC5C,MAAa,qBAAqB;IAC1B,MAAM,CAAC,KAAK,CAAC,MAAM,CACzB,eAAqC,EACrC,UAA4B;QAE5B,MAAM,aAAa,GAAG,MAAM,UAAU,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC;QACpE,IAAI,CAAC,aAAa,EAAE;YACnB,OAAO,eAAe,CAAC;SACvB;QAED,MAAM,eAAe,GAAG,6DAA6B,CAAC,MAAM,CAC3D,MAAM,eAAe,CAAC,qBAAqB,EAAE,EAC7C,UAAU,CACV,CAAC;QACF,OAAO,IAAI,qBAAqB,CAAC,UAAU,EAAE,eAAe,CAAC,CAAC;IAC/D,CAAC;IAED,YACkB,UAAuC,EACvC,YAA0C;QAD1C,eAAU,GAAV,UAAU,CAA6B;QACvC,iBAAY,GAAZ,YAAY,CAA8B;IACzD,CAAC;IAEG,OAAO,KAAI,CAAC;IAEnB,8EAA8E;IAC9E,IAAW,WAAW;QACrB,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;IACpC,CAAC;IAED;;;OAGG;IACI,KAAK,CAAC,gBAAgB;QAC5B,OAAO,IAAI,CAAC,UAAU,CAAC;IACxB,CAAC;IAED;;;OAGG;IACI,KAAK,CAAC,qBAAqB;QACjC,OAAO,IAAI,mDAAwB,EAAE,CAAC;IACvC,CAAC;IAED;;;;;OAKG;IACI,KAAK,CAAC,oBAAoB,CAAC,MAAe;QAChD,OAAO,IAAI,CAAC,YAAY,CAAC;IAC1B,CAAC;CACD;AAtDD,sDAsDC","sourcesContent":["/*!\n * Copyright (c) Microsoft Corporation and contributors. All rights reserved.\n * Licensed under the MIT License.\n */\n\nimport * as api from \"@fluidframework/driver-definitions\";\nimport { IClient } from \"@fluidframework/protocol-definitions\";\nimport { EmptyDeltaStorageService } from \"./emptyDeltaStorageService\";\nimport { ReplayController } from \"./replayController\";\nimport { ReplayDocumentDeltaConnection } from \"./replayDocumentDeltaConnection\";\n\n/**\n * The Replay document service dummies out the snapshot and the delta storage.\n * Delta connection simulates the socket by fetching the ops from delta storage\n * and emitting them with a pre determined delay\n * @internal\n */\n// eslint-disable-next-line import/namespace\nexport class ReplayDocumentService implements api.IDocumentService {\n\tpublic static async create(\n\t\tdocumentService: api.IDocumentService,\n\t\tcontroller: ReplayController,\n\t): Promise<api.IDocumentService> {\n\t\tconst useController = await controller.initStorage(documentService);\n\t\tif (!useController) {\n\t\t\treturn documentService;\n\t\t}\n\n\t\tconst deltaConnection = ReplayDocumentDeltaConnection.create(\n\t\t\tawait documentService.connectToDeltaStorage(),\n\t\t\tcontroller,\n\t\t);\n\t\treturn new ReplayDocumentService(controller, deltaConnection);\n\t}\n\n\tconstructor(\n\t\tprivate readonly controller: api.IDocumentStorageService,\n\t\tprivate readonly deltaStorage: api.IDocumentDeltaConnection,\n\t) {}\n\n\tpublic dispose() {}\n\n\t// TODO: Issue-2109 Implement detach container api or put appropriate comment.\n\tpublic get resolvedUrl(): api.IResolvedUrl {\n\t\tthrow new Error(\"Not implemented\");\n\t}\n\n\t/**\n\t * Connects to a storage endpoint for snapshot service and blobs.\n\t * @returns returns the dummy document storage service for replay driver.\n\t */\n\tpublic async connectToStorage(): Promise<api.IDocumentStorageService> {\n\t\treturn this.controller;\n\t}\n\n\t/**\n\t * Connects to a delta storage endpoint for getting ops between a range.\n\t * @returns returns the dummy document delta storage service for replay driver.\n\t */\n\tpublic async connectToDeltaStorage(): Promise<api.IDocumentDeltaStorageService> {\n\t\treturn new EmptyDeltaStorageService();\n\t}\n\n\t/**\n\t * Connects to a delta storage endpoint of provided documentService to get ops and then replaying\n\t * them so as to mimic a delta stream endpoint.\n\t * @param client - Client that connects to socket.\n\t * @returns returns the delta stream service which replay ops from --from to --to arguments.\n\t */\n\tpublic async connectToDeltaStream(client: IClient): Promise<api.IDocumentDeltaConnection> {\n\t\treturn this.deltaStorage;\n\t}\n}\n"]}
|
|
@@ -9,6 +9,7 @@ import { ReplayController } from "./replayController";
|
|
|
9
9
|
* The Replay document service dummies out the snapshot and the delta storage.
|
|
10
10
|
* Delta connection simulates the socket by fetching the ops from delta storage
|
|
11
11
|
* and emitting them with a pre determined delay
|
|
12
|
+
* @internal
|
|
12
13
|
*/
|
|
13
14
|
export declare class ReplayDocumentService implements api.IDocumentService {
|
|
14
15
|
private readonly controller;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"replayDocumentService.d.ts","sourceRoot":"","sources":["../src/replayDocumentService.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,GAAG,MAAM,oCAAoC,CAAC;AAC1D,OAAO,EAAE,OAAO,EAAE,MAAM,sCAAsC,CAAC;AAE/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAGtD
|
|
1
|
+
{"version":3,"file":"replayDocumentService.d.ts","sourceRoot":"","sources":["../src/replayDocumentService.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,GAAG,MAAM,oCAAoC,CAAC;AAC1D,OAAO,EAAE,OAAO,EAAE,MAAM,sCAAsC,CAAC;AAE/D,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAGtD;;;;;GAKG;AAEH,qBAAa,qBAAsB,YAAW,GAAG,CAAC,gBAAgB;IAkBhE,OAAO,CAAC,QAAQ,CAAC,UAAU;IAC3B,OAAO,CAAC,QAAQ,CAAC,YAAY;WAlBV,MAAM,CACzB,eAAe,EAAE,GAAG,CAAC,gBAAgB,EACrC,UAAU,EAAE,gBAAgB,GAC1B,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC;gBAcd,UAAU,EAAE,GAAG,CAAC,uBAAuB,EACvC,YAAY,EAAE,GAAG,CAAC,wBAAwB;IAGrD,OAAO;IAGd,IAAW,WAAW,IAAI,GAAG,CAAC,YAAY,CAEzC;IAED;;;OAGG;IACU,gBAAgB,IAAI,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC;IAIrE;;;OAGG;IACU,qBAAqB,IAAI,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC;IAI/E;;;;;OAKG;IACU,oBAAoB,CAAC,MAAM,EAAE,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC;CAGzF"}
|
|
@@ -6,16 +6,18 @@
|
|
|
6
6
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
7
|
exports.ReplayDocumentServiceFactory = void 0;
|
|
8
8
|
const telemetry_utils_1 = require("@fluidframework/telemetry-utils");
|
|
9
|
-
const replayDocumentDeltaConnection_1 = require("./replayDocumentDeltaConnection");
|
|
10
|
-
const replayDocumentService_1 = require("./replayDocumentService");
|
|
9
|
+
const replayDocumentDeltaConnection_1 = require("./replayDocumentDeltaConnection.cjs");
|
|
10
|
+
const replayDocumentService_1 = require("./replayDocumentService.cjs");
|
|
11
|
+
/**
|
|
12
|
+
* @internal
|
|
13
|
+
*/
|
|
11
14
|
class ReplayDocumentServiceFactory {
|
|
15
|
+
static create(from, to, documentServiceFactory) {
|
|
16
|
+
return new ReplayDocumentServiceFactory(documentServiceFactory, new replayDocumentDeltaConnection_1.ReplayControllerStatic(from, to));
|
|
17
|
+
}
|
|
12
18
|
constructor(documentServiceFactory, controller) {
|
|
13
19
|
this.documentServiceFactory = documentServiceFactory;
|
|
14
20
|
this.controller = controller;
|
|
15
|
-
this.protocolName = documentServiceFactory.protocolName;
|
|
16
|
-
}
|
|
17
|
-
static create(from, to, documentServiceFactory) {
|
|
18
|
-
return new ReplayDocumentServiceFactory(documentServiceFactory, new replayDocumentDeltaConnection_1.ReplayControllerStatic(from, to));
|
|
19
21
|
}
|
|
20
22
|
/**
|
|
21
23
|
* Creates a replay document service which uses the document service of provided
|
|
@@ -26,7 +28,10 @@ class ReplayDocumentServiceFactory {
|
|
|
26
28
|
async createDocumentService(resolvedUrl, logger, clientIsSummarizer) {
|
|
27
29
|
// Always include isReplay: true on events for the Replay Driver.
|
|
28
30
|
// It's used in testing/debugging scenarios, so we want to be able to filter these events out sometimes.
|
|
29
|
-
const replayLogger = telemetry_utils_1.
|
|
31
|
+
const replayLogger = (0, telemetry_utils_1.createChildLogger)({
|
|
32
|
+
logger,
|
|
33
|
+
properties: { all: { isReplay: true } },
|
|
34
|
+
});
|
|
30
35
|
return replayDocumentService_1.ReplayDocumentService.create(await this.documentServiceFactory.createDocumentService(resolvedUrl, replayLogger, clientIsSummarizer), this.controller);
|
|
31
36
|
}
|
|
32
37
|
// TODO: Issue-2109 Implement detach container api or put appropriate comment.
|
|
@@ -35,4 +40,4 @@ class ReplayDocumentServiceFactory {
|
|
|
35
40
|
}
|
|
36
41
|
}
|
|
37
42
|
exports.ReplayDocumentServiceFactory = ReplayDocumentServiceFactory;
|
|
38
|
-
//# sourceMappingURL=replayDocumentServiceFactory.
|
|
43
|
+
//# sourceMappingURL=replayDocumentServiceFactory.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"replayDocumentServiceFactory.cjs","sourceRoot":"","sources":["../src/replayDocumentServiceFactory.ts"],"names":[],"mappings":";AAAA;;;GAGG;;;AASH,qEAAoE;AAEpE,uFAAyE;AACzE,uEAAgE;AAEhE;;GAEG;AACH,MAAa,4BAA4B;IACjC,MAAM,CAAC,MAAM,CACnB,IAAY,EACZ,EAAU,EACV,sBAA+C;QAE/C,OAAO,IAAI,4BAA4B,CACtC,sBAAsB,EACtB,IAAI,sDAAsB,CAAC,IAAI,EAAE,EAAE,CAAC,CACpC,CAAC;IACH,CAAC;IAED,YACkB,sBAA+C,EAC/C,UAA4B;QAD5B,2BAAsB,GAAtB,sBAAsB,CAAyB;QAC/C,eAAU,GAAV,UAAU,CAAkB;IAC3C,CAAC;IAEJ;;;;;OAKG;IACI,KAAK,CAAC,qBAAqB,CACjC,WAAyB,EACzB,MAA6B,EAC7B,kBAA4B;QAE5B,iEAAiE;QACjE,wGAAwG;QACxG,MAAM,YAAY,GAAG,IAAA,mCAAiB,EAAC;YACtC,MAAM;YACN,UAAU,EAAE,EAAE,GAAG,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,EAAE;SACvC,CAAC,CAAC;QAEH,OAAO,6CAAqB,CAAC,MAAM,CAClC,MAAM,IAAI,CAAC,sBAAsB,CAAC,qBAAqB,CACtD,WAAW,EACX,YAAY,EACZ,kBAAkB,CAClB,EACD,IAAI,CAAC,UAAU,CACf,CAAC;IACH,CAAC;IAED,8EAA8E;IACvE,KAAK,CAAC,eAAe,CAC3B,gBAA8B,EAC9B,WAAyB,EACzB,MAA6B,EAC7B,kBAA4B;QAE5B,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;IACpC,CAAC;CACD;AAtDD,oEAsDC","sourcesContent":["/*!\n * Copyright (c) Microsoft Corporation and contributors. All rights reserved.\n * Licensed under the MIT License.\n */\n\nimport {\n\tIDocumentService,\n\tIDocumentServiceFactory,\n\tIResolvedUrl,\n} from \"@fluidframework/driver-definitions\";\nimport { ISummaryTree } from \"@fluidframework/protocol-definitions\";\nimport { ITelemetryBaseLogger } from \"@fluidframework/core-interfaces\";\nimport { createChildLogger } from \"@fluidframework/telemetry-utils\";\nimport { ReplayController } from \"./replayController\";\nimport { ReplayControllerStatic } from \"./replayDocumentDeltaConnection\";\nimport { ReplayDocumentService } from \"./replayDocumentService\";\n\n/**\n * @internal\n */\nexport class ReplayDocumentServiceFactory implements IDocumentServiceFactory {\n\tpublic static create(\n\t\tfrom: number,\n\t\tto: number,\n\t\tdocumentServiceFactory: IDocumentServiceFactory,\n\t) {\n\t\treturn new ReplayDocumentServiceFactory(\n\t\t\tdocumentServiceFactory,\n\t\t\tnew ReplayControllerStatic(from, to),\n\t\t);\n\t}\n\n\tpublic constructor(\n\t\tprivate readonly documentServiceFactory: IDocumentServiceFactory,\n\t\tprivate readonly controller: ReplayController,\n\t) {}\n\n\t/**\n\t * Creates a replay document service which uses the document service of provided\n\t * documentServiceFactory for connecting to delta stream endpoint.\n\t * @param resolvedUrl - URL to be used for connecting to endpoints.\n\t * @returns returns the requested document service\n\t */\n\tpublic async createDocumentService(\n\t\tresolvedUrl: IResolvedUrl,\n\t\tlogger?: ITelemetryBaseLogger,\n\t\tclientIsSummarizer?: boolean,\n\t): Promise<IDocumentService> {\n\t\t// Always include isReplay: true on events for the Replay Driver.\n\t\t// It's used in testing/debugging scenarios, so we want to be able to filter these events out sometimes.\n\t\tconst replayLogger = createChildLogger({\n\t\t\tlogger,\n\t\t\tproperties: { all: { isReplay: true } },\n\t\t});\n\n\t\treturn ReplayDocumentService.create(\n\t\t\tawait this.documentServiceFactory.createDocumentService(\n\t\t\t\tresolvedUrl,\n\t\t\t\treplayLogger,\n\t\t\t\tclientIsSummarizer,\n\t\t\t),\n\t\t\tthis.controller,\n\t\t);\n\t}\n\n\t// TODO: Issue-2109 Implement detach container api or put appropriate comment.\n\tpublic async createContainer(\n\t\tcreateNewSummary: ISummaryTree,\n\t\tresolvedUrl: IResolvedUrl,\n\t\tlogger?: ITelemetryBaseLogger,\n\t\tclientIsSummarizer?: boolean,\n\t): Promise<IDocumentService> {\n\t\tthrow new Error(\"Not implemented\");\n\t}\n}\n"]}
|
|
@@ -4,13 +4,15 @@
|
|
|
4
4
|
*/
|
|
5
5
|
import { IDocumentService, IDocumentServiceFactory, IResolvedUrl } from "@fluidframework/driver-definitions";
|
|
6
6
|
import { ISummaryTree } from "@fluidframework/protocol-definitions";
|
|
7
|
-
import { ITelemetryBaseLogger } from "@fluidframework/
|
|
7
|
+
import { ITelemetryBaseLogger } from "@fluidframework/core-interfaces";
|
|
8
8
|
import { ReplayController } from "./replayController";
|
|
9
|
+
/**
|
|
10
|
+
* @internal
|
|
11
|
+
*/
|
|
9
12
|
export declare class ReplayDocumentServiceFactory implements IDocumentServiceFactory {
|
|
10
13
|
private readonly documentServiceFactory;
|
|
11
14
|
private readonly controller;
|
|
12
15
|
static create(from: number, to: number, documentServiceFactory: IDocumentServiceFactory): ReplayDocumentServiceFactory;
|
|
13
|
-
readonly protocolName: any;
|
|
14
16
|
constructor(documentServiceFactory: IDocumentServiceFactory, controller: ReplayController);
|
|
15
17
|
/**
|
|
16
18
|
* Creates a replay document service which uses the document service of provided
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"replayDocumentServiceFactory.d.ts","sourceRoot":"","sources":["../src/replayDocumentServiceFactory.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,
|
|
1
|
+
{"version":3,"file":"replayDocumentServiceFactory.d.ts","sourceRoot":"","sources":["../src/replayDocumentServiceFactory.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EACN,gBAAgB,EAChB,uBAAuB,EACvB,YAAY,EACZ,MAAM,oCAAoC,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,MAAM,sCAAsC,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,iCAAiC,CAAC;AAEvE,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAItD;;GAEG;AACH,qBAAa,4BAA6B,YAAW,uBAAuB;IAa1E,OAAO,CAAC,QAAQ,CAAC,sBAAsB;IACvC,OAAO,CAAC,QAAQ,CAAC,UAAU;WAbd,MAAM,CACnB,IAAI,EAAE,MAAM,EACZ,EAAE,EAAE,MAAM,EACV,sBAAsB,EAAE,uBAAuB;gBAS9B,sBAAsB,EAAE,uBAAuB,EAC/C,UAAU,EAAE,gBAAgB;IAG9C;;;;;OAKG;IACU,qBAAqB,CACjC,WAAW,EAAE,YAAY,EACzB,MAAM,CAAC,EAAE,oBAAoB,EAC7B,kBAAkB,CAAC,EAAE,OAAO,GAC1B,OAAO,CAAC,gBAAgB,CAAC;IAmBf,eAAe,CAC3B,gBAAgB,EAAE,YAAY,EAC9B,WAAW,EAAE,YAAY,EACzB,MAAM,CAAC,EAAE,oBAAoB,EAC7B,kBAAkB,CAAC,EAAE,OAAO,GAC1B,OAAO,CAAC,gBAAgB,CAAC;CAG5B"}
|