-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathserver.py
398 lines (342 loc) · 16.2 KB
/
server.py
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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
#Features:
#*Connect as a user to a chat and msg other people
#Feature to do
#Colors for each user
#Customizations
#Logg all chat messages
#Logg connection info
try:
import sentry_sdk
sentry_sdk.init(
dsn="https://[email protected]/4506667638784000",
# Set traces_sample_rate to 1.0 to capture 100%
# of transactions for performance monitoring.
traces_sample_rate=1.0,
# Set profiles_sample_rate to 1.0 to profile 100%
# of sampled transactions.
# We recommend adjusting this value in production.
profiles_sample_rate=1.0,
)
#importing socket and threading to work
import socket, threading,time,random, uuid, hashlib, logging, sys, os
from os import system
from getpass import getpass
#from socket import *
from threading import Thread
#Port and IP config options. Either static or ask on launch
#port = 12000
#ip = '127.0.0.1'
ip = input("Enter an IP address of Server: ")
port = int(input("Input the Port Number (5555): "))
logging.basicConfig(filename='Server.log',level=logging.DEBUG, format='%(asctime)s:%(levelname)s:%(message)s', datefmt='%m/%d/%Y %I:%M:%S %p')
logging.info(f'\n\n ----- NEW INSTANCE -----\n')
#local Vars
format = 'utf8'
clients = []
handles = []
adminAccounts = []
stopThread = False
#Custom Messages Varaibles
disconnectMsg = "/close"
kickMsg = "/kick"
banMsg = "/ban"
promoteMsg = "/promote"
infoMs = "/info"
msgMsg= "/msg"
onlineMsg = "/online"
helpMsg = "/help"
regularCommands = [f"/exit: To Exit the Program\n /help: For list of all commands\n /online: List of Who is Online"]
adminCommands = regularCommands + ["\n/kick: ''To kick a user temporarily \n /ban: To fully ban a username\n"]
#starting socket
try:
print(f'Trying: {ip} and {port}')
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind((ip, port))
server.listen()
except Exception as error:
exc_type, exc_obj, exc_tb = sys.exc_info()
logging.critical(f'Line: {exc_tb.tb_lineno} : Type: {exc_type} : Starting Socket: {error}')
#INcreasing socket port number by random 1 to 999 if first bind fails.
a = random.randint(1,999)
port += a
print(f'Trying: {ip} and {port}')
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind((ip, port))
server.listen()
system("title "+(f'Connected to: {ip} with Port: {port}.'))
#Creating adminaccounts from user input
while True:
with open('.adminAccounts.encrypted','r') as file:
adminAccounts = dict(map(str.split, file))
print(f'These are the existing admin accounts: {adminAccounts}\n')
cont = input("Would you like to add more admin Accounts? (y/n): ")
if cont == 'n':
break
adminUsers = input("\nEnter a username for a admin account (Type in 'done' when finished): ")
if adminUsers == 'done':
break
#Making sure passwords match
while True:
adminPass = getpass("Enter Password (Input will be blank): ")
adminPass2 = getpass("Re-Enter Password (Input will be blank): ")
if adminPass2 != adminPass:
print("Passwords dont match - lets try again")
continue
else:
print(f'Account: {adminUsers} with password: {adminPass} was added to server admins! \n')
logging.info(f'Account: {adminUsers} was added to server admins! ')
break
#hashing password
salt = uuid.uuid4().hex
hashedPass = hashlib.sha256(salt.encode() + adminPass.encode()).hexdigest() + ':' + salt
#writing account info to file and adding it to varible that will be used to check later.
with open('.adminAccounts.encrypted','a+') as file:
account = (adminUsers + ' ' + hashedPass)
file.write(f'{account}\n')
print(f"{account}-wrote user to adminAccounts file")
print("-----Done with Admin account creation!-----\n")
#Getting accounts from file
with open('.adminAccounts.encrypted','r') as file:
adminAccounts = dict(map(str.split, file))
#Functions with features.
def kickUser(name,fromHandle):
if name in handles:
print("Starting Kick Process")
nameIndex = handles.index(name)
clientToKick = clients[nameIndex]
clients.remove(clientToKick)
clientToKick.send("YOu were kicked by an admin! \n Press (enter) to exit".encode(format))
clientToKick.close()
handles.remove(name)
broadcast(f'{name} was kicked by an admin'.encode(format))
else:
print("Name not in hanldes")
logging.info(f'User: {fromHandle} tried to kick: {name} but they were not online!')
fromHandle.send("User not Online".encode(format))
# def infoUser(origClient, name):
# if name in handles:
# nameIndex = handles.index(name)
# clientInfo = clients[nameIndex]
# info = clients(clientInfo)
# origClient.send(f"Info for user {name} is {info}".encode(format))
# broadcast(f'{origClient} got info about {name}'.encode(format))
#Funciton t send messages to everyone
def broadcast(message):
print(message)
logging.info(message)
#sending a msg to all the clinets in teh client list
for client in clients:
client.send(message)
# #Funciton t send messages to single user
# def direct(client, message):
# #sending a msg to all the clinets in teh client list
# client.send(message)
# #Printing messages to server console screen.
# print({str(message.decode(format))})
#Fucntion to receive client messages
def handleMsg(client):
while True:
try:
handle = handles[clients.index(client)]
msg = client.recv(1024)
command = str(msg.decode(format))
# Checking if user sent a command.
if command[len(handle)+2:len(handle)+3] == '/' and handle in adminAccounts:
print(f"{handle} -- is running {command}")
logging.info(f"{handle} -- is running {command}")
command = command[len(handle) + 2:]
if kickMsg in command:
print("Command is kick")
whoKick = command[6:]
# client.send("Attempting to Kick User")
print("Sending to kickUser def " )
kickUser(str(whoKick),client)
elif banMsg in command:
print("Command is ban")
whoBan = command[5:]
with open('Banned.txt','a+') as file:
file.write(f'{whoBan}\n')
print(f"{whoBan}-wrote user to ban file")
print("Sending to kickUser def " )
kickUser(str(whoBan),client)
elif helpMsg in command:
client.send(str(adminCommands).encode(format))
elif onlineMsg in command:
client.send(f'This is who is Online: \n{str(handles)}\n'.encode(format))
elif "/exit" in command:
if client in clients:
# Iff error happens we weill disconnect client and remove them from our list.
index = clients.index(client)
clients.remove(client)
client.close()
handle = handles[index]
handles.remove(handle)
print(f'{str(handle)} : left the chat! ')
broadcast(f'{str(handle)} : left the chat! '.encode(format))
continue
else:
client.send(f"\nNot a valid command, try these:\n {str(adminCommands)}\n".encode(format))
elif onlineMsg in command:
client.send(f'This is who is Online: \n{str(handles)}\n'.encode(format))
elif helpMsg in command:
client.send(str(regularCommands).encode(format))
elif "/exit" in command:
if client in clients:
# Iff error happens we weill disconnect client and remove them from our list.
index = clients.index(client)
clients.remove(client)
client.close()
handle = handles[index]
handles.remove(handle)
print(f'{str(handle)} : left the chat! ')
broadcast(f'{str(handle)} : left the chat! '.encode(format))
continue
elif command[len(handle)+2:len(handle)+3 ] == '/':
client.send("Not an Admin - Ask for a promotion".encode(format))
print(f"This was done----{command}----")
else:
# print("Sending to brodcast")
broadcast(msg)
except Exception as error:
if client in clients:
exc_type, exc_obj, exc_tb = sys.exc_info()
logging.critical(f'Line: {exc_tb.tb_lineno} : Type: {exc_type} : Handle Msg Fun: {error}')
#Iff error happens we weill disconnect client and remove them from our list.
index = clients.index(client)
clients.remove(client)
#client.close()
handle = handles[index]
handles.remove(handle)
#print(f'{str(handle)} : left the chat! ')
broadcast(f'{str(handle)} : left the chat! '.encode(format))
continue
#FUnction to resive clients
def take():
while True:
try:
#Recivng connection from clinet
client, address = server.accept()
logging.info(f'New Client Connected with: {client} and {address}')
except Exception as error:
if client in clients:
exc_type, exc_obj, exc_tb = sys.exc_info()
logging.critical(f'Line: {exc_tb.tb_lineno} : Type: {exc_type} : Accepting Client Connection (Take Function): {error}')
client.send(error)
client.close()
continue
syn = client.recv(1024).decode(format)
if syn != 'SYN':
client.close()
continue
print(f'!!!!!!!{str(client)} wth {str(address)}: has connected')
#SYN TO Client
# client.send('MOTD: Welcome to Hacked.FYI Chat Room'.encode(format))
#ACK and Handle To client
client.send('HANDLE'.encode(format))
#Handle From client
handle = client.recv(1024).decode(format)
#Chekcing if handle is less than 15 chars
if len(handle) >= 20:
client.send('Please have a handle less than 20 char'.encode(format))
print("Handle bigger than 20 chars")
#print(error)
client.close()
continue
else:
#client.send("\nHandle Accepted \n ".encode(format))
print("Handle Acepted")
logging.info(f'Clients Handle: {handle}')
#print(f'-{handle}-')
#
with open('Banned.txt', 'r') as file:
banned = file.readlines()
# Seeing if user bannded
print(banned)
if handle+'\n' in banned:
print(f'{handle} is in Banned list - ')
client.send('BAN'.encode(format))
time.sleep(5)
client.close()
continue
else:
print(print(f'-{handle}- not banned'))
try:
with open('.adminAccounts.encrypted','r') as file:
adminAccounts = dict(map(str.split, file))
# print(adminAccounts)
#adminAccounts = {'admin':'adminpass','jane':'janepass'}
# print(adminAccounts)
print("Trying if handle in adminAccounts {}".format(handle))
#Seeing if user has admin rights
if handle in adminAccounts:
print("Handle in admin accounts")
client.send("PASS".encode(format))
password = client.recv(1024).decode(format)
print(password)
# password = password[len(handle)+2:]
print(f'-{password}-')
#print(password[len(handle)+2:])
#Checking if password is right
hashPass, salt = adminAccounts[handle].split(':')
enteredHashedPass = hashlib.sha256(salt.encode()+password.encode()).hexdigest()
if enteredHashedPass != hashPass:
print("WRong password")
client.send("REFUSE".encode(format))
client.close()
continue
else:
client.send('You are an Admin! Welcome'.encode(format))
print(f'Admin: {handle} has connected to server')
else:
print("User not in adminAccounts")
except Exception as error:
exc_type, exc_obj, exc_tb = sys.exc_info()
logging.critical(f'Line: {exc_tb.tb_lineno} : Type: {exc_type} : Auth-Admin: {error}')
print(error)
pass
#Adding new user info to our list
print("Adding to clients list")
handles.append(handle)
clients.append(client)
#printing to console the handle of user and telling eveyrone who joined. and telling client they are on the serve.r
client.send(f'\nConnected to server with Handle: {str(handle)}'.encode(format))
broadcast("\n--- {} joined!---\n".format(handle).encode(format))
client.send(f'This is who is Online: {str(handles)}'.encode(format))
#Starting the thread to handle Clients
threadHandle = threading.Thread(target=handleMsg, args=(client,))
threadHandle.start()
#Function for teh server to write commands and reboot
def write():
while True:
global stopThread
if stopThread:
for x in clients:
x.close()
break
#input from server
serverName = '-----Server: '
message =input('')
#checking if special command.
if message.startswith('/'):
closeIn = int(input("When to shut down server: "))
howLongDown = int(input("How long will it be down: "))
if message.startswith('/close'):
reason = ('{} \n Will shut down in: {} Seconds! \n and will come back in {} Min!\n'.format(serverName, message[7:],closeIn, howLongDown))
# reason = serverName + message[7:] + ' Will shut down in: ' + {closeIn} + 'Min! \n '
broadcast(reason.encode(format))
time.sleep(closeIn)
stopThread = True
else:
broadcast(f'\n {serverName} {message} \n'.encode(format))
# print(f"Starting Server on {ip,port}")
logging.info(f'Server Started on: {ip, port}')
print(f'Server Started on: {ip, port}')
#take()
threadTake = threading.Thread(target=take)
threadTake.start()
threadWrite = threading.Thread(target=write)
threadWrite.start()
except Exception as error:
exc_type, exc_obj, exc_tb = sys.exc_info()
logging.critical(f'Line: {exc_tb.tb_lineno} : Type: {exc_type} : Main TRY Error: {error}')
input('Press Enter to exit')