Merge pull request #2361 from balena-os/host-config-noproxy

Do not write `noProxy` to redsocks.conf
This commit is contained in:
flowzone-app[bot] 2024-08-08 16:39:55 +00:00 committed by GitHub
commit 6eafde9f59
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 136 additions and 5 deletions

View File

@ -94,7 +94,7 @@ export async function patch(
} }
if (conf.network.proxy != null) { if (conf.network.proxy != null) {
const targetConf = conf.network.proxy; const { noProxy, ...targetConf } = conf.network.proxy;
// It's possible for appIds to be an empty array, but patch shouldn't fail // It's possible for appIds to be an empty array, but patch shouldn't fail
// as it's not dependent on there being any running user applications. // as it's not dependent on there being any running user applications.
return updateLock.lock(appIds, { force }, async () => { return updateLock.lock(appIds, { force }, async () => {
@ -114,7 +114,7 @@ export async function patch(
redsocks: targetConf, redsocks: targetConf,
}, },
); );
await setProxy(patchedConf, targetConf.noProxy); await setProxy(patchedConf, noProxy);
}); });
} }
} }

View File

@ -157,7 +157,7 @@ export async function readProxy(): Promise<HostProxyConfig | undefined> {
// Build proxy object // Build proxy object
const proxy = { const proxy = {
...redsocksConf.redsocks, ...redsocksConf.redsocks,
...(noProxy.length && { noProxy }), ...(noProxy.length > 0 && { noProxy }),
}; };
// Assumes mandatory proxy config fields (type, ip, port) are present, // Assumes mandatory proxy config fields (type, ip, port) are present,

View File

@ -1,4 +1,5 @@
import { expect } from 'chai'; import { expect } from 'chai';
import { stripIndent } from 'common-tags';
import type { TestFs } from 'mocha-pod'; import type { TestFs } from 'mocha-pod';
import { testfs } from 'mocha-pod'; import { testfs } from 'mocha-pod';
import * as path from 'path'; import * as path from 'path';
@ -138,7 +139,7 @@ describe('host-config', () => {
}); });
it('patches proxy', async () => { it('patches proxy', async () => {
await patch({ const newConf = {
network: { network: {
proxy: { proxy: {
ip: 'example2.org', ip: 'example2.org',
@ -149,7 +150,8 @@ describe('host-config', () => {
noProxy: ['balena.io', '222.22.2.2'], noProxy: ['balena.io', '222.22.2.2'],
}, },
}, },
}); };
await patch(newConf);
const { network } = await get(); const { network } = await get();
expect(network).to.have.property('proxy'); expect(network).to.have.property('proxy');
expect(network.proxy).to.have.property('ip', 'example2.org'); expect(network.proxy).to.have.property('ip', 'example2.org');
@ -161,6 +163,32 @@ describe('host-config', () => {
'balena.io', 'balena.io',
'222.22.2.2', '222.22.2.2',
]); ]);
await expect(fs.readFile(redsocksConf, 'utf-8')).to.eventually.equal(
stripIndent`
base {
log_debug = off;
log_info = on;
log = stderr;
daemon = off;
redirector = iptables;
}
redsocks {
ip = example2.org;
port = 1090;
type = http-relay;
login = "bar";
password = "foo";
local_ip = 127.0.0.1;
local_port = 12345;
}` + '\n',
);
expect(await fs.readFile(noProxy, 'utf-8')).to.equal(stripIndent`
balena.io
222.22.2.2
`);
}); });
it('patches proxy fields specified while leaving unspecified fields unchanged', async () => { it('patches proxy fields specified while leaving unspecified fields unchanged', async () => {
@ -183,6 +211,34 @@ describe('host-config', () => {
'152.10.30.4', '152.10.30.4',
'253.1.1.0/16', '253.1.1.0/16',
]); ]);
await expect(fs.readFile(redsocksConf, 'utf-8')).to.eventually.equal(
stripIndent`
base {
log_debug = off;
log_info = on;
log = stderr;
daemon = off;
redirector = iptables;
}
redsocks {
ip = example2.org;
port = 1090;
type = socks5;
login = "foo";
password = "bar";
local_ip = 127.0.0.1;
local_port = 12345;
}` + '\n',
);
expect(await fs.readFile(noProxy, 'utf-8')).to.equal(
stripIndent`
152.10.30.4
253.1.1.0/16
` + '\n',
);
}); });
it('patches proxy to empty if input is empty', async () => { it('patches proxy to empty if input is empty', async () => {
@ -219,6 +275,81 @@ describe('host-config', () => {
expect(await fs.readFile(redsocksConf, 'utf-8')).to.equal(rawConf); expect(await fs.readFile(redsocksConf, 'utf-8')).to.equal(rawConf);
}); });
// Check that a bad configuration is fixed by a new patch
it('ignores unsupported fields when reading proxy', async () => {
const badConf =
stripIndent`
base {
log_debug = off;
log_info = on;
log = stderr;
daemon = off;
redirector = iptables;
}
redsocks {
ip = example2.org;
port = 1090;
type = socks5;
login = "foo";
password = "bar";
local_ip = 127.0.0.2;
local_port = 12345;
noProxy = bad.server.com
}` + '\n';
await fs.writeFile(redsocksConf, badConf);
await fs.writeFile(noProxy, 'bad.server.com');
await expect(get()).to.eventually.deep.equal({
network: {
hostname: 'deadbeef',
proxy: {
ip: 'example2.org',
port: 1090,
type: 'socks5',
login: 'foo',
password: 'bar',
noProxy: ['bad.server.com'],
},
},
});
await patch({
network: {
proxy: {
ip: 'example2.org',
noProxy: ['bad.server.com'],
} as any,
},
});
await expect(fs.readFile(redsocksConf, 'utf-8')).to.eventually.equal(
stripIndent`
base {
log_debug = off;
log_info = on;
log = stderr;
daemon = off;
redirector = iptables;
}
redsocks {
ip = example2.org;
port = 1090;
type = socks5;
login = "foo";
password = "bar";
local_ip = 127.0.0.1;
local_port = 12345;
}` + '\n',
);
expect(await fs.readFile(noProxy, 'utf-8')).to.equal(
stripIndent`
bad.server.com
`,
);
});
it('skips restarting proxy services when part of redsocks-conf.target', async () => { it('skips restarting proxy services when part of redsocks-conf.target', async () => {
(dbus.servicePartOf as SinonStub).resolves(['redsocks-conf.target']); (dbus.servicePartOf as SinonStub).resolves(['redsocks-conf.target']);
await patch({ await patch({