aboutsummaryrefslogtreecommitdiff
path: root/background/reverse_use_info.js
blob: 688cd644f30ace1d6aaf2ba6f7c186af9bee78c0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
/**
* Myext scripts and bags usage index
*
* Copyright (C) 2021 Wojtek Kosior
*
* Dual-licensed under:
*   - 0BSD license
*   - GPLv3 or (at your option) any later version
*/

"use strict";

/*
 * We want to count referenes to scripts and bags in order to know,
 * for example, whether one can be safely deleted.
 */

(() => {
    const TYPE_PREFIX = window.TYPE_PREFIX;
    const get_storage = window.get_storage;
    const make_once = window.make_once;

    var component_uses = new Map();
    var storage;

    function add_use_info_by_item(prefix, item, components)
    {
	for (let component of components) {
	    component = component.join("");

	    let used_by_info = component_uses.get(component);

	    if (used_by_info === undefined) {
		used_by_info = {};
		component_uses.set(component, used_by_info);
	    }

	    if (used_by_info[prefix] === undefined)
		used_by_info[prefix] = new Set();

	    used_by_info[prefix].add(item);
	}
    }

    function remove_use_info_by_item(prefix, item, components)
    {
	for (let component of components) {
	    used_by_info = component_uses.get(component.join(""));
	    if (used_by_info === undefined ||
		used_by_info[prefix] === undefined)
		return;

	    used_by_info[prefix].delete(item);
	}
    }

    function build_reverse_uses_info(entries_it, type_prefix)
    {
	for (let [item, components] of storage.get_all_it(type_prefix))
	    add_use_info_by_item(type_prefix, item, components);
    }

    function handle_change(change)
    {
	if (change.old_val !== undefined)
	    remove_use_info_by_item(change.prefix, change.item, change.old_val);

	if (change.new_val !== undefined)
	    add_use_info_by_item(change.prefix, change.item, change.new_val);
    }

    function get_uses(arg1, arg2=undefined)
    {
	let [prefix, item] = [arg1, arg2];

	if (arg2 === undefined)
	    [prefix, item] = arg1;

	return component_uses.get(prefix + item);
    }

    async function init()
    {
	storage = await get_storage();

	prefixes = [TYPE_PREFIX.PAGE, TYPE_PREFIX.BAG];
	for (let prefix of prefixes)
	    build_reverse_uses_info(prefix);

	storage.add_change_listener(handle_change, prefixes);

	return get_uses;
    }

    window.get_reverse_use_info = make_once(init);
})();