aboutsummaryrefslogtreecommitdiff
path: root/docker/netiquette/src/injector/iptables.mjs
blob: 584b5603b2a2b871bf53b197d2fd7e7a4ea0782e (plain) (blame)
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
'use strict'

let l;
export default l = async (path, readFile, exec, log) => {

  const load_static_rules = async path => 
    (await readFile(path, 'utf-8'))
      .split('\n')
      .filter(e => e)

  const get_current_rules = async () => 
    (await exec('iptables -S INPUT'))
      .stdout
      .split('\n')
      .filter(e => e.match(/^-A INPUT/g))

  const compute_rules_to_add = (current, target) =>
    target.filter(r => !current.includes(r))

  const compute_rules_to_del = (current, target) =>
    current
      .filter(r => !target.includes(r))
      .map(r => r.replace(/^-A INPUT/g, '-D INPUT'))

  const update_rules = async (current, target) =>
    await Promise.all([
      ...compute_rules_to_del(current, target),
      ...compute_rules_to_add(current, target)
    ].map(r => exec(`iptables ${r}`)))

  const build_target_rules = (tag_list) => 
    tag_list
      .map(t => /^public_port=(\d+)(-(\d+))?\/(udp|tcp)/g.exec(t))
      .filter(t => t)
      .map(t => new Object({ start: t[1], stop: t[3], protocol: t[4] }))
      .map(t => t.stop
        ? `-A INPUT -p ${t.protocol} --match multiport --dports ${t.start}:${t.stop} -j ACCEPT`
        : `-A INPUT -p ${t.protocol} --dport ${t.start} -j ACCEPT`)

  const do_log = (tag_list, r) => {
    //log('[iptables]', tag_list)
    log(`[iptables] ran ${r.length} commands`)
  }
  
  const static_rules = path ? await load_static_rules(path) : []
  log(`[iptables] initialized with ${static_rules.length} static rules`)
  return async tag_list =>
    do_log(
      tag_list,
      await update_rules(
        await get_current_rules(),
        [...static_rules, ...build_target_rules(tag_list)]))
}