2019-05-15 11:37:38 +00:00
|
|
|
#!/usr/bin/env node
|
|
|
|
|
|
|
|
if (!process.argv[2] || ['help', '-h', '--help'].includes(process.argv[2])) {
|
|
|
|
console.log(`
|
|
|
|
Sync changes in the javascript code to a running local mode supervisor on a device on the local network
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
./sync-debug.js <device IP>
|
|
|
|
|
|
|
|
Note that the device should be running a debug image.
|
|
|
|
`);
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
const ip = process.argv[2];
|
|
|
|
|
|
|
|
const { Livepush } = require('livepush');
|
|
|
|
const { fs } = require('mz');
|
|
|
|
const dockerode = require('dockerode');
|
|
|
|
const chokidar = require('chokidar');
|
|
|
|
const _ = require('lodash');
|
|
|
|
|
2019-06-13 12:20:24 +00:00
|
|
|
let lastReadTimestamp = null;
|
|
|
|
const setupLogs = async (containerId, docker) => {
|
|
|
|
const container = docker.getContainer(containerId);
|
|
|
|
const stream = await container.logs({
|
|
|
|
stdout: true,
|
|
|
|
stderr: true,
|
|
|
|
follow: true,
|
|
|
|
timestamps: true,
|
|
|
|
// We start from 0, as we risk not getting any logs to
|
|
|
|
// properly seed the value if the host and remote times differ
|
|
|
|
since: lastReadTimestamp != null ? lastReadTimestamp : 0,
|
|
|
|
});
|
|
|
|
stream.on('data', chunk => {
|
|
|
|
const { message, timestamp } = extractMessage(chunk);
|
|
|
|
lastReadTimestamp = Math.floor(timestamp.getTime() / 1000);
|
|
|
|
process.stdout.write(message);
|
|
|
|
});
|
|
|
|
stream.on('end', () => {
|
|
|
|
setupLogs(containerId, docker);
|
2019-05-30 22:30:49 +00:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-05-15 11:37:38 +00:00
|
|
|
function extractMessage(msgBuf) {
|
|
|
|
// Non-tty message format from:
|
|
|
|
// https://docs.docker.com/engine/api/v1.30/#operation/ContainerAttach
|
2019-06-13 12:20:24 +00:00
|
|
|
if (_.includes([0, 1, 2], msgBuf[0])) {
|
2019-05-15 11:37:38 +00:00
|
|
|
// Take the header from this message, and parse it as normal
|
|
|
|
msgBuf = msgBuf.slice(8);
|
|
|
|
}
|
2019-06-13 12:20:24 +00:00
|
|
|
const str = msgBuf.toString();
|
|
|
|
const space = str.indexOf(' ');
|
|
|
|
return {
|
|
|
|
timestamp: new Date(str.slice(0, space)),
|
|
|
|
message: str.slice(space + 1),
|
|
|
|
};
|
2019-05-15 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
2019-06-13 12:20:24 +00:00
|
|
|
const docker = new dockerode({
|
|
|
|
host: ip,
|
|
|
|
port: 2375,
|
|
|
|
});
|
|
|
|
|
2019-05-29 11:44:48 +00:00
|
|
|
let changedFiles = [];
|
|
|
|
let deletedFiles = [];
|
|
|
|
|
2019-06-13 12:20:24 +00:00
|
|
|
const performLivepush = _.debounce(async livepush => {
|
2019-05-29 11:44:48 +00:00
|
|
|
await livepush.performLivepush(changedFiles, deletedFiles);
|
|
|
|
changedFiles = [];
|
|
|
|
deletedFiles = [];
|
|
|
|
}, 1000);
|
|
|
|
|
2019-05-15 11:37:38 +00:00
|
|
|
(async () => {
|
2019-05-29 11:44:48 +00:00
|
|
|
console.log('Starting up...');
|
2019-05-15 11:37:38 +00:00
|
|
|
// Get the supervisor container id
|
|
|
|
const container = await docker.getContainer('resin_supervisor').inspect();
|
2019-05-29 11:44:48 +00:00
|
|
|
console.log('Supervisor container id: ', container.Id);
|
2019-05-15 11:37:38 +00:00
|
|
|
const containerId = container.Id;
|
|
|
|
const image = container.Image;
|
|
|
|
|
2019-06-13 12:20:24 +00:00
|
|
|
setupLogs(containerId, docker);
|
|
|
|
|
2019-05-15 11:37:38 +00:00
|
|
|
const livepush = await Livepush.init(
|
|
|
|
await fs.readFile('Dockerfile.debug'),
|
|
|
|
'.',
|
|
|
|
containerId,
|
|
|
|
// a bit of a hack, as the multistage images aren't
|
|
|
|
// present, but it shouldn't make a difference as these
|
|
|
|
// will never change
|
2019-07-23 14:00:39 +00:00
|
|
|
_.times(6, () => image),
|
2019-05-15 11:37:38 +00:00
|
|
|
docker,
|
|
|
|
);
|
|
|
|
|
|
|
|
chokidar
|
|
|
|
.watch('.', {
|
|
|
|
ignored: /((^|[\/\\])\..|node_modules.*)/,
|
2019-05-29 11:44:48 +00:00
|
|
|
ignoreInitial: false,
|
2019-05-15 11:37:38 +00:00
|
|
|
})
|
|
|
|
.on('add', path => {
|
2019-05-29 11:44:48 +00:00
|
|
|
changedFiles.push(path);
|
2019-05-30 22:30:49 +00:00
|
|
|
performLivepush(livepush, containerId, docker);
|
2019-05-15 11:37:38 +00:00
|
|
|
})
|
|
|
|
.on('change', path => {
|
2019-05-29 11:44:48 +00:00
|
|
|
changedFiles.push(path);
|
2019-05-30 22:30:49 +00:00
|
|
|
performLivepush(livepush, containerId, docker);
|
2019-05-15 11:37:38 +00:00
|
|
|
})
|
|
|
|
.on('unlink', path => {
|
2019-05-29 11:44:48 +00:00
|
|
|
deletedFiles.push(path);
|
2019-05-30 22:30:49 +00:00
|
|
|
performLivepush(livepush, containerId, docker);
|
2019-05-15 11:37:38 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
livepush.on('commandExecute', ({ command }) => {
|
|
|
|
console.log('SYNC: executing:', command);
|
|
|
|
});
|
|
|
|
livepush.on('commandOutput', ({ output }) => {
|
2019-06-13 12:20:24 +00:00
|
|
|
const message = output.data.toString();
|
|
|
|
if (message.trim().length !== 0) {
|
|
|
|
process.stdout.write(`\t${message}`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
livepush.on('commandReturn', ({ returnCode }) => {
|
|
|
|
if (returnCode !== 0) {
|
|
|
|
console.log(`\tSYNC: Command return non zero exit status: ${returnCode}`);
|
|
|
|
}
|
2019-05-15 11:37:38 +00:00
|
|
|
});
|
|
|
|
livepush.on('containerRestart', () => {
|
|
|
|
console.log('SYNC: Restarting container...');
|
|
|
|
});
|
|
|
|
})();
|