-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.js
157 lines (156 loc) · 4.87 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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
require('dotenv').config()
const { Client, Intents } = require('discord.js');
const client = new Client({ intents: [Intents.FLAGS.GUILDS] });
const dot = require('dot-object')
const SockJS = require('sockjs-client')
const axios = require('axios')
const socks = [new SockJS('https://screeps.com/socket/'),new SockJS('https://screeps.com/season/socket/')]
const baseUrl = 'https://screeps.com'
//change the type of annoncement here, these key words will trigger from the console in screeps
const typesOfAnnouncement = ['announcement','report','transfer','defence alert','local transfer','local away transfer','socket']
let user = process.env.SCREEPS_USER
let userId = undefined
let screepsToken = process.env.SCREEPS_TOKEN
const messageInterval = 1
main()
async function main(){
await getUserId()
if (userId){
client.login(process.env.DISCORD_BOT_TOKEN).then(socks.forEach((sock)=>runSocket(sock)))
}
}
async function getUserId(){
let url = `${baseUrl}/api/auth/me`
await axios.get(url,
{headers:{
'X-Token': screepsToken,
'X-Username': user
}}
).then((res)=> {
if (res.data && res.data._id){
userId = res.data._id
}
}
)
}
async function runSocket(sock){
let inactivityTimeout;
const INACTIVITY_LIMIT = 60000; // Set the inactivity timeout limit (e.g., 60 seconds)
function resetInactivityTimeout() {
clearTimeout(inactivityTimeout);
inactivityTimeout = setTimeout(() => {
sendMessage(`inactivity detected, attempting restart ${this.url}`,'socket')
console.log('Inactivity detected. Closing and restarting socket.');
sock.close(); // Close the socket
}, INACTIVITY_LIMIT);
}
sock.onopen = function() {
//connecting to screeps websocket
console.log('open');
console.log('authorising, token:',screepsToken != undefined)
sock.send('auth '+screepsToken)
sendMessage(`connected ${this.url}`,'socket')
resetInactivityTimeout()
};
sock.onmessage = function(e) {
//listening to screeps console
if (e.data){
let split = e.data.split(' ')
if (split[0] == 'auth' && split[1] == 'ok'){
sock.send(`subscribe user:${userId}/console`)
}
if (e.data[0]=='['){
let obj = JSON.parse(e.data)
if (obj && obj[1] && obj[1].messages && obj[1].messages.log){
let logs = obj[1].messages.log
for (let i in logs){
let log = logs[i]
if (log){
let split = log.split('|')
for (let t in typesOfAnnouncement){
if (split && split[0] == typesOfAnnouncement[t]){
writeAnnouncement(split,typesOfAnnouncement[t])
}
}
}
}
}
resetInactivityTimeout()
}
}
};
sock.onError = function(err) {
sendMessage(`error ${this.url} ${err}`,'socket')
};
sock.onClose = function() {
//connecting to screeps websocket
clearTimeout(inactivityTimeout);
sendMessage(`disconnected ${this.url}`,'socket')
setTimeout(() => runSocket(sock), 2000);
};
}
function writeAnnouncement(data,type){
if (data && Array.isArray(data) && data.length === 3){
//each console log should be in the format announcementType/Game.shard.name/message
let dataSection = data.slice(2)
let dataStr = dataSection.join()
let message = `\` ${type} from ${data[1]}: ${dataStr} \``
console.log('log',message)
sendMessage(message,type)
} else {
console.log('bad data',JSON.stringify(data))
}
}
async function sendMessage(toSendText,type){
//send the message to discord on a specific channel for that announcement type
let guilds = await client.guilds.fetch();
let channelName = type+'s'
let split = channelName.split(' ')
if (split.length>0){
//channel names can't have spaces -> convert to hyphens
channelName = split.join('-')
}
guilds.each(guild => processGuild(guild.id))
async function processGuild(guildid){
let guild = await client.guilds.fetch(guildid)
if (!guild){
console.log('no guild',guildid)
return
}
if (!guild.available){
console.log(guild.name,'unavailable')
return
}
//ensure this type of channel exists
let channels = await guild.channels.fetch();
let channel = channels.find((c)=>c.name === channelName)
if (!channel){
channel = await guild.channels.create(channelName)
console.log('creating channel',channelName)
}
if (!channel){
return
}
//check we dont have any similar messages in message interval
let messageManager = channel.messages
let messages = await messageManager.fetch(undefined,{limit:50})
let canPost = true
messages.each((message)=>{
if (message.content && message.content === toSendText){
let time = Date.now()
let difference = time - message.createdTimestamp
let hours = difference/3600000
//to prevent spam - only sends a duplicate message after x hours
if (hours < 3){
canPost = false
}
}
})
if (canPost){
console.log(`posting ${toSendText}`)
channel.send(toSendText)
} else {
console.log(`duplicate ${toSendText}`)
}
}
}