forked from hden/gke-static-ip
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
83 lines (77 loc) · 2.98 KB
/
index.js
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
require('dotenv').config()
const _ = require('underscore')
const { find } = require('obj-case')
const debug = console.log.bind(console)
const Compute = require('@google-cloud/compute')
const compute = new Compute({ projectId: process.env.PROJECT_ID })
async function getStaticIPs (prefix = '') {
const items = await compute.getAddresses({ filter: `name eq ${prefix}.*` }).then(_.flatten)
const reservedIPs = new Set()
const staticIPs = new Map(await Promise.all(items.map(async function (address) {
const metadata = await address.getMetadata().then(_.flatten)
const ip = find(metadata, '0.address')
const status = find(metadata, '0.status')
if (status === 'RESERVED') {
reservedIPs.add(ip)
}
return [ip, status]
})))
return { staticIPs, reservedIPs }
}
async function getInstances (prefix = '') {
const items = await compute.getVMs({ filter: `name eq ${prefix}.*` }).then(_.flatten)
return new Map(await Promise.all(items.map(async function (instance) {
const metadata = await instance.getMetadata().then(_.flatten)
const accessConfig = find(metadata, '0.networkInterfaces.0.accessConfigs.0')
return [instance, accessConfig]
})))
}
function request (instance, obj = {}) {
return new Promise((resolve, reject) => {
instance.request(obj, (e, operation) => {
if (e) {
return reject(e)
}
operation.once('error', reject)
operation.once('complete', resolve)
})
})
}
function deleteAccessConfig (instance, accessConfig = 'external-nat', networkInterface = 'nic0') {
return request(instance, {
method: 'POST',
uri: '/deleteAccessConfig',
qs: { networkInterface, accessConfig }
})
}
function addAccessConfig (instance, json = {}, networkInterface = 'nic0') {
return request(instance, {
method: 'POST',
uri: '/addAccessConfig',
qs: { networkInterface },
json
})
}
exports.main = async function main () {
const instances = await getInstances(process.env.INSTANCE_PREFIX)
const { staticIPs, reservedIPs } = await getStaticIPs(process.env.IP_PREFIX)
debug('Found %d reserved IP.', reservedIPs.size)
if ((instances.size > 0) && (reservedIPs.size > 0)) {
const availableIPs = Array.from(reservedIPs.values())
for (let [instance, accessConfig] of instances) {
const currentIP = find(accessConfig, 'natIP')
if (!staticIPs.has(currentIP) && (availableIPs.length > 0)) {
// Try to replace the ephemeral IP.
const reservedIP = availableIPs.pop()
debug('Replcing %s\'s IP from %s to %s.', instance.name, currentIP, reservedIP)
if (currentIP) {
// The access config is immutable, so it must be deleted first.
await deleteAccessConfig(instance, process.env.ACCESS_CONFIG_NAME)
debug('Removed existing IP for %s', instance.name)
}
await addAccessConfig(instance, Object.assign({}, accessConfig, { natIP: reservedIP }))
debug('Assigned reserved IP %s for instance %s.', reservedIP, instance.name)
}
}
}
}