Project

General

Profile

Download (4.61 KB) Statistics
| Branch: | Tag: | Revision:

haketilo / common / storage_light.js @ 263d03d5

1
/**
2
 * This file is part of Haketilo.
3
 *
4
 * Function: Storage manager, lighter than the previous one.
5
 *
6
 * Copyright (C) 2021 Wojtek Kosior
7
 *
8
 * This program is free software: you can redistribute it and/or modify
9
 * it under the terms of the GNU General Public License as published by
10
 * the Free Software Foundation, either version 3 of the License, or
11
 * (at your option) any later version.
12
 *
13
 * This program is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16
 * GNU General Public License for more details.
17
 *
18
 * As additional permission under GNU GPL version 3 section 7, you
19
 * may distribute forms of that code without the copy of the GNU
20
 * GPL normally required by section 4, provided you include this
21
 * license notice and, in case of non-source distribution, a URL
22
 * through which recipients can access the Corresponding Source.
23
 * If you modify file(s) with this exception, you may extend this
24
 * exception to your version of the file(s), but you are not
25
 * obligated to do so. If you do not wish to do so, delete this
26
 * exception statement from your version.
27
 *
28
 * As a special exception to the GPL, any HTML file which merely
29
 * makes function calls to this code, and for that purpose
30
 * includes it by reference shall be deemed a separate work for
31
 * copyright law purposes. If you modify this code, you may extend
32
 * this exception to your version of the code, but you are not
33
 * obligated to do so. If you do not wish to do so, delete this
34
 * exception statement from your version.
35
 *
36
 * You should have received a copy of the GNU General Public License
37
 * along with this program.  If not, see <https://www.gnu.org/licenses/>.
38
 *
39
 * I, Wojtek Kosior, thereby promise not to sue for violation of this file's
40
 * license. Although I request that you do not make use this code in a
41
 * proprietary program, I am not going to enforce this in court.
42
 */
43

    
44
/*
45
 * IMPORTS_START
46
 * IMPORT TYPE_PREFIX
47
 * IMPORT raw_storage
48
 * IMPORT is_mozilla
49
 * IMPORT observables
50
 */
51

    
52
const reg_spec = new Set(["\\", "[", "]", "(", ")", "{", "}", ".", "*", "+"]);
53
const escape_reg_special = c => reg_spec.has(c) ? "\\" + c : c;
54

    
55
function make_regex(name)
56
{
57
    return new RegExp(`^${name.split("").map(escape_reg_special).join("")}\$`);
58
}
59

    
60
const listeners_by_callback = new Map();
61

    
62
function listen(callback, prefix, name)
63
{
64
    let by_prefix = listeners_by_callback.get(callback);
65
    if (!by_prefix) {
66
	by_prefix = new Map();
67
	listeners_by_callback.set(callback, by_prefix);
68
    }
69

    
70
    let by_name = by_prefix.get(prefix);
71
    if (!by_name) {
72
	by_name = new Map();
73
	by_prefix.set(prefix, by_name);
74
    }
75

    
76
    let name_reg = by_name.get(name);
77
    if (!name_reg) {
78
	name_reg = name.test ? name : make_regex(name);
79
	by_name.set(name, name_reg);
80
    }
81
}
82

    
83
function no_listen(callback, prefix, name)
84
{
85
    const by_prefix = listeners_by_callback.get(callback);
86
    if (!by_prefix)
87
	return;
88

    
89
    const by_name = by_prefix.get(prefix);
90
    if (!by_name)
91
	return;
92

    
93
    const name_reg = by_name.get(name);
94
    if (!name_reg)
95
	return;
96

    
97
    by_name.delete(name);
98

    
99
    if (by_name.size === 0)
100
	by_prefix.delete(prefix);
101

    
102
    if (by_prefix.size === 0)
103
	listeners_by_callback.delete(callback);
104
}
105

    
106
function storage_change_callback(changes, area)
107
{
108
    if (is_mozilla && area !== "local")
109
    {console.log("area", area);return;}
110

    
111
    for (const item of Object.keys(changes)) {
112
	for (const [callback, by_prefix] of listeners_by_callback.entries()) {
113
	    const by_name = by_prefix.get(item[0]);
114
	    if (!by_name)
115
		continue;
116

    
117
	    for (const reg of by_name.values()) {
118
		if (!reg.test(item.substring(1)))
119
		    continue;
120

    
121
		try {
122
		    callback(item, changes[item]);
123
		} catch(e) {
124
		    console.error(e);
125
		}
126
	    }
127
	}
128
    }
129
}
130

    
131
raw_storage.listen(storage_change_callback);
132

    
133

    
134
const created_observables = new Map();
135

    
136
async function observe(prefix, name)
137
{
138
    const observable = observables.make();
139
    const callback = (it, ch) => observables.set(observable, ch.newValue);
140
    listen(callback, prefix, name);
141
    created_observables.set(observable, [callback, prefix, name]);
142
    observables.silent_set(observable, await raw_storage.get(prefix + name));
143

    
144
    return observable;
145
}
146

    
147
const observe_var = name => observe(TYPE_PREFIX.VAR, name);
148

    
149
function no_observe(observable)
150
{
151
    no_listen(...created_observables.get(observable) || []);
152
    created_observables.delete(observable);
153
}
154

    
155
const light_storage = {};
156
Object.assign(light_storage, raw_storage);
157
Object.assign(light_storage,
158
	      {listen, no_listen, observe, observe_var, no_observe});
159

    
160
/*
161
 * EXPORTS_START
162
 * EXPORT light_storage
163
 * EXPORTS_END
164
 */
(14-14/16)