aboutsummaryrefslogtreecommitdiff
path: root/background/reverse_use_info.mjs
blob: c38c52fe0470b2e92031b1d8dcdbf35183684376 (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
/**
* Myext scripts and bundles usage index
*
* Copyright (C) 2021 Wojtek Kosior
*
* Dual-licensed under:
*   - 0BSD license
*   - GPLv3 or (at your option) any later version
*/

import TYPE_PREFIX from '/common/stored_types.mjs';
import get_storage from './storage.mjs';
import make_once from '/common/once.mjs';

"use strict";

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

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.BUNDLE];
    for (let prefix of prefixes)
	build_reverse_uses_info(prefix);

    storage.add_change_listener(handle_change, prefixes);

    return get_uses;
}

export default make_once(init);