299 lines
7.9 KiB
TypeScript
299 lines
7.9 KiB
TypeScript
import { choose } from "../utils/choose";
|
|
import { v4 as uuidv4 } from "uuid";
|
|
import { assert } from "../utils/assert";
|
|
import type { RelativePath, SyncSettings } from "sync-client";
|
|
import { LogLevel } from "sync-client";
|
|
import { MockClient } from "./mock-client";
|
|
import chalk from "chalk";
|
|
|
|
export class MockAgent extends MockClient {
|
|
private readonly writtenContents: string[] = [];
|
|
private readonly pendingActions: Promise<unknown>[] = [];
|
|
private doNotTouch: string[] = [];
|
|
|
|
public constructor(
|
|
initialSettings: Partial<SyncSettings>,
|
|
public readonly name: string,
|
|
private readonly color: string,
|
|
private readonly doDeletes: boolean
|
|
) {
|
|
super(initialSettings);
|
|
}
|
|
|
|
public async init(): Promise<void> {
|
|
await super.init();
|
|
|
|
this.client.logger.addOnMessageListener((message) => {
|
|
const formatted = chalk.hex(this.color)(
|
|
`[${this.name}] ${message.timestamp.toISOString()} ${message.level} ${message.message}`
|
|
);
|
|
|
|
switch (message.level) {
|
|
case LogLevel.ERROR:
|
|
console.error(formatted);
|
|
// Let's not ignore errors
|
|
process.exit(1);
|
|
case LogLevel.WARNING:
|
|
console.warn(formatted);
|
|
break;
|
|
case LogLevel.INFO:
|
|
console.info(formatted);
|
|
break;
|
|
case LogLevel.DEBUG:
|
|
console.debug(formatted);
|
|
break;
|
|
}
|
|
});
|
|
|
|
this.client.logger.info("Agent initialized");
|
|
}
|
|
|
|
public async delete(path: RelativePath): Promise<void> {
|
|
assert(
|
|
this.doDeletes,
|
|
`Agent ${this.name} tried to delete file ${path} while doDeletes is false`
|
|
);
|
|
|
|
await super.delete(path);
|
|
}
|
|
|
|
public async act(): Promise<void> {
|
|
const options: (() => Promise<unknown>)[] = [
|
|
this.createFileAction.bind(this),
|
|
this.changeFetchChangesUpdateIntervalMsAction.bind(this),
|
|
this.disableSyncAction.bind(this),
|
|
this.enableSyncAction.bind(this)
|
|
];
|
|
|
|
const files = await this.listAllFiles();
|
|
|
|
if (files.length > 0) {
|
|
options.push(
|
|
this.renameFileAction.bind(this, files),
|
|
this.updateFileAction.bind(this, files)
|
|
);
|
|
|
|
if (this.doDeletes) {
|
|
options.push(this.deleteFileAction.bind(this, files));
|
|
}
|
|
}
|
|
|
|
this.pendingActions.push(
|
|
(async (): Promise<unknown> => {
|
|
try {
|
|
return await choose(options)();
|
|
} catch (error) {
|
|
this.client.logger.error(
|
|
`Failed to perform an action: ${error}`
|
|
);
|
|
this.client.logger.info(JSON.stringify(this.data, null, 2));
|
|
this.client.logger.info(
|
|
JSON.stringify(this.localFiles, null, 2)
|
|
);
|
|
throw error;
|
|
}
|
|
})()
|
|
);
|
|
}
|
|
|
|
public async finish(): Promise<void> {
|
|
await Promise.all(this.pendingActions);
|
|
await this.client.settings.setSetting("isSyncEnabled", true);
|
|
this.client.stop();
|
|
await this.client.syncer.waitForSyncQueue();
|
|
await this.client.syncer.applyRemoteChangesLocally();
|
|
}
|
|
|
|
public assertFileSystemsAreConsistent(otherAgent: MockAgent): void {
|
|
const globalFiles = Array.from(otherAgent.localFiles.keys());
|
|
const localFiles = Array.from(this.localFiles.keys());
|
|
|
|
const missingInOther = localFiles.filter(
|
|
(file) => !otherAgent.localFiles.has(file)
|
|
);
|
|
const missingInLocal = globalFiles.filter(
|
|
(file) => !this.localFiles.has(file)
|
|
);
|
|
|
|
try {
|
|
assert(
|
|
missingInOther.length === 0,
|
|
`Files from ${this.name} missing in ${otherAgent.name}: ${missingInOther.join(", ")}`
|
|
);
|
|
assert(
|
|
missingInLocal.length === 0,
|
|
`Files from ${otherAgent.name} missing in ${this.name}: ${missingInLocal.join(", ")}`
|
|
);
|
|
|
|
for (const file of globalFiles) {
|
|
const localContent = new TextDecoder().decode(
|
|
this.localFiles.get(file)
|
|
);
|
|
const otherContent = new TextDecoder().decode(
|
|
otherAgent.localFiles.get(file)
|
|
);
|
|
assert(
|
|
localContent === otherContent,
|
|
`Content mismatch for file ${file}:\n${localContent}\n${otherContent}`
|
|
);
|
|
}
|
|
} catch (e) {
|
|
this.client.logger.info(
|
|
"Local data: " + JSON.stringify(this.data, null, 2)
|
|
);
|
|
this.client.logger.info(
|
|
"Local files: " +
|
|
Array.from(otherAgent.localFiles.keys()).join(", ")
|
|
);
|
|
otherAgent.client.logger.info(
|
|
"Local data: " + JSON.stringify(otherAgent.data, null, 2)
|
|
);
|
|
otherAgent.client.logger.info(
|
|
"Local files: " +
|
|
Array.from(otherAgent.localFiles.keys()).join(", ")
|
|
);
|
|
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
public assertAllContentIsPresentOnce(): void {
|
|
for (const content of this.writtenContents) {
|
|
const found = Array.from(this.localFiles.keys()).filter((key) => {
|
|
return new TextDecoder()
|
|
.decode(this.localFiles.get(key))
|
|
.includes(content);
|
|
});
|
|
|
|
if (this.doDeletes) {
|
|
assert(
|
|
found.length <= 1,
|
|
`[${this.name}] Content ${content} found in ${found.join(", ")}`
|
|
);
|
|
} else {
|
|
assert(
|
|
found.length >= 1,
|
|
`[${this.name}] Content ${content} not found in any files`
|
|
);
|
|
|
|
assert(
|
|
found.length <= 1,
|
|
`[${this.name}] Content ${content} found in multiple files: ${found.join(", ")}`
|
|
);
|
|
|
|
const [file] = found;
|
|
const fileContent = new TextDecoder().decode(
|
|
this.localFiles.get(file)
|
|
);
|
|
assert(
|
|
fileContent.split(content).length == 2,
|
|
`Content ${content} (of ${this.name}) found more than once in file ${file}. File content:\n${fileContent}`
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
private async createFileAction(): Promise<void> {
|
|
const file = this.getFileName();
|
|
|
|
if (await this.exists(file)) {
|
|
return;
|
|
}
|
|
|
|
const content = this.getContent();
|
|
this.client.logger.info(
|
|
`Decided to create file ${file} with content ${content}`
|
|
);
|
|
|
|
return this.create(
|
|
file,
|
|
new TextEncoder().encode(` |${content}| `)
|
|
);
|
|
}
|
|
|
|
private async changeFetchChangesUpdateIntervalMsAction(): Promise<void> {
|
|
this.client.logger.info(
|
|
`Decided to change fetchChangesUpdateIntervalMs`
|
|
);
|
|
return this.client.settings.setSetting(
|
|
"fetchChangesUpdateIntervalMs",
|
|
Math.random() * 2000 + 100
|
|
);
|
|
}
|
|
|
|
private async disableSyncAction(): Promise<void> {
|
|
this.client.logger.info(`Decided to disable sync`);
|
|
await this.client.settings.setSetting("isSyncEnabled", false);
|
|
}
|
|
|
|
private async enableSyncAction(): Promise<void> {
|
|
this.client.logger.info(`Decided to enable sync`);
|
|
await this.client.settings.setSetting("isSyncEnabled", true);
|
|
this.doNotTouch = [];
|
|
}
|
|
|
|
private async renameFileAction(files: RelativePath[]): Promise<void> {
|
|
const file = choose(files);
|
|
|
|
// We can't edit files offline that have been renamed while offline.
|
|
// Otherwise, the resolution logic couldn't handle it.
|
|
if (this.doNotTouch.includes(file)) {
|
|
this.client.logger.info(
|
|
`Skipping file ${file} because it has been updated while offline`
|
|
);
|
|
return;
|
|
}
|
|
|
|
const newName = this.getFileName();
|
|
|
|
if (await this.exists(newName)) {
|
|
return;
|
|
}
|
|
|
|
this.client.logger.info(`Decided to rename file ${file} to ${newName}`);
|
|
if (!this.client.settings.getSettings().isSyncEnabled) {
|
|
this.doNotTouch.push(file, newName);
|
|
}
|
|
|
|
return this.rename(file, newName);
|
|
}
|
|
|
|
private async updateFileAction(files: RelativePath[]): Promise<void> {
|
|
const file = choose(files);
|
|
|
|
// We can't edit files offline that have been renamed while offline.
|
|
// Otherwise, the resolution logic couldn't handle it.
|
|
if (this.doNotTouch.includes(file)) {
|
|
this.client.logger.info(
|
|
`Skipping file ${file} because it has been renamed while offline`
|
|
);
|
|
return;
|
|
}
|
|
|
|
const content = this.getContent();
|
|
this.client.logger.info(
|
|
`Decided to update file ${file} with ${content}`
|
|
);
|
|
if (!this.client.settings.getSettings().isSyncEnabled) {
|
|
this.doNotTouch.push(file);
|
|
}
|
|
await this.atomicUpdateText(file, (old) => old + ` |${content}| `);
|
|
}
|
|
|
|
private async deleteFileAction(files: RelativePath[]): Promise<void> {
|
|
const file = choose(files);
|
|
this.client.logger.info(`Decided to delete file ${file}`);
|
|
return this.delete(file);
|
|
}
|
|
|
|
private getContent(): string {
|
|
const uuid = uuidv4();
|
|
this.writtenContents.push(uuid);
|
|
return uuid;
|
|
}
|
|
|
|
private getFileName(): string {
|
|
// Simulate name collisions between the clients
|
|
return `file-${Math.floor(Math.random() * 64)}.md`;
|
|
}
|
|
}
|