Skip to content

Commit

Permalink
Merge pull request #6 from GlueOps/fix/various-bug-fixes
Browse files Browse the repository at this point in the history
Fix/various bug fixes
  • Loading branch information
NichArchA82 authored Sep 7, 2024
2 parents 08f33fb + d70002c commit 3ec232b
Show file tree
Hide file tree
Showing 3 changed files with 145 additions and 90 deletions.
4 changes: 2 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ while the bolt server listens on port 3000 by default. Read more in
### Steps
1. pull down the latest image from ghcr

2. You can then run the bot `docker run --env-file <path-to-.env-file -d -p 5000:5000 -p 3000:3000 --name slack-bot <ghcr image>`
2. You can then run the bot `docker run --env-file <path-to-.env-file -d -p 5000:5000 -p 3000:3000 --name dev-bot <ghcr image>`
**If you change the bolt port or the server port, make sure to update that in the docker command**

3. You can find an example.env file in bot/example.env. Steps to generate all the tokens required and set up the bot are below.
Expand All @@ -30,7 +30,7 @@ If you would like to change the source code, or compile the slack Bot yourself:
3. Compile the bot code
`docker build -t <DOCKER_IMAGE_NAME> .`

4. You can then run the bot `docker run --env-file <path-to-.env-file -d -p 5000:5000 -p 3000:3000 --name slack-bot <ghcr image>`
4. You can then run the bot `docker run --env-file <path-to-.env-file -d -p 5000:5000 -p 3000:3000 --name dev-bot <ghcr image>`
**If you change the bolt port or the server port, make sure to update that in the docker command**

5. You can find an example.env file in bot/example.env. Steps to generate all the tokens required and set up the bot are below.
Expand Down
2 changes: 1 addition & 1 deletion bot/src/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ const app = new App({
signingSecret: process.env.SIGNING_SECRET,
socketMode: true,
appToken: process.env.APP_TOKEN,
logLevel: LogLevel.TRACE,
LogLevel: LogLevel.DEBUG,
logger: customLogger,
});

Expand Down
229 changes: 142 additions & 87 deletions command-handler/src/util/hetzner-servers.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,75 +14,112 @@ const delay = (ms) => {
}

const region = "hel1";
const userData = `#cloud-config\nruncmd:\n - ['sh', '-c', 'curl -fsSL https://tailscale.com/install.sh | sh']\n - ['tailscale', 'up', '--authkey=${process.env.TAILSCALE_AUTH_KEY}']\n - ['tailscale', 'set', '--ssh']\n - ['tailscale', 'set', '--accept-routes']\n - ['passwd', '-d', 'root']`
const serverType = 'cx42';
const image = 'debian-12';
const userData = `
#cloud-config
runcmd:
- ['sh', '-c', 'curl -fsSL https://tailscale.com/install.sh | sh']
- ['tailscale', 'up', '--authkey=${process.env.TAILSCALE_AUTH_KEY}']
- ['tailscale', 'set', '--ssh']
- ['tailscale', 'set', '--accept-routes']
- ['passwd', '-d', 'root']
`

export default {
//create the hetzner server
createServer: async ({ app, body }) => {
//auto generate the name
const serverName = uniqueNamesGenerator({
dictionaries: [colors, animals ],
dictionaries: [ colors, animals ],
separator: '-'
});

// Call the users.info method using the WebClient
const info = await app.client.users.info({
user: body.user.id
})
.catch(error => {
let info;
try {
info = await app.client.users.info({
user: body.user.id
});
} catch (error) {
log.error('There was an error calling the user.info method in slack', error);
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Failed to get user info from slack`
});

return;
}

const userEmail = formatUser(info.user.profile.email);

//hetzner api to create the server
await axios.post('https://api.hetzner.cloud/v1/servers',
{
"automount": false,
"image": "debian-12",
"labels": {
"owner": `${userEmail}`
},
"location": `${region}`,
"name": `${serverName}`,
"public_net": {
"enable_ipv4": true,
"enable_ipv6": false
},
"server_type": "cx42",
"start_after_create": true,
"user_data": userData
}, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`,
'Content-Type': 'application/json'
}
})
.catch(error => {
try {
await axios.post('https://api.hetzner.cloud/v1/servers',
{
"automount": false,
"image": image,
"labels": {
"owner": userEmail
},
"location": region,
"name": serverName,
"public_net": {
"enable_ipv4": true,
"enable_ipv6": false
},
"server_type": serverType,
"start_after_create": true,
"user_data": userData
}, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`,
'Content-Type': 'application/json'
}
});
} catch (error) {
log.error('There was an error creating the server', axiosError(error));
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Failed to create a server in hetzner.`
});

return;
}

//set 2 minute delay
await delay(1000 * 60 * 2);

//get servers and info from tailscale
const { deviceId, deviceIP } = await getDevices(serverName);

//set tag in tailscale
await axios.post(`https://api.tailscale.com/api/v2/device/${deviceId}/tags`,
{
"tags": [
`tag:${userEmail}`
]
}, {
headers: {
'Authorization': `Bearer ${process.env.TAILSCALE_API_TOKEN}`,
'Content-Type': 'application/json'
}
})
.catch(error => {
log.error('There was an error setting tags in tailscale', axiosError(error));
});
try {
await axios.post(`https://api.tailscale.com/api/v2/device/${deviceId}/tags`,
{
"tags": [
`tag:${userEmail}`
]
}, {
headers: {
'Authorization': `Bearer ${process.env.TAILSCALE_API_TOKEN}`,
'Content-Type': 'application/json'
}
});
} catch (error) {
log.error('There was an error setting tags in tailscale', axiosError(error));

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Failed to set tags in tailscale.`
});
}

//return info for tailscale
app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
Expand Down Expand Up @@ -116,23 +153,36 @@ export default {
})
.catch(error => {
log.error('Failed to delete device in tailscale', axiosError(error));

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Failed to delete Server: ${serverName} from Tailscale`
});
});

//delete server from hetzner
await axios.delete(`https://api.hetzner.cloud/v1/servers/${vmid}`, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`
}
})
.catch(error => {
try {
await axios.delete(`https://api.hetzner.cloud/v1/servers/${vmid}`, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`
}
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server: ${serverName} has been deleted.`
});
} catch (error) {
log.error('Failed to delete the server in hetzner', axiosError(error));
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server: ${serverName} has been deleted.`
});
app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Failed to delete Server: ${serverName} from Hetzner.`
});
}
},

//list the servers
Expand Down Expand Up @@ -202,49 +252,54 @@ export default {

//start a hetzner server
startServer: async ({ app, body, vmid }) => {
await axios.post(`https://api.hetzner.cloud/v1/servers/${vmid}/actions/poweron`, null, {
try {
await axios.post(`https://api.hetzner.cloud/v1/servers/${vmid}/actions/poweron`, null, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`
}
})
.catch(error => {
log.error('Error starting the server', axiosError(error));
app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} failed to start.`
});
return;
});


app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} has been started.`
});
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} has been started.`
});

} catch (error) {
log.error('Error starting the server', axiosError(error));
app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} failed to start.`
});

return;
}
},

//stop a hetzner server
stopServer: async ({ app, body, vmid }) => {
await axios.post(`https://api.hetzner.cloud/v1/servers/${vmid}/actions/poweroff`, null, {
try {
await axios.post(`https://api.hetzner.cloud/v1/servers/${vmid}/actions/poweroff`, null, {
headers: {
'Authorization': `Bearer ${process.env.HETZNER_API_TOKEN}`
}
})
.catch(error => {
log.error('Error stopping the server', axiosError(error));
app.client.chat.postEphemeral({
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} failed to stop.`
});
return;
});

app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} has been stopped.`
});
text: `Server Id: ${vmid} has been stopped.`
});
} catch (error) {
log.error('Error stopping the server', axiosError(error));
app.client.chat.postEphemeral({
channel: `${body.channel.id}`,
user: `${body.user.id}`,
text: `Server Id: ${vmid} failed to stop.`
});

return;
}
}
}

0 comments on commit 3ec232b

Please sign in to comment.