LP1825896 Store workstations in Hatch when available
authorBill Erickson <berickxx@gmail.com>
Tue, 23 Apr 2019 14:57:31 +0000 (07:57 -0700)
committerJason Boyer <JBoyer@eoli.info>
Fri, 13 Dec 2019 14:36:28 +0000 (09:36 -0500)
When Hatch is enabled, use Hatch for storing workstation registration
information.

If workstations are found in localStorage, they are merged into the
collection of workstations stored in hatch and removed from
localStorage.

Include DB udpate to add workstation setting 'eg.hatch.enable.printing'
so that it may live on the server.

Signed-off-by: Bill Erickson <berickxx@gmail.com>
Signed-off-by: Jason Boyer <JBoyer@eoli.info>
14 files changed:
Open-ILS/src/eg2/src/app/core/auth.service.ts
Open-ILS/src/eg2/src/app/core/hatch.service.ts [new file with mode: 0644]
Open-ILS/src/eg2/src/app/core/store.service.ts
Open-ILS/src/eg2/src/app/share/print/hatch.service.ts [deleted file]
Open-ILS/src/eg2/src/app/share/print/print.component.ts
Open-ILS/src/eg2/src/app/staff/admin/workstation/workstations/workstations.component.ts
Open-ILS/src/eg2/src/app/staff/login.component.ts
Open-ILS/src/eg2/src/app/staff/resolver.service.ts
Open-ILS/src/sql/Pg/950.data.seed-values.sql
Open-ILS/src/sql/Pg/upgrade/XXXX.data.hatch-enable-print.sql [new file with mode: 0644]
Open-ILS/web/js/ui/default/staff/admin/workstation/app.js
Open-ILS/web/js/ui/default/staff/app.js
Open-ILS/web/js/ui/default/staff/offline.js
Open-ILS/web/js/ui/default/staff/services/hatch.js

index dad2acd..a173d63 100644 (file)
@@ -286,22 +286,22 @@ export class AuthService {
         }
 
         return new Promise((resolve, reject) => {
-            const workstations =
-                this.store.getLocalItem('eg.workstation.all');
+            return this.store.getWorkstations().then(workstations => {
 
-            if (workstations) {
-                const ws = workstations.filter(
-                    w => Number(w.id) === Number(this.user().wsid()))[0];
+                if (workstations) {
+                    const ws = workstations.filter(
+                        w => Number(w.id) === Number(this.user().wsid()))[0];
 
-                if (ws) {
-                    this.activeUser.workstation = ws.name;
-                    this.workstationState = AuthWsState.VALID;
-                    return resolve();
+                    if (ws) {
+                        this.activeUser.workstation = ws.name;
+                        this.workstationState = AuthWsState.VALID;
+                        return resolve();
+                    }
                 }
-            }
 
-            this.workstationState = AuthWsState.NOT_FOUND_LOCAL;
-            reject();
+                this.workstationState = AuthWsState.NOT_FOUND_LOCAL;
+                reject();
+            });
         });
     }
 
diff --git a/Open-ILS/src/eg2/src/app/core/hatch.service.ts b/Open-ILS/src/eg2/src/app/core/hatch.service.ts
new file mode 100644 (file)
index 0000000..b61ee17
--- /dev/null
@@ -0,0 +1,124 @@
+import {Injectable, EventEmitter} from '@angular/core';
+
+export class HatchMessage {
+    msgid: number;
+    resolver: (HatchMessage) => void; // promise resolver
+    rejector: (HatchMessage) => void; // promise rejector
+    status: number;
+    message: string; // error message
+    from: string;
+    action: string;
+    settings: any;
+    content: string;
+    // Response from Hatch.
+    response: any;
+    contentType: string;
+    showDialog: boolean;
+
+    constructor(hash: any) {
+        if (hash) {
+            Object.keys(hash).forEach(key => this[key] = hash[key]);
+        }
+    }
+}
+
+@Injectable({providedIn: 'root'})
+export class HatchService {
+
+    isAvailable: boolean;
+    msgId: number;
+    messages: {[msgid: number]: HatchMessage};
+
+    constructor() {
+        this.isAvailable = null;
+        this.messages = {};
+        this.msgId = 1;
+    }
+
+    connect(): boolean {
+
+        if (this.isAvailable !== null) {
+            return this.isAvailable;
+        }
+
+        // When the Hatch extension loads, it tacks an attribute onto
+        // the top-level documentElement to indicate it's available.
+        if (!window.document.documentElement.getAttribute('hatch-is-open')) {
+            console.debug('Could not connect to Hatch');
+            return this.isAvailable = false;
+        }
+
+        window.addEventListener('message', event => {
+
+            // We only accept messages from our own content script.
+            if (event.source !== window) { return; }
+
+            // We only care about messages from the Hatch extension.
+            if (event.data && event.data.from === 'extension') {
+
+                // Avoid logging full Hatch responses. they can get large.
+                console.debug(
+                    `Hatch responded to message ID ${event.data.msgid}`);
+
+                this.handleResponse(event.data);
+            }
+        });
+
+        return this.isAvailable = true;
+    }
+
+    // Send a request from the browser to Hatch.
+    sendRequest(msg: HatchMessage): Promise<HatchMessage> {
+        if (this.isAvailable === false) {
+            return Promise.reject('Hatch is not connected');
+        }
+
+        msg.msgid = this.msgId++;
+        msg.from = 'page';
+        this.messages[msg.msgid] = msg;
+        window.postMessage(msg, window.location.origin);
+
+        return new Promise((resolve, reject) => {
+            msg.resolver = resolve;
+            msg.rejector = reject;
+        });
+    }
+
+    // Handle the data sent back to the browser from Hatch.
+    handleResponse(data: any) {
+
+        const msg = this.messages[data.msgid];
+        if (!msg) {
+            console.warn(`No Hatch request found with ID ${data.msgid}`);
+            return;
+        }
+
+        delete this.messages[data.msgid];
+        msg.response = data.content;
+        msg.message = data.message;
+        msg.status = Number(data.status);
+
+        if (msg.status === 200) {
+            msg.resolver(msg);
+        } else {
+            console.error(`Hatch request returned status ${msg.status}`, msg);
+            msg.rejector(msg);
+        }
+    }
+
+    getItem(key: string): Promise<any> {
+        const msg = new HatchMessage({action: 'get', key: key});
+        return this.sendRequest(msg).then((m: HatchMessage) => m.response);
+    }
+
+    setItem(key: string, val: any): Promise<any> {
+        const msg = new HatchMessage({action: 'set', key: key, content: val});
+        return this.sendRequest(msg).then((m: HatchMessage) => m.response);
+    }
+
+    removeItem(key: string): Promise<any> {
+        const msg = new HatchMessage({action: 'remove', key: key});
+        return this.sendRequest(msg).then((m: HatchMessage) => m.response);
+    }
+}
+
index 46dd621..7b239d2 100644 (file)
  */
 import {Injectable} from '@angular/core';
 import {CookieService} from 'ngx-cookie';
+import {HatchService} from './hatch.service';
+
+const WS_ALL_KEY = 'eg.workstation.all';
+const WS_DEF_KEY = 'eg.workstation.default';
 
 @Injectable({providedIn: 'root'})
 export class StoreService {
@@ -30,7 +34,8 @@ export class StoreService {
     ];
 
     constructor(
-        private cookieService: CookieService) {
+        private cookieService: CookieService,
+        private hatch: HatchService) {
     }
 
     private parseJson(valJson: string): any {
@@ -74,6 +79,31 @@ export class StoreService {
             {path : this.loginSessionBasePath, secure: true});
     }
 
+    setWorkstations(val: any, isJson?: boolean): Promise<any> {
+        if (this.hatch.isAvailable) {
+            return this.hatch.setItem(WS_ALL_KEY, val).then(
+                ok => {
+                    // When clearing workstations, remove the default.
+                    if (!val || val.length === 0) {
+                        return this.hatch.removeItem(WS_DEF_KEY);
+                    }
+                }
+            );
+        } else {
+            return Promise.resolve(
+                this.setLocalItem(WS_ALL_KEY, val, isJson));
+        }
+    }
+
+    setDefaultWorkstation(val: string, isJson?: boolean): Promise<any> {
+        if (this.hatch.isAvailable) {
+            return this.hatch.setItem(WS_DEF_KEY, val);
+        } else {
+            return Promise.resolve(
+                this.setLocalItem(WS_DEF_KEY, val, isJson));
+        }
+    }
+
     getLocalItem(key: string): any {
         return this.parseJson(window.localStorage.getItem(key));
     }
@@ -86,6 +116,79 @@ export class StoreService {
         return this.parseJson(this.cookieService.get(key));
     }
 
+    getWorkstations(): Promise<any> {
+        if (this.hatch.isAvailable) {
+            return this.mergeWorkstations().then(ok => {
+                this.removeLocalItem(WS_ALL_KEY);
+                return this.hatch.getItem(WS_ALL_KEY);
+            });
+        } else {
+            return Promise.resolve(this.getLocalItem(WS_ALL_KEY));
+        }
+    }
+
+    // See if any workstatoins are stored in local storage.  If so, also
+    // see if we have any stored in Hatch.  If both, merged workstations
+    // from localStorage in Hatch storage, skipping any whose name
+    // collide with a workstation in Hatch.  If none exist in Hatch,
+    // copy the localStorage workstations over wholesale.
+    mergeWorkstations(): Promise<any> {
+        const existing = this.getLocalItem(WS_ALL_KEY);
+
+        if (!existing || existing.length === 0) {
+            return Promise.resolve();
+        }
+
+        return this.hatch.getItem(WS_ALL_KEY).then(inHatch => {
+
+            if (!inHatch || inHatch.length === 0) {
+                // Nothing to merge, copy the data over directly
+                return this.hatch.setItem('eg.workstation.all', existing);
+            }
+
+            const addMe: any = [];
+            existing.forEach(ws => {
+                const match = inHatch.filter(w => w.name === ws.name)[0];
+                if (!match) {
+                    console.log(
+                        'Migrating workstation from local storage to hatch: '
+                        + ws.name
+                    );
+                    addMe.push(ws);
+                }
+            });
+            inHatch = inHatch.concat(addMe);
+            return this.hatch.setItem(WS_ALL_KEY, inHatch);
+        });
+    }
+
+    getDefaultWorkstation(): Promise<any> {
+        if (this.hatch.isAvailable) {
+            return this.hatch.getItem(WS_DEF_KEY).then(name => {
+                if (name) {
+                    // We have a default in Hatch, remove any lingering
+                    // value from localStorage.
+                    this.removeLocalItem(WS_DEF_KEY);
+                    return name;
+                } else {
+                    // Nothing in Hatch, see if we have a localStorage
+                    // value to migrate to Hatch
+                    name = this.getLocalItem(WS_DEF_KEY);
+                    if (name) {
+                        console.debug(
+                            'Migrating default workstation to Hatch ' + name);
+                        return this.hatch.setItem(WS_DEF_KEY, name)
+                        .then(ok => name);
+                    } else {
+                        return null;
+                    }
+                }
+            });
+        } else {
+            return Promise.resolve(this.getLocalItem(WS_DEF_KEY));
+        }
+    }
+
     removeLocalItem(key: string): void {
         window.localStorage.removeItem(key);
     }
@@ -98,6 +201,16 @@ export class StoreService {
         this.cookieService.remove(key, {path : this.loginSessionBasePath});
     }
 
+    removeDefaultWorkstation(val: string, isJson?: boolean): Promise<any> {
+        if (this.hatch.isAvailable) {
+            return this.hatch.removeItem(WS_DEF_KEY);
+        } else {
+            return Promise.resolve(
+                this.removeLocalItem(WS_DEF_KEY));
+        }
+    }
+
+
     clearLoginSessionItems(): void {
         this.loginSessionKeys.forEach(
             key => this.removeLoginSessionItem(key)
diff --git a/Open-ILS/src/eg2/src/app/share/print/hatch.service.ts b/Open-ILS/src/eg2/src/app/share/print/hatch.service.ts
deleted file mode 100644 (file)
index bd087b7..0000000
+++ /dev/null
@@ -1,109 +0,0 @@
-import {Injectable, EventEmitter} from '@angular/core';
-
-export class HatchMessage {
-    msgid: number;
-    resolver: (HatchMessage) => void; // promise resolver
-    rejector: (HatchMessage) => void; // promise rejector
-    status: number;
-    message: string; // error message
-    from: string;
-    action: string;
-    settings: any;
-    content: string;
-    // Response from Hatch.
-    response: any;
-    contentType: string;
-    showDialog: boolean;
-
-    constructor(hash: any) {
-        if (hash) {
-            Object.keys(hash).forEach(key => this[key] = hash[key]);
-        }
-    }
-}
-
-@Injectable()
-export class HatchService {
-
-    isAvailable: boolean;
-    msgId: number;
-    messages: {[msgid: number]: HatchMessage};
-
-    constructor() {
-        this.isAvailable = null;
-        this.messages = {};
-        this.msgId = 1;
-    }
-
-    connect(): boolean {
-
-        if (this.isAvailable !== null) {
-            return this.isAvailable;
-        }
-
-        // When the Hatch extension loads, it tacks an attribute onto
-        // the top-level documentElement to indicate it's available.
-        if (!window.document.documentElement.getAttribute('hatch-is-open')) {
-            console.warn('Could not connect to Hatch');
-            return this.isAvailable = false;
-        }
-
-        window.addEventListener('message', event => {
-
-            // We only accept messages from our own content script.
-            if (event.source !== window) { return; }
-
-            // We only care about messages from the Hatch extension.
-            if (event.data && event.data.from === 'extension') {
-
-                // Avoid logging full Hatch responses. they can get large.
-                console.debug(
-                    `Hatch responded to message ID ${event.data.msgid}`);
-
-                this.handleResponse(event.data);
-            }
-        });
-
-        return this.isAvailable = true;
-    }
-
-    // Send a request from the browser to Hatch.
-    sendRequest(msg: HatchMessage): Promise<HatchMessage> {
-        if (this.isAvailable === false) {
-            return Promise.reject('Hatch is not connected');
-        }
-
-        msg.msgid = this.msgId++;
-        msg.from = 'page';
-        this.messages[msg.msgid] = msg;
-        window.postMessage(msg, window.location.origin);
-
-        return new Promise((resolve, reject) => {
-            msg.resolver = resolve;
-            msg.rejector = reject;
-        });
-    }
-
-    // Handle the data sent back to the browser from Hatch.
-    handleResponse(data: any) {
-
-        const msg = this.messages[data.msgid];
-        if (!msg) {
-            console.warn(`No Hatch request found with ID ${data.msgid}`);
-            return;
-        }
-
-        delete this.messages[data.msgid];
-        msg.response = data.content;
-        msg.message = data.message;
-        msg.status = Number(data.status);
-
-        if (msg.status === 200) {
-            msg.resolver(msg);
-        } else {
-            console.error(`Hatch request returned status ${msg.status}`, msg);
-            msg.rejector(msg);
-        }
-    }
-}
-
index 20c0567..0d780f3 100644 (file)
@@ -2,7 +2,7 @@ import {Component, OnInit, TemplateRef, ElementRef, Renderer2} from '@angular/co
 import {PrintService, PrintRequest} from './print.service';
 import {StoreService} from '@eg/core/store.service';
 import {ServerStoreService} from '@eg/core/server-store.service';
-import {HatchService, HatchMessage} from './hatch.service';
+import {HatchService, HatchMessage} from '@eg/core/hatch.service';
 import {ToastService} from '@eg/share/toast/toast.service';
 import {StringService} from '@eg/share/string/string.service';
 
@@ -26,6 +26,8 @@ export class PrintComponent implements OnInit {
 
     printQueue: PrintRequest[];
 
+    useHatch: boolean;
+
     constructor(
         private renderer: Renderer2,
         private elm: ElementRef,
@@ -45,6 +47,9 @@ export class PrintComponent implements OnInit {
 
         this.htmlContainer =
             this.renderer.selectRootElement('#eg-print-html-container');
+
+        this.serverStore.getItem('eg.hatch.enable.printing')
+            .then(use => this.useHatch = use);
     }
 
     handlePrintRequest(printReq: PrintRequest) {
@@ -163,7 +168,7 @@ export class PrintComponent implements OnInit {
             show_dialog: printReq.showDialog
         });
 
-        if (this.useHatch()) {
+        if (this.useHatch) {
             this.printViaHatch(printReq);
         } else {
             // Here the needed HTML is already in the page.
@@ -171,11 +176,6 @@ export class PrintComponent implements OnInit {
         }
     }
 
-    useHatch(): boolean {
-        return this.store.getLocalItem('eg.hatch.enable.printing')
-            && this.hatch.connect();
-    }
-
     printViaHatch(printReq: PrintRequest) {
 
         // Send a full HTML document to Hatch
index 457fb94..afd7940 100644 (file)
@@ -50,13 +50,18 @@ export class WorkstationsComponent implements OnInit {
     ) {}
 
     ngOnInit() {
-        this.workstations = this.store.getLocalItem('eg.workstation.all') || [];
-        this.defaultName = this.store.getLocalItem('eg.workstation.default');
-        this.selectedName = this.auth.workstation() || this.defaultName;
-        const rm = this.route.snapshot.paramMap.get('remove');
-        if (rm) {
-            this.removeSelected(this.removeWorkstation = rm);
-        }
+        this.store.getWorkstations()
+
+        .then(wsList => {
+            this.workstations = wsList || [];
+            return this.store.getDefaultWorkstation();
+
+        }).then(def => {
+            this.defaultName = def;
+            this.selectedName = this.auth.workstation() || this.defaultName;
+            const rm = this.route.snapshot.paramMap.get('remove');
+            if (rm) { this.removeSelected(this.removeWorkstation = rm); }
+        });
 
         // TODO: use the org selector limitPerm option
         this.perm.hasWorkPermAt(['REGISTER_WORKSTATION'], true)
@@ -85,7 +90,7 @@ export class WorkstationsComponent implements OnInit {
     setDefault(): void {
       if (this.selected()) {
             this.defaultName = this.selected().name;
-            this.store.setLocalItem('eg.workstation.default', this.defaultName);
+            this.store.setDefaultWorkstation(this.defaultName);
         }
     }
 
@@ -95,11 +100,11 @@ export class WorkstationsComponent implements OnInit {
         }
 
         this.workstations = this.workstations.filter(w => w.name !== name);
-        this.store.setLocalItem('eg.workstation.all', this.workstations);
+        this.store.setWorkstations(this.workstations);
 
         if (this.defaultName === name) {
             this.defaultName = null;
-            this.store.removeLocalItem('eg.workstation.default');
+            this.store.removeWorkstations();
         }
     }
 
@@ -170,7 +175,7 @@ export class WorkstationsComponent implements OnInit {
         };
 
         this.workstations.push(ws);
-        this.store.setLocalItem('eg.workstation.all', this.workstations);
+        this.store.setWorkstations(this.workstations);
         this.newName = '';
         // when registering our first workstation, mark it as the
         // default and show it as selected in the ws selector.
index de889f9..2f2b931 100644 (file)
@@ -36,10 +36,14 @@ export class StaffLoginComponent implements OnInit {
         // Focus username
         this.renderer.selectRootElement('#username').focus();
 
-        this.workstations = this.store.getLocalItem('eg.workstation.all');
-        this.args.workstation =
-            this.store.getLocalItem('eg.workstation.default');
-        this.applyWorkstation();
+        this.store.getWorkstations()
+        .then(wsList => {
+            this.workstations = wsList;
+            return this.store.getDefaultWorkstation();
+        }).then(def => {
+            this.args.workstation = def;
+            this.applyWorkstation();
+        });
     }
 
     applyWorkstation() {
index 94b469e..853bf40 100644 (file)
@@ -9,6 +9,7 @@ import {AuthService, AuthWsState} from '@eg/core/auth.service';
 import {PermService} from '@eg/core/perm.service';
 import {OrgService} from '@eg/core/org.service';
 import {FormatService} from '@eg/core/format.service';
+import {HatchService} from '@eg/core/hatch.service';
 
 const LOGIN_PATH = '/staff/login';
 const WS_MANAGE_PATH = '/staff/admin/workstation/workstations/manage';
@@ -26,6 +27,7 @@ export class StaffResolver implements Resolve<Observable<any>> {
         private router: Router,
         private route: ActivatedRoute,
         private ngLocation: Location,
+        private hatch: HatchService,
         private store: StoreService,
         private org: OrgService,
         private net: NetService,
@@ -38,6 +40,8 @@ export class StaffResolver implements Resolve<Observable<any>> {
         route: ActivatedRouteSnapshot,
         state: RouterStateSnapshot): Observable<any> {
 
+        this.hatch.connect();
+
         // Staff cookies stay in /$base/staff/
         // NOTE: storing session data at '/' so it can be shared by
         // Angularjs apps.
index 32dcd0f..a49cf02 100644 (file)
@@ -20297,3 +20297,14 @@ VALUES (
     'Grid Config: circ.patron.xact_details_details_payments',
     'cwst', 'label')
 );
+
+INSERT INTO config.workstation_setting_type (name, grp, datatype, label)
+VALUES (
+    'eg.hatch.enable.printing', 'gui', 'bool',
+    oils_i18n_gettext(
+        'eg.hatch.enable.printing',
+        'Use Hatch for printing',
+        'cwst', 'label'
+    )
+);
+
diff --git a/Open-ILS/src/sql/Pg/upgrade/XXXX.data.hatch-enable-print.sql b/Open-ILS/src/sql/Pg/upgrade/XXXX.data.hatch-enable-print.sql
new file mode 100644 (file)
index 0000000..315c85b
--- /dev/null
@@ -0,0 +1,18 @@
+BEGIN;
+
+-- SELECT evergreen.upgrade_deps_block_check('TODO', :eg_version);
+
+INSERT INTO config.workstation_setting_type (name, grp, datatype, label)
+VALUES (
+    'eg.hatch.enable.printing', 'gui', 'bool',
+    oils_i18n_gettext(
+        'eg.hatch.enable.printing',
+        'Use Hatch for printing',
+        'cwst', 'label'
+    )
+);
+
+
+COMMIT;
+
+
index d863844..70f87b9 100644 (file)
@@ -72,16 +72,16 @@ function($q , $timeout , $location , egCore , egConfirmDialog) {
     var service = {};
 
     service.get_all = function() {
-        return egCore.hatch.getItem('eg.workstation.all')
+        return egCore.hatch.getWorkstations()
         .then(function(all) { return all || [] });
     }
 
     service.get_default = function() {
-        return egCore.hatch.getItem('eg.workstation.default');
+        return egCore.hatch.getDefaultWorkstation();
     }
 
     service.set_default = function(name) {
-        return egCore.hatch.setItem('eg.workstation.default', name);
+        return egCore.hatch.setDefaultWorkstation(name);
     }
 
     service.register_workstation = function(base_name, name, org_id) {
@@ -140,7 +140,7 @@ function($q , $timeout , $location , egCore , egConfirmDialog) {
         return service.get_all()
         .then(function(all) {
             all.push(new_ws);
-            return egCore.hatch.setItem('eg.workstation.all', all)
+            return egCore.hatch.setWorkstations(all)
             .then(function() { return new_ws });
         });
     }
@@ -150,13 +150,13 @@ function($q , $timeout , $location , egCore , egConfirmDialog) {
     service.remove_workstation = function(name) {
         console.debug('Removing workstation: ' + name);
 
-        return egCore.hatch.getItem('eg.workstation.all')
+        return egCore.hatch.getWorkstations()
 
         // remove from list of all workstations
         .then(function(all) {
             if (!all) all = [];
             var keep = all.filter(function(ws) {return ws.name != name});
-            return egCore.hatch.setItem('eg.workstation.all', keep)
+            return egCore.hatch.setWorkstations(keep);
 
         }).then(function() { 
 
@@ -165,7 +165,7 @@ function($q , $timeout , $location , egCore , egConfirmDialog) {
         }).then(function(def) {
             if (def == name) {
                 console.debug('Removing default workstation: ' + name);
-                return egCore.hatch.removeItem('eg.workstation.default');
+                return egCore.hatch.removeDefaultWorkstation();
             }
         });
     }
@@ -234,8 +234,13 @@ function($scope , egCore) {
     $scope.setContentType = function(type) { $scope.contentType = type }
     $scope.setContentType('text/plain');
 
+    var hatchPrinting = false;
+    egCore.hatch.usePrinting().then(function(answer) {
+        hatchPrinting = answer;
+    });
+
     $scope.useHatchPrinting = function() {
-        return egCore.hatch.usePrinting();
+        return hatchPrinting;
     }
 
     $scope.hatchIsOpen = function() {
@@ -959,15 +964,18 @@ function($scope , egCore , ngToast) {
     var hatch = egCore.hatch;  // convenience
 
     $scope.hatch_available = hatch.hatchAvailable;
-    $scope.hatch_printing = hatch.usePrinting();
     $scope.hatch_settings = hatch.useSettings();
     $scope.hatch_offline  = hatch.useOffline();
 
+    hatch.usePrinting().then(function(answer) {
+        $scope.hatch_printing = answer;
+    });
+
     // Apply Hatch settings as changes occur in the UI.
     
     $scope.$watch('hatch_printing', function(newval) {
         if (typeof newval != 'boolean') return;
-        hatch.setLocalItem('eg.hatch.enable.printing', newval);
+        hatch.setItem('eg.hatch.enable.printing', newval);
     });
 
     $scope.$watch('hatch_settings', function(newval) {
index 053f401..e38b76f 100644 (file)
@@ -60,7 +60,7 @@ function($routeProvider , $locationProvider) {
         // if the user is already logged in, jump to splash page
         if (egCore.auth.user()) $location.path('/');
 
-        egCore.hatch.getItem('eg.workstation.all')
+        egCore.hatch.getWorkstations()
         .then(function(all) {
             if (all && all.length) {
                 $scope.workstations = all.map(function(a) { return a.name });
@@ -79,7 +79,7 @@ function($routeProvider , $locationProvider) {
                     }
                 } else {
                     // no workstation requested; use the default
-                    egCore.hatch.getItem('eg.workstation.default')
+                    egCore.hatch.getDefaultWorkstation()
                     .then(function(ws) {
                         $scope.args = {workstation : ws}
                     });
index 2a3f820..21e10b0 100644 (file)
@@ -360,7 +360,7 @@ function($routeProvider , $locationProvider , $compileProvider) {
             if (setting !== undefined) $scope.do_check_changed = true;
         });
 
-        egCore.hatch.getItem('eg.workstation.all')
+        egCore.hatch.getWorkstations()
         .then(function(all) {
             if (all && all.length) {
                 $scope.workstations = all;
@@ -381,7 +381,7 @@ function($routeProvider , $locationProvider , $compileProvider) {
                     }
                 } else {
                     // no workstation requested; use the default
-                    egCore.hatch.getItem('eg.workstation.default')
+                    egCore.hatch.getDefaultWorkstation()
                     .then(function(ws) {
                         var ws_obj = all.filter(function(w) {
                             return ws == w.name
index 5b6f923..c7e2f39 100644 (file)
@@ -44,36 +44,11 @@ angular.module('egCoreMod')
     service.serverSettingSummaries = {};
 
     /**
-     * List string prefixes for On-Call storage keys. On-Call keys
-     * are those that can be set/get/remove'd from localStorage when
-     * Hatch is not avaialable, even though Hatch is configured as the
-     * primary storage location for the key in question.  On-Call keys
-     * are those that allow the user to login and perform basic admin
-     * tasks (like disabling Hatch) even when Hatch is down.
-     * AKA Browser Staff Run Level 3.
-     * Note that no attempt is made to synchronize data between Hatch
-     * and localStorage for On-Call keys.  Only one destation is active 
-     * at a time and each maintains its own data separately.
-     */
-    service.onCallPrefixes = ['eg.workstation'];
-    
-    // Returns true if the key can be set/get in localStorage even when 
-    // Hatch is not available.
-    service.keyIsOnCall = function(key) {
-        var oncall = false;
-        angular.forEach(service.onCallPrefixes, function(pfx) {
-            if (key.match(new RegExp('^' + pfx))) 
-                oncall = true;
-        });
-        return oncall;
-    }
-
-    /**
      * Settings with these prefixes will always live in the browser.
      */
     service.browserOnlyPrefixes = [
-        'eg.workstation', 
-        'eg.hatch',
+        'eg.hatch.enable.settings', // deprecated
+        'eg.hatch.enable.offline', // deprecated
         'eg.cache',
         'current_tag_table_marc21_biblio',
         'FFPos',
@@ -233,24 +208,169 @@ angular.module('egCoreMod')
         );
     }
 
-    // TODO: once Hatch is printing-only, should probably store
-    // this preference on the server.
     service.usePrinting = function() {
-        return service.getLocalItem('eg.hatch.enable.printing');
+        return service.getItem('eg.hatch.enable.printing');
     }
 
+    // DEPRECATED
     service.useSettings = function() {
         return service.getLocalItem('eg.hatch.enable.settings');
     }
 
+    // DEPRECATED
     service.useOffline = function() {
         return service.getLocalItem('eg.hatch.enable.offline');
     }
 
+    service.getWorkstations = function() {
+        if (service.hatchAvailable) {
+            return service.mergeWorkstations().then(
+                function() {
+                    service.removeLocalItem('eg.workstation.all');
+                    return service.getRemoteItem('eg.workstation.all');
+                }
+            );
+        } else {
+            return $q.when(service.getLocalItem('eg.workstation.all'));
+        }
+    }
+
+    // See if any workstations are stored in local storage.  If so, also
+    // see if we have any stored in Hatch.  If both, merged workstations
+    // from localStorage in Hatch storage, skipping any whose name
+    // collide with a workstation in Hatch.  If none exist in Hatch,
+    // copy the localStorage workstations over wholesale.
+    service.mergeWorkstations = function() {
+        var existing = service.getLocalItem('eg.workstation.all');
+
+        if (!existing || existing.length === 0) {
+            return $q.when();
+        }
+
+        return service.getRemoteItem('eg.workstation.all')
+        .then(function(inHatch) {
+
+            if (!inHatch || inHatch.length === 0) {
+                // Nothing to merge, copy the data over directly
+                console.debug('No workstations in hatch to merge');
+                return service.setRemoteItem('eg.workstation.all', existing);
+            }
+
+            var addMe = [];
+            existing.forEach(function(ws) {
+                var match = inHatch.filter(
+                    function(w) {return w.name === ws.name})[0];
+                if (!match) {
+                    console.log(
+                        'Migrating workstation from local storage to hatch: ' 
+                        + ws.name
+                    );
+                    addMe.push(ws);
+                }
+            });
+            inHatch = inHatch.concat(addMe);
+            return service.setRemoteItem('eg.workstation.all', inHatch);
+        });
+    }
+
+    service.getDefaultWorkstation = function() {
+
+        if (service.hatchAvailable) {
+            return service.getRemoteItem('eg.workstation.default')
+            .then(function(name) {
+                if (name) {
+                    // We have a default in Hatch, remove any lingering
+                    // value from localStorage.
+                    service.removeLocalItem('eg.workstation.default');
+                    return name;
+                }
+
+                name = service.getLocalItem('eg.workstation.default');
+                if (name) {
+                    console.log('Migrating default workstation to Hatch ' + name);
+                    return service.setRemoteItem('eg.workstation.default', name)
+                    .then(function() {return name;});
+                }
+
+                return null;
+            });
+        } else {
+            return $q.when(service.getLocalItem('eg.workstation.default'));
+        }
+    }
+
+    service.setWorkstations = function(workstations, isJson) {
+        if (service.hatchAvailable) {
+            return service.setRemoteItem('eg.workstation.all', workstations);
+        } else {
+            return $q.when(
+                service.setLocalItem('eg.workstation.all', workstations, isJson));
+        }
+    }
+
+    service.setDefaultWorkstation = function(name, isJson) {
+        if (service.hatchAvailable) {
+            return service.setRemoteItem('eg.workstation.default', name);
+        } else {
+            return $q.when(
+                service.setLocalItem('eg.workstation.default', name, isJson));
+        }
+    }
+
+    service.removeWorkstations = function() {
+        if (service.hatchAvailable) {
+            return service.removeRemoteItem('eg.workstation.all');
+        } else {
+            return $q.when(
+                service.removeLocalItem('eg.workstation.all'));
+        }
+    }
+
+    service.removeDefaultWorkstation = function() {
+        if (service.hatchAvailable) {
+            return service.removeRemoteItem('eg.workstation.default');
+        } else {
+            return $q.when(
+                service.removeLocalItem('eg.workstation.default'));
+        }
+    }
+
+
+    // Workstation actions always use Hatch when it's available
+    service.getWorkstationItem = function(key) {
+        if (service.hatchAvailable) {
+            return service.getRemoteItem(key);
+        } else {
+            return $q.when(service.getLocalItem(key));
+        }
+    }
+
+    service.setWorkstationItem = function(key, value) {
+        if (service.hatchAvailable) {
+            return service.setRemoteItem(key, value);
+        } else {
+            return $q.when(service.setLocalItem(key, value));
+        }
+    }
+
+    service.removeWorkstationItem = function(key) {
+        if (service.hatchAvailable) {
+            return service.removeRemoteItem(key);
+        } else {
+            return $q.when(service.removeLocalItem(key));
+        }
+    }
+
+    service.keyIsWorkstation = function(key) {
+        return Boolean(key.match(/eg.workstation/));
+    }
+
     // get the value for a stored item
     service.getItem = function(key) {
 
-        console.debug('getting item: ' + key);
+        if (service.keyIsWorkstation(key)) {
+            return service.getWorkstationItem(key);
+        }
 
         if (!service.keyStoredInBrowser(key)) {
             return service.getServerItem(key);
@@ -260,14 +380,7 @@ angular.module('egCoreMod')
 
         service.getBrowserItem(key).then(
             function(val) { deferred.resolve(val); },
-
             function() { // Hatch error
-                if (service.keyIsOnCall(key)) {
-                    console.warn("Unable to getItem from Hatch: " + key + 
-                        ". Retrieving item from local storage instead");
-                    deferred.resolve(service.getLocalItem(key));
-                }
-
                 deferred.reject("Unable to getItem from Hatch: " + key);
             }
         );
@@ -388,6 +501,10 @@ angular.module('egCoreMod')
      */
     service.setItem = function(key, value) {
 
+        if (service.keyIsWorkstation(key)) {
+            return service.setWorkstationItem(key, value);
+        }
+
         if (!service.keyStoredInBrowser(key)) {
             return service.setServerItem(key, value);
         }
@@ -397,13 +514,6 @@ angular.module('egCoreMod')
             function(val) {deferred.resolve(val);},
 
             function() { // Hatch error
-
-                if (service.keyIsOnCall(key)) {
-                    console.warn("Unable to setItem in Hatch: " + 
-                        key + ". Setting in local storage instead");
-
-                    deferred.resolve(service.setLocalItem(key, value));
-                }
                 deferred.reject("Unable to setItem in Hatch: " + key);
             }
         );
@@ -627,13 +737,6 @@ angular.module('egCoreMod')
         if (service.hatchAvailable)
             return service.appendRemoteItem(key, value);
 
-        if (service.keyIsOnCall(key)) {
-            console.warn("Unable to appendItem in Hatch: " + 
-                key + ". Setting in local storage instead");
-
-            return $q.when(service.appendLocalItem(key, value));
-        }
-
         console.error("Unable to appendItem in Hatch: " + key);
         return $q.reject();
     }
@@ -686,6 +789,10 @@ angular.module('egCoreMod')
     // remove a stored item
     service.removeItem = function(key) {
 
+        if (service.keyIsWorkstation(key)) {
+            return service.removeWorkstationItem(key);
+        }
+
         if (!service.keyStoredInBrowser(key)) {
             return service.removeServerItem(key);
         }
@@ -694,14 +801,6 @@ angular.module('egCoreMod')
         service.removeBrowserItem(key).then(
             function(response) {deferred.resolve(response);},
             function() { // Hatch error
-
-                if (service.keyIsOnCall(key)) {
-                    console.warn("Unable to removeItem from Hatch: " + key + 
-                        ". Removing item from local storage instead");
-
-                    deferred.resolve(service.removeLocalItem(key));
-                }
-
                 deferred.reject("Unable to removeItem from Hatch: " + key);
             }
         );