Fix server synchronization
This commit is contained in:
parent
20d88852b1
commit
acde2a7db1
|
@ -1,25 +1,28 @@
|
||||||
import { Project } from "../models/project";
|
import { Project } from "../models/project";
|
||||||
import { usePrevious } from "./use-previous";
|
import { usePrevious } from "./use-previous";
|
||||||
import { useEffect, useState } from "preact/hooks";
|
import { useEffect, useState, useRef } from "preact/hooks";
|
||||||
import { generate as diff } from "json-merge-patch";
|
import { generate as diff } from "json-merge-patch";
|
||||||
import useDebounce from "./use-debounce";
|
import useDebounce from "./use-debounce";
|
||||||
|
|
||||||
export interface ServerSyncOptions {
|
export interface ServerSyncOptions {
|
||||||
projectURL: string
|
projectURL: string
|
||||||
refreshInterval: number
|
refreshInterval: number
|
||||||
syncDelay: number
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const defaultOptions = {
|
export const defaultOptions = {
|
||||||
refreshInterval: 10000,
|
refreshInterval: 5000,
|
||||||
projectURL: `//${window.location.host}/api/v1/projects`,
|
projectURL: `//${window.location.host}/api/v1/projects`,
|
||||||
syncDelay: 5000,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useServerSync(project: Project, applyServerUpdate: (project: Project) => void, options: ServerSyncOptions = defaultOptions) {
|
export function useServerSync(project: Project, applyServerUpdate: (project: Project) => void, options: ServerSyncOptions = defaultOptions) {
|
||||||
options = Object.assign({}, defaultOptions, options);
|
options = Object.assign({}, defaultOptions, options);
|
||||||
|
|
||||||
const [ version, setVersion ] = useState(0);
|
const [ serverState, setServerState ] = useState({
|
||||||
|
version: 0,
|
||||||
|
project: project,
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log('server state', serverState);
|
||||||
|
|
||||||
const handleAPIResponse = (res: Response) => {
|
const handleAPIResponse = (res: Response) => {
|
||||||
// If the project does not yet exist, create it
|
// If the project does not yet exist, create it
|
||||||
|
@ -27,58 +30,56 @@ export function useServerSync(project: Project, applyServerUpdate: (project: Pro
|
||||||
return createProject(project, options)
|
return createProject(project, options)
|
||||||
.then(res => res.json())
|
.then(res => res.json())
|
||||||
.then(result => {
|
.then(result => {
|
||||||
setVersion(result.version);
|
setServerState({ version: result.version, project });
|
||||||
})
|
})
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
// In case of conflict, notify of new server version
|
// In case of conflict, notify of new server version
|
||||||
if (res.status === 409) {
|
if (res.status === 409) {
|
||||||
|
console.log("server conflict detected");
|
||||||
return res.json().then((result: any) => {
|
return res.json().then((result: any) => {
|
||||||
|
console.log('received response', result);
|
||||||
applyServerUpdate(result.project);
|
applyServerUpdate(result.project);
|
||||||
setVersion(result.version);
|
setServerState({ version: result.version, project: result.project });
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the server version is not modified, do nothing
|
// If the server version is not modified, do nothing
|
||||||
if (res.status === 304) {
|
if (res.status === 304) {
|
||||||
|
console.log("no modification on server");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
return res.json().then((result: any) => {
|
return res.json().then((result: any) => {
|
||||||
|
console.log('received response', result);
|
||||||
applyServerUpdate(result.project);
|
applyServerUpdate(result.project);
|
||||||
setVersion(result.version);
|
setServerState({ version: result.version, project: result.project });
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// Force refresh periodically
|
// Force refresh periodically
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const intervalId = window.setInterval(() => {
|
const intervalId = window.setInterval(() => {
|
||||||
refreshProject(project, version, options).then(handleAPIResponse);
|
console.time("server-sync");
|
||||||
|
refreshProject(project, serverState.version, options).then(handleAPIResponse)
|
||||||
|
.then(() => {
|
||||||
|
console.timeEnd("server-sync");
|
||||||
|
});
|
||||||
}, options.refreshInterval);
|
}, options.refreshInterval);
|
||||||
return () => clearInterval(intervalId);
|
return () => clearInterval(intervalId);
|
||||||
}, [version]);
|
}, [serverState]);
|
||||||
|
|
||||||
|
// Send delayed update
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const timeoutId =window.setTimeout(() => {
|
const timeoutId = window.setTimeout(() => {
|
||||||
console.log('executing debounced patch');
|
const patch = diff(serverState.project, project);
|
||||||
|
|
||||||
let previousProject: Project|any = usePrevious(project);
|
|
||||||
if (!previousProject) previousProject = {};
|
|
||||||
|
|
||||||
// Trigger patch if project has changed
|
|
||||||
|
|
||||||
const patch = diff(previousProject, project);
|
|
||||||
|
|
||||||
console.log('generated patch', patch);
|
|
||||||
|
|
||||||
if (!patch) return;
|
if (!patch) return;
|
||||||
|
patchProject(project, patch, serverState.version, options).then(handleAPIResponse);
|
||||||
patchProject(project, patch, version, options).then(handleAPIResponse);
|
}, 5000);
|
||||||
}, options.syncDelay);
|
|
||||||
return () => clearTimeout(timeoutId);
|
return () => clearTimeout(timeoutId);
|
||||||
});
|
}, [project, serverState]);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function refreshProject(project: Project, version: number, options: ServerSyncOptions): Promise<Response> {
|
function refreshProject(project: Project, version: number, options: ServerSyncOptions): Promise<Response> {
|
||||||
|
|
Loading…
Reference in New Issue