2020-06-15 16:46:33 +00:00
|
|
|
import _ = require('lodash');
|
|
|
|
import { expect } from 'chai';
|
2020-07-10 14:02:07 +00:00
|
|
|
import { stub } from 'sinon';
|
|
|
|
import { child_process } from 'mz';
|
2020-06-15 16:46:33 +00:00
|
|
|
|
|
|
|
import * as firewall from '../../src/lib/firewall';
|
|
|
|
import * as iptables from '../../src/lib/iptables';
|
|
|
|
import { EventEmitter } from 'events';
|
2020-07-10 14:02:07 +00:00
|
|
|
import { Writable } from 'stream';
|
2020-06-15 16:46:33 +00:00
|
|
|
|
|
|
|
class FakeRuleAdaptor {
|
|
|
|
private rules: iptables.Rule[];
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
this.rules = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
public getRuleAdaptor(): iptables.RuleAdaptor {
|
|
|
|
return this.ruleAdaptor.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
private async ruleAdaptor(rules: iptables.Rule[]): Promise<void> {
|
|
|
|
const handleRule = async (rule: iptables.Rule) => {
|
|
|
|
// remove any undefined values from the object...
|
|
|
|
for (const key of Object.getOwnPropertyNames(rule)) {
|
|
|
|
if ((rule as any)[key] === undefined) {
|
|
|
|
delete (rule as any)[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.rules.push(rule);
|
|
|
|
return '';
|
|
|
|
};
|
|
|
|
|
|
|
|
if (_.isArray(rules)) {
|
|
|
|
for (const rule of rules) {
|
|
|
|
await handleRule(rule);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private isSameRule(
|
|
|
|
partial: Partial<iptables.Rule>,
|
|
|
|
rule: iptables.Rule,
|
|
|
|
): boolean {
|
|
|
|
const props = Object.getOwnPropertyNames(partial);
|
|
|
|
for (const prop of props) {
|
|
|
|
if (
|
|
|
|
_.get(rule, prop) === undefined ||
|
|
|
|
!_.isEqual(_.get(rule, prop), _.get(partial, prop))
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public expectRule(testRule: Partial<iptables.Rule>) {
|
|
|
|
return expect(
|
|
|
|
_.some(this.rules, (r) => this.isSameRule(testRule, r)),
|
|
|
|
).to.eq(
|
|
|
|
true,
|
|
|
|
`Rule has not been applied: ${JSON.stringify(
|
|
|
|
testRule,
|
|
|
|
)}\n\n${JSON.stringify(this.rules, null, 2)}`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
public expectNoRule(testRule: Partial<iptables.Rule>) {
|
|
|
|
return expect(
|
|
|
|
_.some(this.rules, (r) => this.isSameRule(testRule, r)),
|
|
|
|
).to.eq(
|
|
|
|
false,
|
|
|
|
`Rule has been applied: ${JSON.stringify(testRule)}\n\n${JSON.stringify(
|
|
|
|
this.rules,
|
|
|
|
null,
|
|
|
|
2,
|
|
|
|
)}`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
public clearHistory() {
|
|
|
|
this.rules = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-10 14:02:07 +00:00
|
|
|
export const realRuleAdaptor = iptables.getDefaultRuleAdaptor();
|
|
|
|
|
|
|
|
const fakeRuleAdaptorManager = new FakeRuleAdaptor();
|
|
|
|
const fakeRuleAdaptor = fakeRuleAdaptorManager.getRuleAdaptor();
|
|
|
|
|
2020-06-15 16:46:33 +00:00
|
|
|
// @ts-expect-error Assigning to a RO property
|
2020-07-10 14:02:07 +00:00
|
|
|
iptables.getDefaultRuleAdaptor = () => {
|
|
|
|
return fakeRuleAdaptor;
|
|
|
|
};
|
2020-06-15 16:46:33 +00:00
|
|
|
|
|
|
|
export interface MockedState {
|
|
|
|
hasAppliedRules: Promise<void>;
|
|
|
|
expectRule: (rule: iptables.Rule) => void;
|
|
|
|
expectNoRule: (rule: iptables.Rule) => void;
|
|
|
|
clearHistory: () => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type MockedConext = (state: MockedState) => Promise<any>;
|
|
|
|
|
|
|
|
const applyFirewallRules = firewall.applyFirewallMode;
|
2020-07-10 14:02:07 +00:00
|
|
|
export const whilstMocked = async (
|
|
|
|
context: MockedConext,
|
|
|
|
ruleAdaptor: iptables.RuleAdaptor = fakeRuleAdaptor,
|
|
|
|
) => {
|
|
|
|
const getOriginalDefaultRuleAdaptor = iptables.getDefaultRuleAdaptor;
|
|
|
|
|
|
|
|
const spawnStub = stub(child_process, 'spawn').callsFake(() => {
|
|
|
|
const fakeProc = new EventEmitter();
|
|
|
|
(fakeProc as any).stdout = new EventEmitter();
|
|
|
|
|
|
|
|
const stdin = new Writable();
|
|
|
|
stdin._write = (
|
|
|
|
chunk: Buffer,
|
|
|
|
_encoding: string,
|
|
|
|
callback: (err?: Error) => void,
|
|
|
|
) => {
|
|
|
|
console.log(chunk.toString('utf8'));
|
|
|
|
callback();
|
|
|
|
fakeProc.emit('close', 1);
|
|
|
|
};
|
|
|
|
(fakeProc as any).stdin = stdin;
|
|
|
|
|
|
|
|
return fakeProc as any;
|
|
|
|
});
|
|
|
|
|
|
|
|
// @ts-expect-error Assigning to a RO property
|
|
|
|
iptables.getDefaultRuleAdaptor = () => {
|
|
|
|
return ruleAdaptor;
|
|
|
|
};
|
|
|
|
|
|
|
|
fakeRuleAdaptorManager.clearHistory();
|
2020-06-15 16:46:33 +00:00
|
|
|
|
|
|
|
const applied = new EventEmitter();
|
|
|
|
|
|
|
|
// @ts-expect-error Assigning to a RO property
|
|
|
|
firewall.applyFirewallMode = async (mode: string) => {
|
|
|
|
await applyFirewallRules(mode);
|
|
|
|
applied.emit('applied');
|
|
|
|
};
|
|
|
|
|
|
|
|
await context({
|
2020-07-10 14:02:07 +00:00
|
|
|
expectRule: (rule) => fakeRuleAdaptorManager.expectRule(rule),
|
|
|
|
expectNoRule: (rule) => fakeRuleAdaptorManager.expectNoRule(rule),
|
|
|
|
clearHistory: () => fakeRuleAdaptorManager.clearHistory(),
|
2020-06-15 16:46:33 +00:00
|
|
|
hasAppliedRules: new Promise((resolve) => {
|
|
|
|
applied.once('applied', () => resolve());
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
|
|
|
|
// @ts-expect-error Assigning to a RO property
|
|
|
|
firewall.applyFirewallMode = applyFirewallRules;
|
2020-07-10 14:02:07 +00:00
|
|
|
|
|
|
|
spawnStub.restore();
|
|
|
|
|
|
|
|
// @ts-expect-error Assigning to a RO property
|
|
|
|
iptables.getDefaultRuleAdaptor = getOriginalDefaultRuleAdaptor;
|
2020-06-15 16:46:33 +00:00
|
|
|
};
|