import {DocEntry} from 'app/common/DocListAPI';
import {DocSnapshots} from 'app/common/DocSnapshot';
import {DocumentUsage} from 'app/common/DocUsage';
import {DocReplacementOptions} from 'app/common/UserAPI';

export interface IDocStorageManager {
  getPath(docName: string): string;
  getSampleDocPath(sampleDocName: string): string|null;
  getCanonicalDocName(altDocName: string): Promise<string>;

  // This method must not be called for the same docName twice in parallel.
  // In the current implementation, it is called in the context of an
  // AsyncCreate[docName].
  prepareLocalDoc(docName: string): Promise<boolean>;
  prepareToCreateDoc(docName: string): Promise<void>;
  prepareFork(srcDocName: string, destDocName: string): Promise<string>;  // Returns filename.

  listDocs(): Promise<DocEntry[]>;
  deleteDoc(docName: string, deletePermanently?: boolean): Promise<void>;
  renameDoc(oldName: string, newName: string): Promise<void>;
  makeBackup(docName: string, backupTag: string): Promise<string>;
  showItemInFolder(docName: string): Promise<void>;
  closeStorage(): Promise<void>;
  closeDocument(docName: string): Promise<void>;
  // Mark document as needing a backup (due to edits, migrations, etc).
  // If reason is set to 'edit' the user-facing timestamp on the document should be updated.
  markAsChanged(docName: string, reason?: 'edit'): void;
  scheduleUsageUpdate(docName: string, usage: DocumentUsage|null, minimizeDelay?: boolean): void;
  testReopenStorage(): void;                // restart storage during tests
  addToStorage(docName: string): Promise<void>;  // add a new local document to storage
  prepareToCloseStorage(): void;            // speed up sync with remote store
  getCopy(docName: string): Promise<string>;  // get an immutable copy of a document

  flushDoc(docName: string): Promise<void>; // flush a document to persistent storage
  // If skipMetadataCache is set, then any caching of snapshots lists should be skipped.
  // Metadata may not be returned in this case.
  getSnapshots(docName: string, skipMetadataCache?: boolean): Promise<DocSnapshots>;
  removeSnapshots(docName: string, snapshotIds: string[]): Promise<void>;
  // Get information about how snapshot generation is going.
  getSnapshotProgress(docName: string): SnapshotProgress;
  replace(docName: string, options: DocReplacementOptions): Promise<void>;
}

/**
 * A very minimal implementation of IDocStorageManager that is just
 * enough to allow an ActiveDoc to open and get to work.
 */
export class TrivialDocStorageManager implements IDocStorageManager {
  public getPath(docName: string): string { return docName; }
  public getSampleDocPath() { return null; }
  public async getCanonicalDocName(altDocName: string) { return altDocName; }
  public async prepareLocalDoc() { return false; }
  public async prepareToCreateDoc() { }
  public async prepareFork(): Promise<never> { throw new Error('no'); }
  public async listDocs() { return []; }
  public async deleteDoc(): Promise<never> { throw new Error('no'); }
  public async renameDoc(): Promise<never> { throw new Error('no'); }
  public async makeBackup(): Promise<never> { throw new Error('no'); }
  public async showItemInFolder(): Promise<never> { throw new Error('no'); }
  public async closeStorage() {}
  public async closeDocument() {}
  public markAsChanged() {}
  public scheduleUsageUpdate() {}
  public testReopenStorage() {}
  public async addToStorage(): Promise<never> { throw new Error('no'); }
  public prepareToCloseStorage() {}
  public async getCopy(): Promise<never> { throw new Error('no'); }
  public async flushDoc() {}
  public async getSnapshots(): Promise<never> { throw new Error('no'); }
  public async removeSnapshots(): Promise<never> { throw new Error('no'); }
  public getSnapshotProgress(): SnapshotProgress { return new EmptySnapshotProgress(); }
  public async replace(): Promise<never> { throw new Error('no'); }
}


/**
 * Some summary information about how snapshot generation is going.
 * Any times are in ms.
 * All information is within the lifetime of a doc worker, not global.
 */
export interface SnapshotProgress {
  /** The last time the document was marked as having changed. */
  lastChangeAt?: number;

  /**
   * The last time a save window started for the document (checking to see
   * if it needs to be pushed, and pushing it if so, possibly waiting
   * quite some time to bundle any other changes).
   */
  lastWindowStartedAt?: number;

  /**
   * The last time the document was either pushed or determined to not
   * actually need to be pushed, after having been marked as changed.
   */
  lastWindowDoneAt?: number;

  /** Number of times the document was pushed. */
  pushes: number;

  /** Number of times the document was not pushed because no change found. */
  skippedPushes: number;

  /** Number of times there was an error trying to push. */
  errors: number;

  /**
   * Number of times the document was marked as changed.
   * Will generally be a lot greater than saves.
   */
  changes: number;

  /** Number of times a save window was started. */
  windowsStarted: number;

  /** Number of times a save window was completed. */
  windowsDone: number;
}

export class EmptySnapshotProgress implements SnapshotProgress {
  public pushes: number = 0;
  public skippedPushes: number = 0;
  public errors: number = 0;
  public changes: number = 0;
  public windowsStarted: number = 0;
  public windowsDone: number = 0;
}