mirror of
https://github.com/iio612/DEFENDER.git
synced 2026-02-13 19:24:23 +00:00
Compare commits
42 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b27b503d78 | ||
|
|
f7c80d190e | ||
|
|
36e3835e6c | ||
|
|
9bfe5925f8 | ||
|
|
2c78025bfb | ||
|
|
c3187e81dd | ||
|
|
979ba40c05 | ||
|
|
f0c0a2d06a | ||
|
|
2be6ece27f | ||
|
|
6801c981ab | ||
|
|
fd88df1017 | ||
|
|
ad5b7ffbf2 | ||
|
|
2b7f1d8bf3 | ||
|
|
cea69c1580 | ||
|
|
f5ff9259e8 | ||
|
|
c404cc3234 | ||
|
|
12c7e5e832 | ||
|
|
3cdee5fddf | ||
|
|
80b329dd5d | ||
|
|
f2b5c48fc9 | ||
|
|
f7b49c151f | ||
|
|
e2a1ec5866 | ||
|
|
1ee9b7e3ff | ||
|
|
cc53eae875 | ||
|
|
8a80840a6a | ||
|
|
4d0087623c | ||
|
|
e25baea0ef | ||
|
|
dc20f5ec3c | ||
|
|
ee02566343 | ||
|
|
11eedbb191 | ||
|
|
110cae3b84 | ||
|
|
7e5e2d4643 | ||
|
|
857cbfc85d | ||
|
|
7422bcad45 | ||
|
|
3518589e9c | ||
|
|
0cf1262d31 | ||
|
|
e14c97de03 | ||
|
|
ff603ab2a4 | ||
|
|
69360be3ad | ||
|
|
d7503768b6 | ||
|
|
bfa90c6bd5 | ||
|
|
b5503d23d7 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -2,7 +2,6 @@
|
|||||||
db/
|
db/
|
||||||
logs/
|
logs/
|
||||||
__pycache__/
|
__pycache__/
|
||||||
mods/mod_jsonrpc.py
|
|
||||||
configuration.json
|
configuration.json
|
||||||
*.log
|
*.log
|
||||||
test.py
|
test.py
|
||||||
76
README.md
76
README.md
@@ -32,30 +32,34 @@ Il permet aux opérateurs de gérer efficacement un canal, tout en offrant aux u
|
|||||||
- Système d'exploitation Linux (Windows non supporté)
|
- Système d'exploitation Linux (Windows non supporté)
|
||||||
- Un server UnrealIRCD corréctement configuré
|
- Un server UnrealIRCD corréctement configuré
|
||||||
- Python version 3.10 ou supérieure
|
- Python version 3.10 ou supérieure
|
||||||
|
```bash
|
||||||
Bash:
|
# Bash
|
||||||
$ git clone https://github.com/adator85/IRC_DEFENDER_MODULES.git
|
$ git clone https://github.com/adator85/IRC_DEFENDER_MODULES.git
|
||||||
- Renommer le fichier exemple_configuration.json en configuration.json
|
# Renommer le fichier exemple_configuration.json en configuration.json
|
||||||
- Configurer le fichier configuration.json
|
# Configurer le fichier configuration.json
|
||||||
$ python3 main.py
|
$ python3 main.py
|
||||||
|
```
|
||||||
Si votre configuration est bonne, votre service est censé etre connecté a votre réseau IRC
|
Si votre configuration est bonne, votre service est censé etre connecté a votre réseau IRC
|
||||||
Pour Les prochains lancement de defender vous devez utiliser la commande suivante:
|
Pour Les prochains lancement de defender vous devez utiliser la commande suivante:
|
||||||
|
|
||||||
Bash:
|
```bash
|
||||||
$ systemctl --user [start | stop | restart | status] defender
|
# Bash
|
||||||
|
$ systemctl --user [start | stop | restart | status] defender
|
||||||
|
```
|
||||||
# Installation manuelle:
|
# Installation manuelle:
|
||||||
Bash:
|
```bash
|
||||||
$ git clone https://github.com/adator85/IRC_DEFENDER_MODULES.git
|
# Bash
|
||||||
$ cd IRC_DEFENDER_MODULES
|
$ git clone https://github.com/adator85/IRC_DEFENDER_MODULES.git
|
||||||
$ python3 -m venv .pyenv
|
$ cd IRC_DEFENDER_MODULES
|
||||||
$ source .pyenv/bin/activate
|
$ python3 -m venv .pyenv
|
||||||
(pyenv)$ pip install sqlalchemy, psutil, requests, faker
|
$ source .pyenv/bin/activate
|
||||||
- Créer un service nommé "defender.service" pour votre service et placer le dans "/PATH/TO/USER/.config/systemd/user/"
|
(pyenv)$ pip install sqlalchemy, psutil, requests, faker, unrealircd_rpc_py
|
||||||
- Si le dossier n'existe pas il faut les créer
|
|
||||||
$ sudo systemctl --user start defender
|
|
||||||
|
|
||||||
|
# Créer un service nommé "defender.service"
|
||||||
|
# pour votre service et placer le dans "/PATH/TO/USER/.config/systemd/user/"
|
||||||
|
# Si le dossier n'existe pas il faut les créer
|
||||||
|
$ sudo systemctl --user start defender
|
||||||
|
```
|
||||||
# Configuration
|
# Configuration
|
||||||
```
|
```
|
||||||
SERVEUR (Serveur)
|
SERVEUR (Serveur)
|
||||||
@@ -102,8 +106,6 @@ Pour Les prochains lancement de defender vous devez utiliser la commande suivant
|
|||||||
DEBUG (Debug)
|
DEBUG (Debug)
|
||||||
DEBUG_LEVEL: Niveau de verbosité des messages de debug (plus grand est le nombre, plus il y a d'informations). (default : 20) Pour une production
|
DEBUG_LEVEL: Niveau de verbosité des messages de debug (plus grand est le nombre, plus il y a d'informations). (default : 20) Pour une production
|
||||||
|
|
||||||
COULEURS (Couleurs)
|
|
||||||
CONFIG_COLOR: Dictionnaire contenant des codes de couleurs IRC pour un meilleur affichage des messages.
|
|
||||||
```
|
```
|
||||||
Modification de la configuration
|
Modification de la configuration
|
||||||
|
|
||||||
@@ -121,13 +123,13 @@ Pour Les prochains lancement de defender vous devez utiliser la commande suivant
|
|||||||
"SERVEUR_SSL": true,
|
"SERVEUR_SSL": true,
|
||||||
|
|
||||||
"SERVICE_NAME": "defender",
|
"SERVICE_NAME": "defender",
|
||||||
"SERVICE_NICKNAME": "Dev-PyDefender",
|
"SERVICE_NICKNAME": "PyDefender",
|
||||||
"SERVICE_REALNAME": "Python Defender Security",
|
"SERVICE_REALNAME": "Python Defender Security",
|
||||||
"SERVICE_USERNAME": "Dev-PyDefender",
|
"SERVICE_USERNAME": "PyDefender",
|
||||||
"SERVICE_HOST": "HOST.DE.TON.DEFENDER",
|
"SERVICE_HOST": "HOST.DE.TON.DEFENDER",
|
||||||
|
|
||||||
"OWNER": "TON_NICK_NAME",
|
"OWNER": "TON_NICK_NAME",
|
||||||
"PASSWORD": "admin"
|
"PASSWORD": "TON_PASSWORD"
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -145,9 +147,9 @@ Pour Les prochains lancement de defender vous devez utiliser la commande suivant
|
|||||||
"SERVEUR_SSL": true,
|
"SERVEUR_SSL": true,
|
||||||
|
|
||||||
"SERVICE_NAME": "defender",
|
"SERVICE_NAME": "defender",
|
||||||
"SERVICE_NICKNAME": "Dev-PyDefender",
|
"SERVICE_NICKNAME": "PyDefender",
|
||||||
"SERVICE_REALNAME": "Python Defender Security",
|
"SERVICE_REALNAME": "Python Defender Security",
|
||||||
"SERVICE_USERNAME": "Dev-PyDefender",
|
"SERVICE_USERNAME": "PyDefender",
|
||||||
"SERVICE_HOST": "HOST.DE.TON.DEFENDER",
|
"SERVICE_HOST": "HOST.DE.TON.DEFENDER",
|
||||||
"SERVICE_INFO": "Network IRC Service",
|
"SERVICE_INFO": "Network IRC Service",
|
||||||
"SERVICE_CHANLOG": "#services",
|
"SERVICE_CHANLOG": "#services",
|
||||||
@@ -157,13 +159,22 @@ Pour Les prochains lancement de defender vous devez utiliser la commande suivant
|
|||||||
"SERVICE_PREFIX": "!",
|
"SERVICE_PREFIX": "!",
|
||||||
|
|
||||||
"OWNER": "TON_NICK_NAME",
|
"OWNER": "TON_NICK_NAME",
|
||||||
"PASSWORD": "admin",
|
"PASSWORD": "TON_PASSWORD",
|
||||||
|
|
||||||
|
"JSONRPC_URL": "https://your.domaine.com:8600/api",
|
||||||
|
"JSONRPC_PATH_TO_SOCKET_FILE": "/PATH/TO/YOUR/IRCD/data/rpc.socket",
|
||||||
|
"JSONRPC_METHOD": "socket",
|
||||||
|
"JSONRPC_USER": "YOUR_RPC_USER",
|
||||||
|
"JSONRPC_PASSWORD": "YOUR_RPC_PASSWORD",
|
||||||
|
|
||||||
"SALON_JAIL": "#jail",
|
"SALON_JAIL": "#jail",
|
||||||
"SALON_JAIL_MODES": "sS",
|
"SALON_JAIL_MODES": "sS",
|
||||||
"SALON_LIBERER": "#welcome",
|
"SALON_LIBERER": "#welcome",
|
||||||
|
|
||||||
"SALON_CLONES": "#clones",
|
"CLONE_CHANNEL": "#clones",
|
||||||
|
"CLONE_CMODES": "+nts",
|
||||||
|
"CLONE_LOG_HOST_EXEMPT": ["HOST.TO.SKIP"],
|
||||||
|
"CLONE_CHANNEL_PASSWORD": "YOUR_CHANNEL_PASSWORD",
|
||||||
|
|
||||||
"API_TIMEOUT": 2,
|
"API_TIMEOUT": 2,
|
||||||
|
|
||||||
@@ -171,18 +182,7 @@ Pour Les prochains lancement de defender vous devez utiliser la commande suivant
|
|||||||
"WHITELISTED_IP": ["127.0.0.1"],
|
"WHITELISTED_IP": ["127.0.0.1"],
|
||||||
"GLINE_DURATION": "30",
|
"GLINE_DURATION": "30",
|
||||||
|
|
||||||
"DEBUG_LEVEL": 10,
|
"DEBUG_LEVEL": 20
|
||||||
|
|
||||||
"CONFIG_COLOR": {
|
|
||||||
"blanche": "\\u0003\\u0030",
|
|
||||||
"noire": "\\u0003\\u0031",
|
|
||||||
"bleue": "\\u0003\\u0020",
|
|
||||||
"verte": "\\u0003\\u0033",
|
|
||||||
"rouge": "\\u0003\\u0034",
|
|
||||||
"jaune": "\\u0003\\u0036",
|
|
||||||
"gras": "\\u0002",
|
|
||||||
"nogc": "\\u0002\\u0003"
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|||||||
45
core/base.py
45
core/base.py
@@ -1,8 +1,21 @@
|
|||||||
import time, threading, os, random, socket, hashlib, ipaddress, logging, requests, json, re, ast
|
import os
|
||||||
|
import re
|
||||||
|
import json
|
||||||
|
import time
|
||||||
|
import random
|
||||||
|
import socket
|
||||||
|
import hashlib
|
||||||
|
import logging
|
||||||
|
import threading
|
||||||
|
import ipaddress
|
||||||
|
|
||||||
|
import ast
|
||||||
|
import requests
|
||||||
|
|
||||||
from dataclasses import fields
|
from dataclasses import fields
|
||||||
from typing import Union, Literal
|
from typing import Union, Literal
|
||||||
from base64 import b64decode, b64encode
|
from base64 import b64decode, b64encode
|
||||||
from datetime import datetime
|
from datetime import datetime, timedelta, timezone
|
||||||
from sqlalchemy import create_engine, Engine, Connection, CursorResult
|
from sqlalchemy import create_engine, Engine, Connection, CursorResult
|
||||||
from sqlalchemy.sql import text
|
from sqlalchemy.sql import text
|
||||||
from core.loadConf import ConfigDataModel
|
from core.loadConf import ConfigDataModel
|
||||||
@@ -35,7 +48,6 @@ class Base:
|
|||||||
with open(version_filename, 'r') as version_data:
|
with open(version_filename, 'r') as version_data:
|
||||||
current_version:dict[str, str] = json.load(version_data)
|
current_version:dict[str, str] = json.load(version_data)
|
||||||
|
|
||||||
# self.DEFENDER_VERSION = current_version["version"]
|
|
||||||
self.Config.current_version = current_version['version']
|
self.Config.current_version = current_version['version']
|
||||||
|
|
||||||
return None
|
return None
|
||||||
@@ -68,6 +80,14 @@ class Base:
|
|||||||
self.logs.warning(f'Github not available to fetch latest version')
|
self.logs.warning(f'Github not available to fetch latest version')
|
||||||
|
|
||||||
def check_for_new_version(self, online:bool) -> bool:
|
def check_for_new_version(self, online:bool) -> bool:
|
||||||
|
"""Check if there is a new version available
|
||||||
|
|
||||||
|
Args:
|
||||||
|
online (bool): True if you want to get the version from github (main branch)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: True if there is a new version available
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
self.logs.debug(f'Checking for a new service version')
|
self.logs.debug(f'Checking for a new service version')
|
||||||
|
|
||||||
@@ -106,7 +126,11 @@ class Base:
|
|||||||
Cette fonction retourne un UNIXTIME de type 12365456
|
Cette fonction retourne un UNIXTIME de type 12365456
|
||||||
Return: Current time in seconds since the Epoch (int)
|
Return: Current time in seconds since the Epoch (int)
|
||||||
"""
|
"""
|
||||||
|
cet_offset = timezone(timedelta(hours=2))
|
||||||
|
now_cet = datetime.now(cet_offset)
|
||||||
|
unixtime_cet = int(now_cet.timestamp())
|
||||||
unixtime = int( time.time() )
|
unixtime = int( time.time() )
|
||||||
|
|
||||||
return unixtime
|
return unixtime
|
||||||
|
|
||||||
def get_datetime(self) -> str:
|
def get_datetime(self) -> str:
|
||||||
@@ -212,6 +236,19 @@ class Base:
|
|||||||
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
def db_update_module(self, user_cmd: str, module_name: str) -> None:
|
||||||
|
"""Modifie la date et le user qui a rechargé le module
|
||||||
|
|
||||||
|
Args:
|
||||||
|
user_cmd (str): le user qui a rechargé le module
|
||||||
|
module_name (str): le module a rechargé
|
||||||
|
"""
|
||||||
|
update_cmd_query = f"UPDATE {self.Config.table_module} SET datetime = :datetime, user = :user WHERE module_name = :module_name"
|
||||||
|
mes_donnees = {'datetime': self.get_datetime(), 'user': user_cmd, 'module_name': module_name}
|
||||||
|
self.db_execute_query(update_cmd_query, mes_donnees)
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
def db_delete_module(self, module_name:str) -> None:
|
def db_delete_module(self, module_name:str) -> None:
|
||||||
"""Supprime les modules de la base de données
|
"""Supprime les modules de la base de données
|
||||||
|
|
||||||
@@ -574,7 +611,7 @@ class Base:
|
|||||||
)
|
)
|
||||||
'''
|
'''
|
||||||
|
|
||||||
table_core_channel = '''CREATE TABLE IF NOT EXISTS core_channel (
|
table_core_channel = f'''CREATE TABLE IF NOT EXISTS {self.Config.table_channel} (
|
||||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
datetime TEXT,
|
datetime TEXT,
|
||||||
module_name TEXT,
|
module_name TEXT,
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
import socket, ssl
|
import socket
|
||||||
|
import ssl
|
||||||
|
import traceback
|
||||||
from ssl import SSLSocket
|
from ssl import SSLSocket
|
||||||
|
from typing import Union
|
||||||
from core.loadConf import Config
|
from core.loadConf import Config
|
||||||
from core.Model import Clones
|
from core.Model import Clones
|
||||||
from core.base import Base
|
from core.base import Base
|
||||||
from typing import Union
|
|
||||||
|
|
||||||
class Connection:
|
class Connection:
|
||||||
|
|
||||||
@@ -15,7 +17,8 @@ class Connection:
|
|||||||
self.nickname = nickname
|
self.nickname = nickname
|
||||||
self.username = username
|
self.username = username
|
||||||
self.realname = realname
|
self.realname = realname
|
||||||
self.clone_chanlog = self.Config.SALON_CLONES
|
self.clone_chanlog = self.Config.CLONE_CHANNEL
|
||||||
|
self.clone_log_exempt = self.Config.CLONE_LOG_HOST_EXEMPT
|
||||||
self.channels:list[str] = channels
|
self.channels:list[str] = channels
|
||||||
self.CHARSET = ['utf-8', 'iso-8859-1']
|
self.CHARSET = ['utf-8', 'iso-8859-1']
|
||||||
self.Clones = CloneObject
|
self.Clones = CloneObject
|
||||||
@@ -157,6 +160,7 @@ class Connection:
|
|||||||
self.Base.logs.error(f"OS Error __connect_to_irc: {soe}")
|
self.Base.logs.error(f"OS Error __connect_to_irc: {soe}")
|
||||||
except AttributeError as atte:
|
except AttributeError as atte:
|
||||||
self.Base.logs.critical(f"{atte}")
|
self.Base.logs.critical(f"{atte}")
|
||||||
|
self.Base.logs.critical(f"{traceback.format_exc()}")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.Base.logs.error(f"Exception: {e}")
|
self.Base.logs.error(f"Exception: {e}")
|
||||||
|
|
||||||
@@ -190,9 +194,10 @@ class Connection:
|
|||||||
for channel in self.channels:
|
for channel in self.channels:
|
||||||
self.send2socket(f"JOIN {channel}")
|
self.send2socket(f"JOIN {channel}")
|
||||||
|
|
||||||
self.send2socket(f"JOIN {self.clone_chanlog}")
|
self.send2socket(f"JOIN {self.clone_chanlog} {self.Config.CLONE_CHANNEL_PASSWORD}")
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
case '422':
|
case '422':
|
||||||
# Missing MOTD
|
# Missing MOTD
|
||||||
self.currentCloneObject.connected = True
|
self.currentCloneObject.connected = True
|
||||||
@@ -200,7 +205,15 @@ class Connection:
|
|||||||
for channel in self.channels:
|
for channel in self.channels:
|
||||||
self.send2socket(f"JOIN {channel}")
|
self.send2socket(f"JOIN {channel}")
|
||||||
|
|
||||||
self.send2socket(f"JOIN {self.clone_chanlog}")
|
self.send2socket(f"JOIN {self.clone_chanlog} {self.Config.CLONE_CHANNEL_PASSWORD}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
case '433':
|
||||||
|
# Nickname already in use
|
||||||
|
self.currentCloneObject.connected = False
|
||||||
|
self.currentCloneObject.init = False
|
||||||
|
self.send2socket(f'QUIT :Thanks and goodbye')
|
||||||
|
self.Base.logs.warning(f"Nickname {self.currentCloneObject.nickname} already in use >> Clone should be disconnected")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
case 'PRIVMSG':
|
case 'PRIVMSG':
|
||||||
@@ -214,13 +227,20 @@ class Connection:
|
|||||||
for i in range(3, len(response)):
|
for i in range(3, len(response)):
|
||||||
message.append(response[i])
|
message.append(response[i])
|
||||||
final_message = ' '.join(message)
|
final_message = ' '.join(message)
|
||||||
self.send2socket(f"PRIVMSG {self.clone_chanlog} :{fullname} => {final_message[1:]}")
|
|
||||||
|
exampt = False
|
||||||
|
for log_exception in self.clone_log_exempt:
|
||||||
|
if log_exception in fullname:
|
||||||
|
exampt = True
|
||||||
|
|
||||||
|
if not exampt:
|
||||||
|
self.send2socket(f"PRIVMSG {self.clone_chanlog} :{fullname} => {final_message[1:]}")
|
||||||
|
|
||||||
if nickname == self.Config.SERVICE_NICKNAME:
|
if nickname == self.Config.SERVICE_NICKNAME:
|
||||||
command = str(response[3]).replace(':','')
|
command = str(response[3]).replace(':','')
|
||||||
|
|
||||||
if command == 'KILL':
|
if command == 'KILL':
|
||||||
self.send2socket(f'QUIT :Thanks and goodbye', disconnect=True)
|
self.send2socket(f'QUIT :Thanks and goodbye')
|
||||||
|
|
||||||
if command == 'JOIN':
|
if command == 'JOIN':
|
||||||
channel_to_join = str(response[4])
|
channel_to_join = str(response[4])
|
||||||
|
|||||||
@@ -1,48 +1,48 @@
|
|||||||
{
|
{
|
||||||
"SERVEUR_IP": "0.0.0.0",
|
"SERVEUR_IP": "YOUR.SERVER.IP",
|
||||||
"SERVEUR_HOSTNAME": "your.host.name",
|
"SERVEUR_HOSTNAME": "YOUR.SERVER.HOST",
|
||||||
"SERVEUR_LINK": "your.link.to.server",
|
"SERVEUR_LINK": "LINK.DE.TON.SERVER",
|
||||||
"SERVEUR_PORT": 7002,
|
"SERVEUR_PORT": 7002,
|
||||||
"SERVEUR_PASSWORD": "link_password",
|
"SERVEUR_PASSWORD": "YOUR_LINK_PASSWORD",
|
||||||
"SERVEUR_ID": "006",
|
"SERVEUR_ID": "006",
|
||||||
"SERVEUR_SSL": true,
|
"SERVEUR_SSL": true,
|
||||||
|
|
||||||
"SERVICE_NAME": "defender",
|
"SERVICE_NAME": "defender",
|
||||||
"SERVICE_NICKNAME": "BotNickname",
|
"SERVICE_NICKNAME": "PyDefender",
|
||||||
"SERVICE_REALNAME": "BotRealname",
|
"SERVICE_REALNAME": "Python Defender Security",
|
||||||
"SERVICE_USERNAME": "BotUsername",
|
"SERVICE_USERNAME": "PyDefender",
|
||||||
"SERVICE_HOST": "your.service.hostname",
|
"SERVICE_HOST": "HOST.DE.TON.DEFENDER",
|
||||||
"SERVICE_INFO": "Network IRC Service",
|
"SERVICE_INFO": "Network IRC Service",
|
||||||
"SERVICE_CHANLOG": "#services",
|
"SERVICE_CHANLOG": "#services",
|
||||||
"SERVICE_SMODES": "+ioqBS",
|
"SERVICE_SMODES": "+ioqBS",
|
||||||
"SERVICE_CMODES": "ntsO",
|
"SERVICE_CMODES": "ntsOP",
|
||||||
"SERVICE_UMODES": "o",
|
"SERVICE_UMODES": "o",
|
||||||
"SERVICE_PREFIX": "!",
|
"SERVICE_PREFIX": "!",
|
||||||
|
|
||||||
"OWNER": "admin",
|
"OWNER": "TON_NICK_NAME",
|
||||||
"PASSWORD": "password",
|
"PASSWORD": "TON_PASSWORD",
|
||||||
|
|
||||||
|
"JSONRPC_URL": "https://your.domaine.com:8600/api",
|
||||||
|
"JSONRPC_PATH_TO_SOCKET_FILE": "/PATH/TO/YOUR/IRCD/data/rpc.socket",
|
||||||
|
"JSONRPC_METHOD": "socket",
|
||||||
|
"JSONRPC_USER": "YOUR_RPC_USER",
|
||||||
|
"JSONRPC_PASSWORD": "YOUR_RPC_PASSWORD",
|
||||||
|
|
||||||
"SALON_JAIL": "#jail",
|
"SALON_JAIL": "#jail",
|
||||||
"SALON_JAIL_MODES": "sS",
|
"SALON_JAIL_MODES": "sS",
|
||||||
"SALON_LIBERER": "#welcome",
|
"SALON_LIBERER": "#welcome",
|
||||||
|
|
||||||
|
"CLONE_CHANNEL": "#clones",
|
||||||
|
"CLONE_CMODES": "+nts",
|
||||||
|
"CLONE_LOG_HOST_EXEMPT": ["HOST.TO.SKIP"],
|
||||||
|
"CLONE_CHANNEL_PASSWORD": "YOUR_CHANNEL_PASSWORD",
|
||||||
|
|
||||||
"API_TIMEOUT": 2,
|
"API_TIMEOUT": 2,
|
||||||
|
|
||||||
"PORTS_TO_SCAN": [3028, 8080, 1080, 1085, 4145, 9050],
|
"PORTS_TO_SCAN": [3028, 8080, 1080, 1085, 4145, 9050],
|
||||||
"WHITELISTED_IP": ["127.0.0.1"],
|
"WHITELISTED_IP": ["127.0.0.1"],
|
||||||
"GLINE_DURATION": "30",
|
"GLINE_DURATION": "30",
|
||||||
|
|
||||||
"DEBUG_LEVEL": 20,
|
"DEBUG_LEVEL": 20
|
||||||
|
|
||||||
"CONFIG_COLOR": {
|
|
||||||
"blanche": "\\u0003\\u0030",
|
|
||||||
"noire": "\\u0003\\u0031",
|
|
||||||
"bleue": "\\u0003\\u0020",
|
|
||||||
"verte": "\\u0003\\u0033",
|
|
||||||
"rouge": "\\u0003\\u0034",
|
|
||||||
"jaune": "\\u0003\\u0036",
|
|
||||||
"gras": "\\u0002",
|
|
||||||
"nogc": "\\u0002\\u0003"
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -1,8 +1,9 @@
|
|||||||
from dataclasses import dataclass
|
|
||||||
from subprocess import check_call, run, CalledProcessError, PIPE
|
|
||||||
from platform import python_version, python_version_tuple
|
|
||||||
from sys import exit
|
|
||||||
import os
|
import os
|
||||||
|
import json
|
||||||
|
from sys import exit, prefix
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from subprocess import check_call, run, CalledProcessError, PIPE, check_output
|
||||||
|
from platform import python_version, python_version_tuple
|
||||||
|
|
||||||
class Install:
|
class Install:
|
||||||
|
|
||||||
@@ -24,13 +25,23 @@ class Install:
|
|||||||
venv_pip_executable: str
|
venv_pip_executable: str
|
||||||
venv_python_executable: str
|
venv_python_executable: str
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Package:
|
||||||
|
name: str = None
|
||||||
|
version: str = None
|
||||||
|
|
||||||
|
DB_PACKAGES: list[Package] = []
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
|
|
||||||
self.set_configuration()
|
self.set_configuration()
|
||||||
|
|
||||||
if self.skip_install:
|
if self.skip_install:
|
||||||
|
self.check_packages_version()
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
self.check_packages_version()
|
||||||
|
|
||||||
# Sinon tester les dependances python et les installer avec pip
|
# Sinon tester les dependances python et les installer avec pip
|
||||||
if self.do_install():
|
if self.do_install():
|
||||||
|
|
||||||
@@ -64,7 +75,7 @@ class Install:
|
|||||||
defender_install_folder=defender_install_folder,
|
defender_install_folder=defender_install_folder,
|
||||||
venv_folder=venv_folder,
|
venv_folder=venv_folder,
|
||||||
venv_cmd_installation=['python3', '-m', 'venv', venv_folder],
|
venv_cmd_installation=['python3', '-m', 'venv', venv_folder],
|
||||||
venv_cmd_requirements=['sqlalchemy','psutil','requests','faker'],
|
venv_cmd_requirements=['sqlalchemy','psutil','requests','faker','unrealircd_rpc_py'],
|
||||||
venv_pip_executable=f'{os.path.join(defender_install_folder, venv_folder, "bin")}{os.sep}pip',
|
venv_pip_executable=f'{os.path.join(defender_install_folder, venv_folder, "bin")}{os.sep}pip',
|
||||||
venv_python_executable=f'{os.path.join(defender_install_folder, venv_folder, "bin")}{os.sep}python'
|
venv_python_executable=f'{os.path.join(defender_install_folder, venv_folder, "bin")}{os.sep}python'
|
||||||
)
|
)
|
||||||
@@ -75,11 +86,13 @@ class Install:
|
|||||||
|
|
||||||
if not os.path.exists(os.path.join(self.config.defender_install_folder, 'core', 'configuration.json')):
|
if not os.path.exists(os.path.join(self.config.defender_install_folder, 'core', 'configuration.json')):
|
||||||
# If configuration file do not exist
|
# If configuration file do not exist
|
||||||
exit("/!\\ Configuration file (configuration.json) doesn't exist /!\\")
|
exit("/!\\ Configuration file (core/configuration.json) doesn't exist! please create it /!\\")
|
||||||
|
|
||||||
# Exclude Windows OS from the installation
|
# Exclude Windows OS from the installation
|
||||||
if os.name == 'nt':
|
if os.name == 'nt':
|
||||||
#print('/!\\ Skip installation /!\\')
|
# If windows, modify pip and python virtual environment executable
|
||||||
|
self.config.venv_pip_executable = f'{os.path.join(defender_install_folder, venv_folder, "Scripts")}{os.sep}pip.exe'
|
||||||
|
self.config.venv_python_executable = f'{os.path.join(defender_install_folder, venv_folder, "Scripts")}{os.sep}python.exe'
|
||||||
self.skip_install = True
|
self.skip_install = True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@@ -91,7 +104,7 @@ class Install:
|
|||||||
def is_root(self) -> bool:
|
def is_root(self) -> bool:
|
||||||
|
|
||||||
if os.geteuid() != 0:
|
if os.geteuid() != 0:
|
||||||
print('User without privileges ==> PASS')
|
print('> User without privileges ==> OK')
|
||||||
return False
|
return False
|
||||||
elif os.geteuid() == 0:
|
elif os.geteuid() == 0:
|
||||||
print('/!\\ Do not use root to install Defender /!\\')
|
print('/!\\ Do not use root to install Defender /!\\')
|
||||||
@@ -123,6 +136,75 @@ class Install:
|
|||||||
print(f"Try to install dependencies ...")
|
print(f"Try to install dependencies ...")
|
||||||
exit(5)
|
exit(5)
|
||||||
|
|
||||||
|
def get_packages_version_from_json(self) -> None:
|
||||||
|
"""This will create Package model with package names and required version
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
|
||||||
|
version_filename = f'.{os.sep}version.json'
|
||||||
|
with open(version_filename, 'r') as version_data:
|
||||||
|
package_info:dict[str, str] = json.load(version_data)
|
||||||
|
|
||||||
|
for name, version in package_info.items():
|
||||||
|
if name == 'version':
|
||||||
|
continue
|
||||||
|
|
||||||
|
self.DB_PACKAGES.append(
|
||||||
|
self.Package(name=name, version=version)
|
||||||
|
)
|
||||||
|
|
||||||
|
return None
|
||||||
|
except FileNotFoundError as fe:
|
||||||
|
print(f"File not found: {fe}")
|
||||||
|
except Exception as err:
|
||||||
|
print(f"General Error: {err}")
|
||||||
|
|
||||||
|
def check_packages_version(self) -> bool:
|
||||||
|
|
||||||
|
try:
|
||||||
|
newVersion = False
|
||||||
|
self.get_packages_version_from_json()
|
||||||
|
|
||||||
|
if not self.config.venv_folder in prefix:
|
||||||
|
print(f"You are probably running a new installation or you are not using your virtual env {self.config.venv_folder}")
|
||||||
|
return newVersion
|
||||||
|
|
||||||
|
print(f"> Checking for dependencies versions ==> WAIT")
|
||||||
|
for package in self.DB_PACKAGES:
|
||||||
|
newVersion = False
|
||||||
|
required_version = package.version
|
||||||
|
installed_version = None
|
||||||
|
|
||||||
|
output = check_output([self.config.venv_pip_executable, 'show', package.name])
|
||||||
|
for line in output.decode().splitlines():
|
||||||
|
if line.startswith('Version:'):
|
||||||
|
installed_version = line.split(':')[1].strip()
|
||||||
|
break
|
||||||
|
|
||||||
|
required_major, required_minor, required_patch = required_version.split('.')
|
||||||
|
installed_major, installed_minor, installed_patch = installed_version.split('.')
|
||||||
|
|
||||||
|
if required_major > installed_major:
|
||||||
|
print(f'> New version of {package.name} is available {installed_version} ==> {required_version}')
|
||||||
|
newVersion = True
|
||||||
|
elif required_major == installed_major and required_minor > installed_minor:
|
||||||
|
print(f'> New version of {package.name} is available {installed_version} ==> {required_version}')
|
||||||
|
newVersion = True
|
||||||
|
elif required_major == installed_major and required_minor == installed_minor and required_patch > installed_patch:
|
||||||
|
print(f'> New version of {package.name} is available {installed_version} ==> {required_version}')
|
||||||
|
newVersion = True
|
||||||
|
|
||||||
|
if newVersion:
|
||||||
|
self.run_subprocess([self.config.venv_pip_executable, 'install', '--upgrade', package.name])
|
||||||
|
|
||||||
|
print(f"> Dependencies versions ==> OK")
|
||||||
|
return newVersion
|
||||||
|
|
||||||
|
except CalledProcessError:
|
||||||
|
print(f"/!\\ Package {package.name} not installed /!\\")
|
||||||
|
except Exception as err:
|
||||||
|
print(f"General Error: {err}")
|
||||||
|
|
||||||
def check_python_version(self) -> bool:
|
def check_python_version(self) -> bool:
|
||||||
"""Test si la version de python est autorisée ou non
|
"""Test si la version de python est autorisée ou non
|
||||||
|
|
||||||
|
|||||||
334
core/irc.py
334
core/irc.py
@@ -1,10 +1,17 @@
|
|||||||
import ssl, re, importlib, sys, time, threading, socket, traceback
|
import sys
|
||||||
|
import socket
|
||||||
|
import threading
|
||||||
|
import ssl
|
||||||
|
import re
|
||||||
|
import importlib
|
||||||
|
import time
|
||||||
|
import traceback
|
||||||
from ssl import SSLSocket
|
from ssl import SSLSocket
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from typing import Union, Literal
|
from typing import Union
|
||||||
from core.loadConf import Config
|
from core.loadConf import Config
|
||||||
from core.Model import User, Admin, Channel, Clones
|
|
||||||
from core.base import Base
|
from core.base import Base
|
||||||
|
from core.Model import User, Admin, Channel, Clones
|
||||||
|
|
||||||
class Irc:
|
class Irc:
|
||||||
|
|
||||||
@@ -88,13 +95,15 @@ class Irc:
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
except ssl.SSLEOFError as soe:
|
except ssl.SSLEOFError as soe:
|
||||||
self.Base.logs.critical(f"SSLEOFError __create_socket: {soe} - {soc.fileno()}")
|
self.Base.logs.critical(f"SSLEOFError: {soe} - {soc.fileno()}")
|
||||||
except ssl.SSLError as se:
|
except ssl.SSLError as se:
|
||||||
self.Base.logs.critical(f"SSLError __create_socket: {se} - {soc.fileno()}")
|
self.Base.logs.critical(f"SSLError: {se} - {soc.fileno()}")
|
||||||
except OSError as oe:
|
except OSError as oe:
|
||||||
self.Base.logs.critical(f"OSError __create_socket: {oe} - {soc.fileno()}")
|
self.Base.logs.critical(f"OSError: {oe} - {soc.fileno()}")
|
||||||
|
if 'connection refused' in str(oe).lower():
|
||||||
|
sys.exit(oe)
|
||||||
except AttributeError as ae:
|
except AttributeError as ae:
|
||||||
self.Base.logs.critical(f"AttributeError __create_socket: {ae} - {soc.fileno()}")
|
self.Base.logs.critical(f"AttributeError: {ae} - {soc.fileno()}")
|
||||||
|
|
||||||
def __ssl_context(self) -> ssl.SSLContext:
|
def __ssl_context(self) -> ssl.SSLContext:
|
||||||
ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
|
ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
|
||||||
@@ -175,8 +184,8 @@ class Irc:
|
|||||||
except AttributeError as atte:
|
except AttributeError as atte:
|
||||||
self.Base.logs.critical(f"AttributeError: {atte}")
|
self.Base.logs.critical(f"AttributeError: {atte}")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.Base.logs.critical(f"Exception: {e}")
|
self.Base.logs.critical(f"General Error: {e}")
|
||||||
self.Base.logs.critical(traceback.print_exc())
|
self.Base.logs.critical(traceback.format_exc())
|
||||||
|
|
||||||
def __link(self, writer:Union[socket.socket, SSLSocket]) -> None:
|
def __link(self, writer:Union[socket.socket, SSLSocket]) -> None:
|
||||||
"""Créer le link et envoyer les informations nécessaires pour la
|
"""Créer le link et envoyer les informations nécessaires pour la
|
||||||
@@ -222,7 +231,7 @@ class Irc:
|
|||||||
writer.send(f":{service_id} MODE {chan} +{cmodes}\r\n".encode(charset))
|
writer.send(f":{service_id} MODE {chan} +{cmodes}\r\n".encode(charset))
|
||||||
writer.send(f":{service_id} MODE {chan} +{umodes} {service_id}\r\n".encode(charset))
|
writer.send(f":{service_id} MODE {chan} +{umodes} {service_id}\r\n".encode(charset))
|
||||||
|
|
||||||
self.Base.logs.debug('Link information sent to the server')
|
self.Base.logs.debug('>> Link information sent to the server')
|
||||||
|
|
||||||
return None
|
return None
|
||||||
except AttributeError as ae:
|
except AttributeError as ae:
|
||||||
@@ -249,7 +258,6 @@ class Irc:
|
|||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
with self.Base.lock:
|
with self.Base.lock:
|
||||||
# print(f">{str(send_message)}")
|
|
||||||
self.IrcSocket.send(f"{send_message}\r\n".encode(self.CHARSET[0]))
|
self.IrcSocket.send(f"{send_message}\r\n".encode(self.CHARSET[0]))
|
||||||
self.Base.logs.debug(f'{send_message}')
|
self.Base.logs.debug(f'{send_message}')
|
||||||
|
|
||||||
@@ -268,9 +276,44 @@ class Irc:
|
|||||||
except OSError as oe:
|
except OSError as oe:
|
||||||
self.Base.logs.error(f"OSError: {oe} - {send_message}")
|
self.Base.logs.error(f"OSError: {oe} - {send_message}")
|
||||||
|
|
||||||
|
def sendNotice(self, msg:str, nickname: str) -> None:
|
||||||
|
"""Sending NOTICE by batches
|
||||||
|
|
||||||
|
Args:
|
||||||
|
msg (str): The message to send to the server
|
||||||
|
nickname (str): The reciever Nickname
|
||||||
|
"""
|
||||||
|
batch_size = self.Config.BATCH_SIZE
|
||||||
|
service_nickname = self.Config.SERVICE_NICKNAME
|
||||||
|
|
||||||
|
for i in range(0, len(str(msg)), batch_size):
|
||||||
|
batch = str(msg)[i:i+batch_size]
|
||||||
|
self.send2socket(f":{service_nickname} NOTICE {nickname} :{batch}")
|
||||||
|
|
||||||
|
def sendPrivMsg(self, msg: str, channel: str = None, nickname: str = None):
|
||||||
|
"""Sending PRIVMSG to a channel or to a nickname by batches
|
||||||
|
could be either channel or nickname not both together
|
||||||
|
Args:
|
||||||
|
msg (str): The message to send
|
||||||
|
channel (str, optional): The receiver channel. Defaults to None.
|
||||||
|
nickname (str, optional): The reciever nickname. Defaults to None.
|
||||||
|
"""
|
||||||
|
batch_size = self.Config.BATCH_SIZE
|
||||||
|
service_nickname = self.Config.SERVICE_NICKNAME
|
||||||
|
|
||||||
|
if not channel is None:
|
||||||
|
for i in range(0, len(str(msg)), batch_size):
|
||||||
|
batch = str(msg)[i:i+batch_size]
|
||||||
|
self.send2socket(f":{service_nickname} PRIVMSG {channel} :{batch}")
|
||||||
|
|
||||||
|
if not nickname is None:
|
||||||
|
for i in range(0, len(str(msg)), batch_size):
|
||||||
|
batch = str(msg)[i:i+batch_size]
|
||||||
|
self.send2socket(f":{service_nickname} PRIVMSG {nickname} :{batch}")
|
||||||
|
|
||||||
def send_response(self, responses:list[bytes]) -> None:
|
def send_response(self, responses:list[bytes]) -> None:
|
||||||
try:
|
try:
|
||||||
# print(data)
|
# print(responses)
|
||||||
for data in responses:
|
for data in responses:
|
||||||
response = data.decode(self.CHARSET[0]).split()
|
response = data.decode(self.CHARSET[0]).split()
|
||||||
self.cmd(response)
|
self.cmd(response)
|
||||||
@@ -452,17 +495,100 @@ class Irc:
|
|||||||
|
|
||||||
except ModuleNotFoundError as moduleNotFound:
|
except ModuleNotFoundError as moduleNotFound:
|
||||||
self.Base.logs.error(f"MODULE_NOT_FOUND: {moduleNotFound}")
|
self.Base.logs.error(f"MODULE_NOT_FOUND: {moduleNotFound}")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.CONFIG_COLOR['rouge']}MODULE_NOT_FOUND{self.Config.CONFIG_COLOR['noire']} ]: {moduleNotFound}")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.COLORS.red}MODULE_NOT_FOUND{self.Config.COLORS.black} ]: {moduleNotFound}")
|
||||||
self.Base.db_delete_module(module_name)
|
self.Base.db_delete_module(module_name)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.Base.logs.error(f"Something went wrong with a module you want to load : {e}")
|
self.Base.logs.error(f"Something went wrong with a module you want to load : {e}")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.CONFIG_COLOR['rouge']}ERROR{self.Config.CONFIG_COLOR['noire']} ]: {e}")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.COLORS.red}ERROR{self.Config.COLORS.black} ]: {e}")
|
||||||
|
|
||||||
|
def unload_module(self, mod_name: str) -> bool:
|
||||||
|
"""Unload a module
|
||||||
|
|
||||||
|
Args:
|
||||||
|
mod_name (str): Module name ex mod_defender
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: True if success
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
module_name = mod_name.lower() # Le nom du module. exemple: mod_defender
|
||||||
|
class_name = module_name.split('_')[1].capitalize() # Nom de la class. exemple: Defender
|
||||||
|
|
||||||
|
if class_name in self.loaded_classes:
|
||||||
|
self.loaded_classes[class_name].unload()
|
||||||
|
for level, command in self.loaded_classes[class_name].commands_level.items():
|
||||||
|
# Supprimer la commande de la variable commands
|
||||||
|
for c in self.loaded_classes[class_name].commands_level[level]:
|
||||||
|
self.commands.remove(c)
|
||||||
|
self.commands_level[level].remove(c)
|
||||||
|
|
||||||
|
del self.loaded_classes[class_name]
|
||||||
|
|
||||||
|
# Supprimer le module de la base de données
|
||||||
|
self.Base.db_delete_module(module_name)
|
||||||
|
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} supprimé")
|
||||||
|
return True
|
||||||
|
|
||||||
|
except Exception as err:
|
||||||
|
self.Base.logs.error(f"General Error: {err}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
def reload_module(self, from_user: str, mod_name: str) -> bool:
|
||||||
|
try:
|
||||||
|
module_name = mod_name.lower() # ==> mod_defender
|
||||||
|
class_name = module_name.split('_')[1].capitalize() # ==> Defender
|
||||||
|
|
||||||
|
if 'mods.' + module_name in sys.modules:
|
||||||
|
self.Base.logs.info('Unload the module ...')
|
||||||
|
self.loaded_classes[class_name].unload()
|
||||||
|
self.Base.logs.info('Module Already Loaded ... reloading the module ...')
|
||||||
|
the_module = sys.modules['mods.' + module_name]
|
||||||
|
importlib.reload(the_module)
|
||||||
|
|
||||||
|
# Supprimer la class déja instancier
|
||||||
|
if class_name in self.loaded_classes:
|
||||||
|
# Supprimer les commandes déclarer dans la classe
|
||||||
|
for level, command in self.loaded_classes[class_name].commands_level.items():
|
||||||
|
# Supprimer la commande de la variable commands
|
||||||
|
for c in self.loaded_classes[class_name].commands_level[level]:
|
||||||
|
self.commands.remove(c)
|
||||||
|
self.commands_level[level].remove(c)
|
||||||
|
|
||||||
|
del self.loaded_classes[class_name]
|
||||||
|
|
||||||
|
my_class = getattr(the_module, class_name, None)
|
||||||
|
new_instance = my_class(self.ircObject)
|
||||||
|
self.loaded_classes[class_name] = new_instance
|
||||||
|
|
||||||
|
self.Base.db_update_module(from_user, mod_name)
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} rechargé")
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} n'est pas chargé !")
|
||||||
|
|
||||||
|
except TypeError as te:
|
||||||
|
self.Base.logs.error(f"A TypeError raised: {te}")
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :A TypeError raised: {te}")
|
||||||
|
self.Base.db_delete_module(module_name)
|
||||||
|
except AttributeError as ae:
|
||||||
|
self.Base.logs.error(f"Missing Attribute: {ae}")
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Missing Attribute: {ae}")
|
||||||
|
self.Base.db_delete_module(module_name)
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(f"Key Error: {ke}")
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Key Error: {ke}")
|
||||||
|
self.Base.db_delete_module(module_name)
|
||||||
|
except Exception as e:
|
||||||
|
self.Base.logs.error(f"Something went wrong with a module you want to reload: {e}")
|
||||||
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Something went wrong with the module: {e}")
|
||||||
|
self.Base.db_delete_module(module_name)
|
||||||
|
|
||||||
def insert_db_admin(self, uid:str, level:int) -> None:
|
def insert_db_admin(self, uid:str, level:int) -> None:
|
||||||
|
|
||||||
if self.User.get_User(uid) is None:
|
if self.User.get_User(uid) is None:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
getUser = self.User.get_User(uid)
|
getUser = self.User.get_User(uid)
|
||||||
|
|
||||||
nickname = getUser.nickname
|
nickname = getUser.nickname
|
||||||
@@ -655,6 +781,30 @@ class Irc:
|
|||||||
# self.Base.create_thread(self.abuseipdb_scan, (cmd[7], ))
|
# self.Base.create_thread(self.abuseipdb_scan, (cmd[7], ))
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
case 'SQUIT':
|
||||||
|
# ['@msgid=QOEolbRxdhpVW5c8qLkbAU;time=2024-09-21T17:33:16.547Z', 'SQUIT', 'defender.deb.biz.st', ':Connection', 'closed']
|
||||||
|
server_hostname = interm_response[1]
|
||||||
|
uid_to_delete = ''
|
||||||
|
for s_user in self.User.UID_DB:
|
||||||
|
if s_user.hostname == server_hostname and 'S' in s_user.umodes:
|
||||||
|
uid_to_delete = s_user.uid
|
||||||
|
|
||||||
|
self.User.delete(uid_to_delete)
|
||||||
|
self.Channel.delete_user_from_all_channel(uid_to_delete)
|
||||||
|
|
||||||
|
case 'SJOIN':
|
||||||
|
# If Server Join channels
|
||||||
|
# [':11Z', 'SJOIN', '1726940687', '#welcome', '+', ':11ZAAAAAB']
|
||||||
|
channel_joined = original_response[3]
|
||||||
|
server_uid = self.Base.clean_uid(original_response[5])
|
||||||
|
|
||||||
|
self.Channel.insert(
|
||||||
|
self.Channel.ChannelModel(
|
||||||
|
name=channel_joined,
|
||||||
|
uids=[server_uid]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
case 'REPUTATION':
|
case 'REPUTATION':
|
||||||
# :001 REPUTATION 91.168.141.239 118
|
# :001 REPUTATION 91.168.141.239 118
|
||||||
try:
|
try:
|
||||||
@@ -723,13 +873,21 @@ class Irc:
|
|||||||
self.Base.logs.info(f"# CHANNEL : {self.Config.SERVICE_CHANLOG} ")
|
self.Base.logs.info(f"# CHANNEL : {self.Config.SERVICE_CHANLOG} ")
|
||||||
self.Base.logs.info(f"# VERSION : {version} ")
|
self.Base.logs.info(f"# VERSION : {version} ")
|
||||||
self.Base.logs.info(f"################################################")
|
self.Base.logs.info(f"################################################")
|
||||||
|
|
||||||
if self.Base.check_for_new_version(False):
|
if self.Base.check_for_new_version(False):
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} : New Version available {version}")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} : New Version available {version}")
|
||||||
|
|
||||||
# Initialisation terminé aprés le premier PING
|
# Initialisation terminé aprés le premier PING
|
||||||
|
self.sendPrivMsg(msg=f'[{self.Config.COLORS.green}INFORMATION{self.Config.COLORS.nogc}] >> Defender is ready', channel=self.Config.SERVICE_CHANLOG)
|
||||||
self.INIT = 0
|
self.INIT = 0
|
||||||
|
|
||||||
|
# Send EOF to other modules
|
||||||
|
for classe_name, classe_object in self.loaded_classes.items():
|
||||||
|
classe_object.cmd(original_response)
|
||||||
|
|
||||||
|
# Stop here When EOS
|
||||||
|
return None
|
||||||
|
|
||||||
case _:
|
case _:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@@ -904,7 +1062,8 @@ class Irc:
|
|||||||
arg.remove(f':{self.Config.SERVICE_PREFIX}')
|
arg.remove(f':{self.Config.SERVICE_PREFIX}')
|
||||||
if not arg[0].lower() in self.commands:
|
if not arg[0].lower() in self.commands:
|
||||||
self.Base.logs.debug(f"This command {arg[0]} is not available")
|
self.Base.logs.debug(f"This command {arg[0]} is not available")
|
||||||
return False
|
self.sendNotice(f"This command [{self.Config.COLORS.bold}{arg[0]}{self.Config.COLORS.bold}] is not available", user_trigger)
|
||||||
|
return None
|
||||||
|
|
||||||
cmd_to_send = convert_to_string.replace(':','')
|
cmd_to_send = convert_to_string.replace(':','')
|
||||||
self.Base.log_cmd(user_trigger, cmd_to_send)
|
self.Base.log_cmd(user_trigger, cmd_to_send)
|
||||||
@@ -938,7 +1097,9 @@ class Irc:
|
|||||||
recieved_unixtime = int(arg[1].replace('\x01',''))
|
recieved_unixtime = int(arg[1].replace('\x01',''))
|
||||||
current_unixtime = self.Base.get_unixtime()
|
current_unixtime = self.Base.get_unixtime()
|
||||||
ping_response = current_unixtime - recieved_unixtime
|
ping_response = current_unixtime - recieved_unixtime
|
||||||
self.send2socket(f':{dnickname} NOTICE {user_trigger} :\x01PING {str(ping_response)} secs\x01')
|
|
||||||
|
self.send2socket(f'PONG :{recieved_unixtime}')
|
||||||
|
self.send2socket(f':{dnickname} NOTICE {user_trigger} :\x01PING {ping_response} secs\x01')
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if not arg[0].lower() in self.commands:
|
if not arg[0].lower() in self.commands:
|
||||||
@@ -967,6 +1128,9 @@ class Irc:
|
|||||||
|
|
||||||
except IndexError as ie:
|
except IndexError as ie:
|
||||||
self.Base.logs.error(f"{ie} / {original_response} / length {str(len(original_response))}")
|
self.Base.logs.error(f"{ie} / {original_response} / length {str(len(original_response))}")
|
||||||
|
except Exception as err:
|
||||||
|
self.Base.logs.error(f"General Error: {err}")
|
||||||
|
self.Base.logs.error(f"General Error: {traceback.format_exc()}")
|
||||||
|
|
||||||
def _hcmds(self, user: str, channel: Union[str, None], cmd: list, fullcmd: list = []) -> None:
|
def _hcmds(self, user: str, channel: Union[str, None], cmd: list, fullcmd: list = []) -> None:
|
||||||
"""_summary_
|
"""_summary_
|
||||||
@@ -1007,7 +1171,7 @@ class Irc:
|
|||||||
case 'notallowed':
|
case 'notallowed':
|
||||||
try:
|
try:
|
||||||
current_command = cmd[0]
|
current_command = cmd[0]
|
||||||
self.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["rouge"]}{current_command}{self.Config.CONFIG_COLOR["noire"]} ] - Accès Refusé à {self.User.get_nickname(fromuser)}')
|
self.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}{current_command}{self.Config.COLORS.black} ] - Accès Refusé à {self.User.get_nickname(fromuser)}')
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : Accès Refusé')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : Accès Refusé')
|
||||||
except IndexError as ie:
|
except IndexError as ie:
|
||||||
self.Base.logs.error(f'{ie}')
|
self.Base.logs.error(f'{ie}')
|
||||||
@@ -1017,7 +1181,7 @@ class Irc:
|
|||||||
current_command = cmd[0]
|
current_command = cmd[0]
|
||||||
uid_to_deauth = self.User.get_uid(fromuser)
|
uid_to_deauth = self.User.get_uid(fromuser)
|
||||||
self.delete_db_admin(uid_to_deauth)
|
self.delete_db_admin(uid_to_deauth)
|
||||||
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}{str(current_command).upper()} ]{self.Config.CONFIG_COLOR['noire']} - {self.User.get_nickname(fromuser)} est désormais déconnecter de {dnickname}")
|
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}{str(current_command).upper()} ]{self.Config.COLORS.black} - {self.User.get_nickname(fromuser)} est désormais déconnecter de {dnickname}")
|
||||||
|
|
||||||
case 'firstauth':
|
case 'firstauth':
|
||||||
# firstauth OWNER_NICKNAME OWNER_PASSWORD
|
# firstauth OWNER_NICKNAME OWNER_PASSWORD
|
||||||
@@ -1063,10 +1227,10 @@ class Irc:
|
|||||||
if cmd_owner == config_owner and cmd_password == config_password:
|
if cmd_owner == config_owner and cmd_password == config_password:
|
||||||
self.Base.db_create_first_admin()
|
self.Base.db_create_first_admin()
|
||||||
self.insert_db_admin(current_uid, 5)
|
self.insert_db_admin(current_uid, 5)
|
||||||
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}{str(current_command).upper()} ]{self.Config.CONFIG_COLOR['noire']} - {self.User.get_nickname(fromuser)} est désormais connecté a {dnickname}")
|
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}{str(current_command).upper()} ]{self.Config.COLORS.black} - {self.User.get_nickname(fromuser)} est désormais connecté a {dnickname}")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Connexion a {dnickname} réussie!")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Connexion a {dnickname} réussie!")
|
||||||
else:
|
else:
|
||||||
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}{str(current_command).upper()} ]{self.Config.CONFIG_COLOR['noire']} - {self.User.get_nickname(fromuser)} a tapé un mauvais mot de pass")
|
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}{str(current_command).upper()} ]{self.Config.COLORS.black} - {self.User.get_nickname(fromuser)} a tapé un mauvais mot de pass")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Mot de passe incorrecte")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Mot de passe incorrecte")
|
||||||
|
|
||||||
case 'auth':
|
case 'auth':
|
||||||
@@ -1089,10 +1253,10 @@ class Irc:
|
|||||||
if not user_from_db is None:
|
if not user_from_db is None:
|
||||||
uid_user = self.User.get_uid(user_to_log)
|
uid_user = self.User.get_uid(user_to_log)
|
||||||
self.insert_db_admin(uid_user, user_from_db[1])
|
self.insert_db_admin(uid_user, user_from_db[1])
|
||||||
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}{str(current_command).upper()} ]{self.Config.CONFIG_COLOR['noire']} - {self.User.get_nickname(fromuser)} est désormais connecté a {dnickname}")
|
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}{str(current_command).upper()} ]{self.Config.COLORS.black} - {self.User.get_nickname(fromuser)} est désormais connecté a {dnickname}")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Connexion a {dnickname} réussie!")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Connexion a {dnickname} réussie!")
|
||||||
else:
|
else:
|
||||||
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}{str(current_command).upper()} ]{self.Config.CONFIG_COLOR['noire']} - {self.User.get_nickname(fromuser)} a tapé un mauvais mot de pass")
|
self.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}{str(current_command).upper()} ]{self.Config.COLORS.black} - {self.User.get_nickname(fromuser)} a tapé un mauvais mot de pass")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Mot de passe incorrecte")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} NOTICE {fromuser} :Mot de passe incorrecte")
|
||||||
|
|
||||||
else:
|
else:
|
||||||
@@ -1223,7 +1387,6 @@ class Irc:
|
|||||||
|
|
||||||
case 'help':
|
case 'help':
|
||||||
|
|
||||||
help = ''
|
|
||||||
count_level_definition = 0
|
count_level_definition = 0
|
||||||
get_admin = self.Admin.get_Admin(uid)
|
get_admin = self.Admin.get_Admin(uid)
|
||||||
if not get_admin is None:
|
if not get_admin is None:
|
||||||
@@ -1231,100 +1394,49 @@ class Irc:
|
|||||||
else:
|
else:
|
||||||
user_level = 0
|
user_level = 0
|
||||||
|
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : **************** LIST DES COMMANDES *****************')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : ***************** LISTE DES COMMANDES *****************')
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : ')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : ')
|
||||||
for levDef in self.commands_level:
|
for levDef in self.commands_level:
|
||||||
|
|
||||||
if int(user_level) >= int(count_level_definition):
|
if int(user_level) >= int(count_level_definition):
|
||||||
|
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : **************** {self.Config.CONFIG_COLOR["noire"]}[ {self.Config.CONFIG_COLOR["verte"]}LEVEL {str(levDef)} {self.Config.CONFIG_COLOR["noire"]}] ****************')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : ***************** {self.Config.COLORS.nogc}[ {self.Config.COLORS.green}LEVEL {str(levDef)} {self.Config.COLORS.nogc}] *****************')
|
||||||
count_commands = 0
|
|
||||||
help = ''
|
|
||||||
for comm in self.commands_level[count_level_definition]:
|
|
||||||
|
|
||||||
help += f"{comm.upper()}"
|
batch = 7
|
||||||
if int(count_commands) < len(self.commands_level[count_level_definition])-1:
|
for i in range(0, len(self.commands_level[count_level_definition]), batch):
|
||||||
help += ' | '
|
groupe = self.commands_level[count_level_definition][i:i + batch] # Extraire le groupe
|
||||||
count_commands += 1
|
batch_commands = ' | '.join(groupe)
|
||||||
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : {batch_commands}')
|
||||||
|
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : {help}')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : ')
|
||||||
|
|
||||||
count_level_definition += 1
|
count_level_definition += 1
|
||||||
|
|
||||||
self.send2socket(f':{dnickname} NOTICE {fromuser} : **************** FIN DES COMMANDES *****************')
|
self.send2socket(f':{dnickname} NOTICE {fromuser} : ***************** FIN DES COMMANDES *****************')
|
||||||
|
|
||||||
case 'load':
|
case 'load':
|
||||||
|
try:
|
||||||
self.load_module(fromuser, str(cmd[1]))
|
# Load a module ex: .load mod_defender
|
||||||
|
mod_name = str(cmd[1])
|
||||||
|
self.load_module(fromuser, mod_name)
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(f"Key Error: {ke} - list recieved: {cmd}")
|
||||||
|
except Exception as err:
|
||||||
|
self.Base.logs.error(f"General Error: {ke} - list recieved: {cmd}")
|
||||||
|
|
||||||
case 'unload':
|
case 'unload':
|
||||||
# unload mod_dktmb
|
# unload mod_defender
|
||||||
try:
|
try:
|
||||||
module_name = str(cmd[1]).lower() # Le nom du module. exemple: mod_defender
|
module_name = str(cmd[1]).lower() # Le nom du module. exemple: mod_defender
|
||||||
class_name = module_name.split('_')[1].capitalize() # Nom de la class. exemple: Defender
|
self.unload_module(module_name)
|
||||||
|
except Exception as err:
|
||||||
if class_name in self.loaded_classes:
|
self.Base.logs.error(f"General Error: {err}")
|
||||||
self.loaded_classes[class_name].unload()
|
|
||||||
for level, command in self.loaded_classes[class_name].commands_level.items():
|
|
||||||
# Supprimer la commande de la variable commands
|
|
||||||
for c in self.loaded_classes[class_name].commands_level[level]:
|
|
||||||
self.commands.remove(c)
|
|
||||||
self.commands_level[level].remove(c)
|
|
||||||
|
|
||||||
del self.loaded_classes[class_name]
|
|
||||||
|
|
||||||
# Supprimer le module de la base de données
|
|
||||||
self.Base.db_delete_module(module_name)
|
|
||||||
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} supprimé")
|
|
||||||
except:
|
|
||||||
self.Base.logs.error(f"Something went wrong with a module you want to load")
|
|
||||||
|
|
||||||
case 'reload':
|
case 'reload':
|
||||||
# reload mod_dktmb
|
# reload mod_defender
|
||||||
try:
|
try:
|
||||||
module_name = str(cmd[1]).lower() # ==> mod_defender
|
module_name = str(cmd[1]).lower() # ==> mod_defender
|
||||||
class_name = module_name.split('_')[1].capitalize() # ==> Defender
|
self.reload_module(from_user=fromuser, mod_name=module_name)
|
||||||
|
|
||||||
if 'mods.' + module_name in sys.modules:
|
|
||||||
self.Base.logs.info('Unload the module ...')
|
|
||||||
self.loaded_classes[class_name].unload()
|
|
||||||
self.Base.logs.info('Module Already Loaded ... reloading the module ...')
|
|
||||||
the_module = sys.modules['mods.' + module_name]
|
|
||||||
importlib.reload(the_module)
|
|
||||||
|
|
||||||
# Supprimer la class déja instancier
|
|
||||||
if class_name in self.loaded_classes:
|
|
||||||
# Supprimer les commandes déclarer dans la classe
|
|
||||||
for level, command in self.loaded_classes[class_name].commands_level.items():
|
|
||||||
# Supprimer la commande de la variable commands
|
|
||||||
for c in self.loaded_classes[class_name].commands_level[level]:
|
|
||||||
self.commands.remove(c)
|
|
||||||
self.commands_level[level].remove(c)
|
|
||||||
|
|
||||||
del self.loaded_classes[class_name]
|
|
||||||
|
|
||||||
my_class = getattr(the_module, class_name, None)
|
|
||||||
new_instance = my_class(self.ircObject)
|
|
||||||
self.loaded_classes[class_name] = new_instance
|
|
||||||
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} rechargé")
|
|
||||||
return False
|
|
||||||
else:
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Module {module_name} n'est pas chargé !")
|
|
||||||
|
|
||||||
except TypeError as te:
|
|
||||||
self.Base.logs.error(f"A TypeError raised: {te}")
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :A TypeError raised: {te}")
|
|
||||||
self.Base.db_delete_module(module_name)
|
|
||||||
except AttributeError as ae:
|
|
||||||
self.Base.logs.error(f"Missing Attribute: {ae}")
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Missing Attribute: {ae}")
|
|
||||||
self.Base.db_delete_module(module_name)
|
|
||||||
except KeyError as ke:
|
|
||||||
self.Base.logs.error(f"Key Error: {ke}")
|
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Key Error: {ke}")
|
|
||||||
self.Base.db_delete_module(module_name)
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.Base.logs.error(f"Something went wrong with a module you want to reload: {e}")
|
self.Base.logs.error(f"Something went wrong with a module you want to reload: {e}")
|
||||||
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Something went wrong with the module: {e}")
|
self.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :Something went wrong with the module: {e}")
|
||||||
@@ -1375,23 +1487,23 @@ class Irc:
|
|||||||
|
|
||||||
self.Base.logs.debug(self.loaded_classes)
|
self.Base.logs.debug(self.loaded_classes)
|
||||||
all_modules = self.Base.get_all_modules()
|
all_modules = self.Base.get_all_modules()
|
||||||
|
loaded = False
|
||||||
|
|
||||||
results = self.Base.db_execute_query(f'SELECT module_name FROM {self.Config.table_module}')
|
results = self.Base.db_execute_query(f'SELECT datetime, user, module_name FROM {self.Config.table_module}')
|
||||||
results = results.fetchall()
|
results = results.fetchall()
|
||||||
|
|
||||||
found = False
|
|
||||||
|
|
||||||
for module in all_modules:
|
for module in all_modules:
|
||||||
for loaded_mod in results:
|
for loaded_mod in results:
|
||||||
if module == loaded_mod[0]:
|
if module == loaded_mod[2]:
|
||||||
found = True
|
loaded_datetime = loaded_mod[0]
|
||||||
|
loaded_user = loaded_mod[1]
|
||||||
|
loaded = True
|
||||||
|
|
||||||
if found:
|
if loaded:
|
||||||
self.send2socket(f":{dnickname} NOTICE {fromuser} :{module} - {self.Config.CONFIG_COLOR['verte']}Loaded{self.Config.CONFIG_COLOR['nogc']}")
|
self.send2socket(f":{dnickname} NOTICE {fromuser} :{module} - {self.Config.COLORS.green}Loaded{self.Config.COLORS.nogc} by {loaded_user} on {loaded_datetime}")
|
||||||
|
loaded = False
|
||||||
else:
|
else:
|
||||||
self.send2socket(f":{dnickname} NOTICE {fromuser} :{module} - {self.Config.CONFIG_COLOR['rouge']}Not Loaded{self.Config.CONFIG_COLOR['nogc']}")
|
self.send2socket(f":{dnickname} NOTICE {fromuser} :{module} - {self.Config.COLORS.red}Not Loaded{self.Config.COLORS.nogc}")
|
||||||
|
|
||||||
found = False
|
|
||||||
|
|
||||||
case 'show_timers':
|
case 'show_timers':
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import json, sys
|
import sys
|
||||||
|
import json
|
||||||
from os import sep
|
from os import sep
|
||||||
from typing import Union, Literal
|
from typing import Union, Literal
|
||||||
from dataclasses import dataclass, field
|
from dataclasses import dataclass, field
|
||||||
@@ -7,6 +8,17 @@ from dataclasses import dataclass, field
|
|||||||
# CONFIGURATION FILE #
|
# CONFIGURATION FILE #
|
||||||
##########################################
|
##########################################
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class ColorModel:
|
||||||
|
white: str = "\x0300"
|
||||||
|
black: str = "\x0301"
|
||||||
|
blue: str = "\x0302"
|
||||||
|
green: str = "\x0303"
|
||||||
|
red: str = "\x0304"
|
||||||
|
yellow: str = "\x0306"
|
||||||
|
bold: str = "\x02"
|
||||||
|
nogc: str = "\x03"
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class ConfigDataModel:
|
class ConfigDataModel:
|
||||||
|
|
||||||
@@ -73,6 +85,21 @@ class ConfigDataModel:
|
|||||||
PASSWORD: str
|
PASSWORD: str
|
||||||
"""The password of the admin of the service"""
|
"""The password of the admin of the service"""
|
||||||
|
|
||||||
|
JSONRPC_URL: str
|
||||||
|
"""The RPC url, if local https://127.0.0.1:PORT/api should be fine"""
|
||||||
|
|
||||||
|
JSONRPC_PATH_TO_SOCKET_FILE: str
|
||||||
|
"""The full path of the socket file (/PATH/TO/YOUR/UNREALIRCD/SOCKET/FILE.socket)"""
|
||||||
|
|
||||||
|
JSONRPC_METHOD: str
|
||||||
|
"""3 methods are available; requests/socket/unixsocket"""
|
||||||
|
|
||||||
|
JSONRPC_USER: str
|
||||||
|
"""The RPC User defined in your unrealircd.conf"""
|
||||||
|
|
||||||
|
JSONRPC_PASSWORD: str
|
||||||
|
"""The RPC Password defined in your unrealircd.conf"""
|
||||||
|
|
||||||
SALON_JAIL: str
|
SALON_JAIL: str
|
||||||
"""The JAIL channel (ex. #jail)"""
|
"""The JAIL channel (ex. #jail)"""
|
||||||
|
|
||||||
@@ -82,8 +109,17 @@ class ConfigDataModel:
|
|||||||
SALON_LIBERER: str
|
SALON_LIBERER: str
|
||||||
"""Channel where the nickname will be released"""
|
"""Channel where the nickname will be released"""
|
||||||
|
|
||||||
SALON_CLONES: str
|
CLONE_CHANNEL: str
|
||||||
"""Channel to host clones"""
|
"""Channel where clones are hosted and will log PRIVMSG"""
|
||||||
|
|
||||||
|
CLONE_CMODES: str
|
||||||
|
"""Clone channel modes"""
|
||||||
|
|
||||||
|
CLONE_LOG_HOST_EXEMPT: list[str]
|
||||||
|
"""Hosts that clones will not log"""
|
||||||
|
|
||||||
|
CLONE_CHANNEL_PASSWORD: str
|
||||||
|
"""Clone password channel"""
|
||||||
|
|
||||||
API_TIMEOUT: int
|
API_TIMEOUT: int
|
||||||
"""Default api timeout in second"""
|
"""Default api timeout in second"""
|
||||||
@@ -100,8 +136,6 @@ class ConfigDataModel:
|
|||||||
DEBUG_LEVEL:Literal[10, 20, 30, 40, 50]
|
DEBUG_LEVEL:Literal[10, 20, 30, 40, 50]
|
||||||
"""Logs level: DEBUG 10 | INFO 20 | WARNING 30 | ERROR 40 | CRITICAL 50"""
|
"""Logs level: DEBUG 10 | INFO 20 | WARNING 30 | ERROR 40 | CRITICAL 50"""
|
||||||
|
|
||||||
CONFIG_COLOR: dict[str, str]
|
|
||||||
|
|
||||||
table_admin: str
|
table_admin: str
|
||||||
"""Admin table"""
|
"""Admin table"""
|
||||||
|
|
||||||
@@ -132,6 +166,12 @@ class ConfigDataModel:
|
|||||||
db_path: str
|
db_path: str
|
||||||
"""The database path"""
|
"""The database path"""
|
||||||
|
|
||||||
|
COLORS: ColorModel = field(default_factory=ColorModel)
|
||||||
|
"""Available colors in Defender"""
|
||||||
|
|
||||||
|
BATCH_SIZE: int = 400
|
||||||
|
"""The batch size used for privmsg and notice"""
|
||||||
|
|
||||||
def __post_init__(self):
|
def __post_init__(self):
|
||||||
# Initialiser SERVICE_ID après la création de l'objet
|
# Initialiser SERVICE_ID après la création de l'objet
|
||||||
self.SERVICE_ID:str = f"{self.SERVEUR_ID}AAAAAB"
|
self.SERVICE_ID:str = f"{self.SERVEUR_ID}AAAAAB"
|
||||||
@@ -150,34 +190,6 @@ class Config:
|
|||||||
with open(conf_filename, 'r') as configuration_data:
|
with open(conf_filename, 'r') as configuration_data:
|
||||||
configuration:dict[str, Union[str, int, list, dict]] = json.load(configuration_data)
|
configuration:dict[str, Union[str, int, list, dict]] = json.load(configuration_data)
|
||||||
|
|
||||||
config_dict = {"CONFIG_COLOR" : {
|
|
||||||
"blanche": "\x0300",
|
|
||||||
"noire": "\x0301",
|
|
||||||
"bleue": "\x0302",
|
|
||||||
"verte": "\x0303",
|
|
||||||
"rouge": "\x0304",
|
|
||||||
"jaune": "\x0306",
|
|
||||||
"gras": "\x02",
|
|
||||||
"nogc": "\x02\x03"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
missing_color = False
|
|
||||||
|
|
||||||
if not "CONFIG_COLOR" in configuration:
|
|
||||||
missing_color = True
|
|
||||||
configuration_color = config_dict
|
|
||||||
else:
|
|
||||||
configuration_color = configuration["CONFIG_COLOR"]
|
|
||||||
|
|
||||||
if missing_color:
|
|
||||||
for key, value in configuration_color.items():
|
|
||||||
configuration_color['CONFIG_COLOR'][key] = str(value).encode('utf-8').decode('unicode_escape')
|
|
||||||
configuration['CONFIG_COLOR'] = configuration_color['CONFIG_COLOR']
|
|
||||||
else:
|
|
||||||
for key, value in configuration['CONFIG_COLOR'].items():
|
|
||||||
configuration['CONFIG_COLOR'][key] = str(value).encode('utf-8').decode('unicode_escape')
|
|
||||||
|
|
||||||
return configuration
|
return configuration
|
||||||
|
|
||||||
except FileNotFoundError as fe:
|
except FileNotFoundError as fe:
|
||||||
@@ -212,16 +224,27 @@ class Config:
|
|||||||
SERVICE_PREFIX=import_config["SERVICE_PREFIX"] if "SERVICE_PREFIX" in import_config else '!',
|
SERVICE_PREFIX=import_config["SERVICE_PREFIX"] if "SERVICE_PREFIX" in import_config else '!',
|
||||||
OWNER=import_config["OWNER"] if "OWNER" in import_config else 'admin',
|
OWNER=import_config["OWNER"] if "OWNER" in import_config else 'admin',
|
||||||
PASSWORD=import_config["PASSWORD"] if "PASSWORD" in import_config else 'admin',
|
PASSWORD=import_config["PASSWORD"] if "PASSWORD" in import_config else 'admin',
|
||||||
|
|
||||||
|
JSONRPC_METHOD=import_config["JSONRPC_METHOD"] if "JSONRPC_METHOD" in import_config else 'socket',
|
||||||
|
JSONRPC_URL=import_config["JSONRPC_URL"] if "JSONRPC_URL" in import_config else None,
|
||||||
|
JSONRPC_PATH_TO_SOCKET_FILE=import_config["JSONRPC_PATH_TO_SOCKET_FILE"] if "JSONRPC_PATH_TO_SOCKET_FILE" in import_config else None,
|
||||||
|
JSONRPC_USER=import_config["JSONRPC_USER"] if "JSONRPC_USER" in import_config else None,
|
||||||
|
JSONRPC_PASSWORD=import_config["JSONRPC_PASSWORD"] if "JSONRPC_PASSWORD" in import_config else None,
|
||||||
|
|
||||||
SALON_JAIL=import_config["SALON_JAIL"] if "SALON_JAIL" in import_config else '#jail',
|
SALON_JAIL=import_config["SALON_JAIL"] if "SALON_JAIL" in import_config else '#jail',
|
||||||
SALON_JAIL_MODES=import_config["SALON_JAIL_MODES"] if "SALON_JAIL_MODES" in import_config else 'sS',
|
SALON_JAIL_MODES=import_config["SALON_JAIL_MODES"] if "SALON_JAIL_MODES" in import_config else 'sS',
|
||||||
SALON_LIBERER=import_config["SALON_LIBERER"] if "SALON_LIBERER" in import_config else '#welcome',
|
SALON_LIBERER=import_config["SALON_LIBERER"] if "SALON_LIBERER" in import_config else '#welcome',
|
||||||
SALON_CLONES=import_config["SALON_CLONES"] if "SALON_CLONES" in import_config else '#clones',
|
|
||||||
|
CLONE_CHANNEL=import_config["CLONE_CHANNEL"] if "CLONE_CHANNEL" in import_config else '#clones',
|
||||||
|
CLONE_CMODES=import_config["CLONE_CMODES"] if "CLONE_CMODES" in import_config else '+nts',
|
||||||
|
CLONE_LOG_HOST_EXEMPT=import_config["CLONE_LOG_HOST_EXEMPT"] if "CLONE_LOG_HOST_EXEMPT" in import_config else [],
|
||||||
|
CLONE_CHANNEL_PASSWORD=import_config["CLONE_CHANNEL_PASSWORD"] if "CLONE_CHANNEL_PASSWORD" in import_config else "clone_Password_1234",
|
||||||
|
|
||||||
API_TIMEOUT=import_config["API_TIMEOUT"] if "API_TIMEOUT" in import_config else 2,
|
API_TIMEOUT=import_config["API_TIMEOUT"] if "API_TIMEOUT" in import_config else 2,
|
||||||
PORTS_TO_SCAN=import_config["PORTS_TO_SCAN"] if "PORTS_TO_SCAN" in import_config else [],
|
PORTS_TO_SCAN=import_config["PORTS_TO_SCAN"] if "PORTS_TO_SCAN" in import_config else [],
|
||||||
WHITELISTED_IP=import_config["WHITELISTED_IP"] if "WHITELISTED_IP" in import_config else ['127.0.0.1'],
|
WHITELISTED_IP=import_config["WHITELISTED_IP"] if "WHITELISTED_IP" in import_config else ['127.0.0.1'],
|
||||||
GLINE_DURATION=import_config["GLINE_DURATION"] if "GLINE_DURATION" in import_config else '30',
|
GLINE_DURATION=import_config["GLINE_DURATION"] if "GLINE_DURATION" in import_config else '30',
|
||||||
DEBUG_LEVEL=import_config["DEBUG_LEVEL"] if "DEBUG_LEVEL" in import_config else 20,
|
DEBUG_LEVEL=import_config["DEBUG_LEVEL"] if "DEBUG_LEVEL" in import_config else 20,
|
||||||
CONFIG_COLOR=import_config["CONFIG_COLOR"],
|
|
||||||
table_admin='core_admin',
|
table_admin='core_admin',
|
||||||
table_commande='core_command',
|
table_commande='core_command',
|
||||||
table_log='core_log',
|
table_log='core_log',
|
||||||
|
|||||||
@@ -58,6 +58,12 @@ class Clone():
|
|||||||
# Load module configuration (Mandatory)
|
# Load module configuration (Mandatory)
|
||||||
self.__load_module_configuration()
|
self.__load_module_configuration()
|
||||||
|
|
||||||
|
self.Base.db_query_channel(action='add', module_name=self.module_name, channel_name=self.Config.CLONE_CHANNEL)
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} JOIN {self.Config.CLONE_CHANNEL}")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} SAMODE {self.Config.CLONE_CHANNEL} +o {self.Config.SERVICE_NICKNAME}")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} MODE {self.Config.CLONE_CHANNEL} +nts")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} MODE {self.Config.CLONE_CHANNEL} +k {self.Config.CLONE_CHANNEL_PASSWORD}")
|
||||||
|
|
||||||
def __set_commands(self, commands:dict[int, list[str]]) -> None:
|
def __set_commands(self, commands:dict[int, list[str]]) -> None:
|
||||||
"""### Rajoute les commandes du module au programme principal
|
"""### Rajoute les commandes du module au programme principal
|
||||||
|
|
||||||
@@ -120,6 +126,10 @@ class Clone():
|
|||||||
for clone in self.ModConfig.clone_nicknames:
|
for clone in self.ModConfig.clone_nicknames:
|
||||||
self.Irc.send2socket(f':{self.Config.SERVICE_NICKNAME} PRIVMSG {clone} :KILL')
|
self.Irc.send2socket(f':{self.Config.SERVICE_NICKNAME} PRIVMSG {clone} :KILL')
|
||||||
|
|
||||||
|
self.Base.db_query_channel(action='del', module_name=self.module_name, channel_name=self.Config.CLONE_CHANNEL)
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} MODE {self.Config.CLONE_CHANNEL} -nts")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} MODE {self.Config.CLONE_CHANNEL} -k {self.Config.CLONE_CHANNEL_PASSWORD}")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} PART {self.Config.CLONE_CHANNEL}")
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def thread_clone_clean_up(self, wait: float):
|
def thread_clone_clean_up(self, wait: float):
|
||||||
@@ -139,6 +149,11 @@ class Clone():
|
|||||||
self.Logs.debug(f'<<{clone_nickname}>> object has been deleted')
|
self.Logs.debug(f'<<{clone_nickname}>> object has been deleted')
|
||||||
|
|
||||||
del clone_to_kill
|
del clone_to_kill
|
||||||
|
|
||||||
|
# If no more clones then stop this thread
|
||||||
|
if not self.Clone.UID_CLONE_DB:
|
||||||
|
break
|
||||||
|
|
||||||
time.sleep(wait)
|
time.sleep(wait)
|
||||||
|
|
||||||
def thread_change_hostname(self):
|
def thread_change_hostname(self):
|
||||||
@@ -166,7 +181,7 @@ class Clone():
|
|||||||
break
|
break
|
||||||
|
|
||||||
def thread_create_clones_with_interval(self, number_of_clones:int, channels: list, connection_interval: float):
|
def thread_create_clones_with_interval(self, number_of_clones:int, channels: list, connection_interval: float):
|
||||||
|
|
||||||
for i in range(number_of_clones):
|
for i in range(number_of_clones):
|
||||||
nickname, username, realname = self.generate_names()
|
nickname, username, realname = self.generate_names()
|
||||||
self.Base.create_thread(
|
self.Base.create_thread(
|
||||||
@@ -289,6 +304,7 @@ class Clone():
|
|||||||
|
|
||||||
case 'connect':
|
case 'connect':
|
||||||
try:
|
try:
|
||||||
|
# clone connect 5
|
||||||
number_of_clones = int(cmd[2])
|
number_of_clones = int(cmd[2])
|
||||||
connection_interval = int(cmd[3]) if len(cmd) == 4 else 0.5
|
connection_interval = int(cmd[3]) if len(cmd) == 4 else 0.5
|
||||||
self.Base.create_thread(
|
self.Base.create_thread(
|
||||||
@@ -350,7 +366,7 @@ class Clone():
|
|||||||
clone_count = len(self.Clone.UID_CLONE_DB)
|
clone_count = len(self.Clone.UID_CLONE_DB)
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :>> Number of connected clones: {clone_count}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :>> Number of connected clones: {clone_count}')
|
||||||
for clone_name in self.Clone.UID_CLONE_DB:
|
for clone_name in self.Clone.UID_CLONE_DB:
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :>> Nickname: {clone_name.nickname} | Username: {clone_name.username} | Realname: {clone_name.realname} | Vhost: {clone_name.vhost} | Connected: {clone_name.connected}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :>> Nickname: {clone_name.nickname} | Username: {clone_name.username} | Realname: {clone_name.realname} | Vhost: {clone_name.vhost} | Init: {clone_name.init} | Live: {clone_name.alive} | Connected: {clone_name.connected}')
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Logs.error(f'{err}')
|
self.Logs.error(f'{err}')
|
||||||
|
|
||||||
|
|||||||
@@ -35,9 +35,14 @@ class Command():
|
|||||||
# Create module commands (Mandatory)
|
# Create module commands (Mandatory)
|
||||||
self.commands_level = {
|
self.commands_level = {
|
||||||
1: ['join', 'part'],
|
1: ['join', 'part'],
|
||||||
2: ['owner', 'deowner', 'op', 'deop', 'halfop', 'dehalfop', 'voice',
|
2: ['owner', 'deowner', 'protect', 'deprotect', 'op', 'deop', 'halfop', 'dehalfop', 'voice',
|
||||||
'devoice', 'opall', 'deopall', 'devoiceall', 'voiceall', 'ban',
|
'devoice', 'opall', 'deopall', 'devoiceall', 'voiceall', 'ban',
|
||||||
'unban','kick', 'kickban', 'umode', 'svsjoin', 'svspart', 'svsnick']
|
'unban','kick', 'kickban', 'umode', 'mode', 'svsjoin', 'svspart', 'svsnick', 'topic',
|
||||||
|
'wallops', 'globops','gnotice','whois', 'names', 'invite', 'inviteme',
|
||||||
|
'sajoin', 'sapart',
|
||||||
|
'kill', 'gline', 'ungline', 'kline', 'unkline', 'shun', 'unshun',
|
||||||
|
'glinelist', 'shunlist', 'klinelist'],
|
||||||
|
3: ['map']
|
||||||
}
|
}
|
||||||
|
|
||||||
# Init the module
|
# Init the module
|
||||||
@@ -58,6 +63,9 @@ class Command():
|
|||||||
self.__load_module_configuration()
|
self.__load_module_configuration()
|
||||||
# End of mandatory methods you can start your customization #
|
# End of mandatory methods you can start your customization #
|
||||||
|
|
||||||
|
self.user_to_notice: str = ''
|
||||||
|
self.show_219: bool = True
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def __set_commands(self, commands:dict[int, list[str]]) -> None:
|
def __set_commands(self, commands:dict[int, list[str]]) -> None:
|
||||||
@@ -91,7 +99,7 @@ class Command():
|
|||||||
)
|
)
|
||||||
'''
|
'''
|
||||||
|
|
||||||
self.Base.db_execute_query(table_logs)
|
# self.Base.db_execute_query(table_logs)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def __load_module_configuration(self) -> None:
|
def __load_module_configuration(self) -> None:
|
||||||
@@ -122,45 +130,81 @@ class Command():
|
|||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def add_defender_channel(self, channel:str) -> bool:
|
|
||||||
"""Cette fonction ajoute les salons de join de Defender
|
|
||||||
|
|
||||||
Args:
|
|
||||||
channel (str): le salon à enregistrer.
|
|
||||||
"""
|
|
||||||
mes_donnees = {'channel': channel}
|
|
||||||
response = self.Base.db_execute_query("SELECT id FROM def_channels WHERE channel = :channel", mes_donnees)
|
|
||||||
|
|
||||||
isChannelExist = response.fetchone()
|
|
||||||
|
|
||||||
if isChannelExist is None:
|
|
||||||
mes_donnees = {'datetime': self.Base.get_datetime(), 'channel': channel}
|
|
||||||
insert = self.Base.db_execute_query(f"INSERT INTO def_channels (datetime, channel) VALUES (:datetime, :channel)", mes_donnees)
|
|
||||||
if insert.rowcount >=0:
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
def delete_defender_channel(self, channel:str) -> bool:
|
|
||||||
"""Cette fonction supprime les salons de join de Defender
|
|
||||||
|
|
||||||
Args:
|
|
||||||
channel (str): le salon à enregistrer.
|
|
||||||
"""
|
|
||||||
mes_donnes = {'channel': channel}
|
|
||||||
response = self.Base.db_execute_query("DELETE FROM def_channels WHERE channel = :channel", mes_donnes)
|
|
||||||
|
|
||||||
affected_row = response.rowcount
|
|
||||||
|
|
||||||
if affected_row > 0:
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
def cmd(self, data:list) -> None:
|
def cmd(self, data:list) -> None:
|
||||||
|
|
||||||
|
service_id = self.Config.SERVICE_ID
|
||||||
|
dnickname = self.Config.SERVICE_NICKNAME
|
||||||
|
dchanlog = self.Config.SERVICE_CHANLOG
|
||||||
|
red = self.Config.COLORS.red
|
||||||
|
green = self.Config.COLORS.green
|
||||||
|
bold = self.Config.COLORS.bold
|
||||||
|
nogc = self.Config.COLORS.nogc
|
||||||
|
cmd = list(data).copy()
|
||||||
|
|
||||||
|
if len(cmd) < 2:
|
||||||
|
return None
|
||||||
|
|
||||||
|
match cmd[1]:
|
||||||
|
# [':irc.deb.biz.st', '403', 'Dev-PyDefender', '#Z', ':No', 'such', 'channel']
|
||||||
|
case '403' | '401':
|
||||||
|
try:
|
||||||
|
message = ' '.join(cmd[3:])
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} :[{red}ERROR MSG{nogc}] {message}")
|
||||||
|
self.Base.logs.error(f"{cmd[1]} - {message}")
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case '006' | '018':
|
||||||
|
try:
|
||||||
|
# [':irc.deb.biz.st', '006', 'Dev-PyDefender', ':`-services.deb.biz.st', '------', '|', 'Users:', '9', '(47.37%)', '[00B]']
|
||||||
|
# [':irc.deb.biz.st', '018', 'Dev-PyDefender', ':4', 'servers', 'and', '19', 'users,', 'average', '4.75', 'users', 'per', 'server']
|
||||||
|
message = ' '.join(cmd[3:])
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} : [{green}SERVER MSG{nogc}] {message}")
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case '219':
|
||||||
|
try:
|
||||||
|
# [':irc.deb.biz.st', '219', 'Dev-PyDefender', 's', ':End', 'of', '/STATS', 'report']
|
||||||
|
if not self.show_219:
|
||||||
|
# If there is a result in 223 then stop here
|
||||||
|
self.show_219 = True
|
||||||
|
return None
|
||||||
|
|
||||||
|
type_of_stats = str(cmd[3])
|
||||||
|
|
||||||
|
match type_of_stats:
|
||||||
|
case 's':
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} : No shun")
|
||||||
|
case 'G':
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} : No gline")
|
||||||
|
case 'k':
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} : No kline")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case '223':
|
||||||
|
try:
|
||||||
|
# [':irc.deb.biz.st', '223', 'Dev-PyDefender', 'G', '*@162.142.125.217', '67624', '18776', 'irc.deb.biz.st', ':Proxy/Drone', 'detected.', 'Check', 'https://dronebl.org/lookup?ip=162.142.125.217', 'for', 'details.']
|
||||||
|
self.show_219 = False
|
||||||
|
host = str(cmd[4])
|
||||||
|
author = str(cmd[7])
|
||||||
|
reason = ' '.join(cmd[8:])
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {self.user_to_notice} : {bold}Author{nogc}: {author} - {bold}Host{nogc}: {host} - {bold}Reason{nogc}: {reason}")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
||||||
@@ -346,6 +390,60 @@ class Command():
|
|||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'protect':
|
||||||
|
# /mode #channel +a user
|
||||||
|
# .protect #channel user
|
||||||
|
try:
|
||||||
|
if fromchannel is None:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
|
return False
|
||||||
|
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} +a {fromuser}")
|
||||||
|
return True
|
||||||
|
|
||||||
|
# deowner nickname
|
||||||
|
if len(cmd) == 2:
|
||||||
|
nickname = cmd[1]
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} +a {nickname}")
|
||||||
|
return True
|
||||||
|
|
||||||
|
nickname = cmd[2]
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} +a {nickname}")
|
||||||
|
|
||||||
|
except IndexError as e:
|
||||||
|
self.Logs.warning(f'_hcmd DEOWNER: {str(e)}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'deprotect':
|
||||||
|
# /mode #channel -a user
|
||||||
|
# .deprotect #channel user
|
||||||
|
try:
|
||||||
|
if fromchannel is None:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
|
return False
|
||||||
|
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} -a {fromuser}")
|
||||||
|
return True
|
||||||
|
|
||||||
|
# deowner nickname
|
||||||
|
if len(cmd) == 2:
|
||||||
|
nickname = cmd[1]
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} -a {nickname}")
|
||||||
|
return True
|
||||||
|
|
||||||
|
nickname = cmd[2]
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {fromchannel} -a {nickname}")
|
||||||
|
|
||||||
|
except IndexError as e:
|
||||||
|
self.Logs.warning(f'_hcmd DEOWNER: {str(e)}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
case 'halfop':
|
case 'halfop':
|
||||||
# /mode #channel +h user
|
# /mode #channel +h user
|
||||||
# .halfop #channel user
|
# .halfop #channel user
|
||||||
@@ -459,7 +557,7 @@ class Command():
|
|||||||
try:
|
try:
|
||||||
sentchannel = str(cmd[1]) if self.Base.Is_Channel(cmd[1]) else None
|
sentchannel = str(cmd[1]) if self.Base.Is_Channel(cmd[1]) else None
|
||||||
if sentchannel is None:
|
if sentchannel is None:
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} ban [#SALON] [NICKNAME]')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
return False
|
return False
|
||||||
|
|
||||||
nickname = cmd[2]
|
nickname = cmd[2]
|
||||||
@@ -468,7 +566,7 @@ class Command():
|
|||||||
self.Logs.debug(f'{fromuser} has banned {nickname} from {sentchannel}')
|
self.Logs.debug(f'{fromuser} has banned {nickname} from {sentchannel}')
|
||||||
except IndexError as e:
|
except IndexError as e:
|
||||||
self.Logs.warning(f'_hcmd BAN: {str(e)}')
|
self.Logs.warning(f'_hcmd BAN: {str(e)}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} ban [#SALON] [NICKNAME]')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#SALON] [NICKNAME]')
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
@@ -577,6 +675,174 @@ class Command():
|
|||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'topic':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} TOPIC #channel THE_TOPIC_MESSAGE")
|
||||||
|
return None
|
||||||
|
|
||||||
|
chan = str(cmd[1])
|
||||||
|
if not self.Base.Is_Channel(chan):
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :The channel must start with #")
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} TOPIC #channel THE_TOPIC_MESSAGE")
|
||||||
|
return None
|
||||||
|
|
||||||
|
topic_msg = ' '.join(cmd[2:]).strip()
|
||||||
|
|
||||||
|
if topic_msg:
|
||||||
|
self.Irc.send2socket(f':{dnickname} TOPIC {chan} :{topic_msg}')
|
||||||
|
else:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :You need to specify the topic")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'wallops':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} WALLOPS THE_WALLOPS_MESSAGE")
|
||||||
|
return None
|
||||||
|
|
||||||
|
wallops_msg = ' '.join(cmd[1:]).strip()
|
||||||
|
|
||||||
|
if wallops_msg:
|
||||||
|
self.Irc.send2socket(f':{dnickname} WALLOPS {wallops_msg} ({dnickname})')
|
||||||
|
else:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :You need to specify the wallops message")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'globops':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} GLOBOPS THE_GLOBOPS_MESSAGE")
|
||||||
|
return None
|
||||||
|
|
||||||
|
globops_msg = ' '.join(cmd[1:]).strip()
|
||||||
|
|
||||||
|
if globops_msg:
|
||||||
|
self.Irc.send2socket(f':{dnickname} GLOBOPS {globops_msg} ({dnickname})')
|
||||||
|
else:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :You need to specify the globops message")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'gnotice':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} THE_GLOBAL_NOTICE_MESSAGE")
|
||||||
|
return None
|
||||||
|
|
||||||
|
gnotice_msg = ' '.join(cmd[1:]).strip()
|
||||||
|
|
||||||
|
if gnotice_msg:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE $*.* :[{self.Config.COLORS.red}GLOBAL NOTICE{self.Config.COLORS.nogc}] {gnotice_msg}')
|
||||||
|
else:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :You need to specify the global notice message")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'whois':
|
||||||
|
try:
|
||||||
|
self.user_to_notice = fromuser
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} NICKNAME")
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
|
||||||
|
if self.User.get_nickname(nickname) is None:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :Nickname not found !")
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} NICKNAME")
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} WHOIS {nickname}')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'names':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 1:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} #CHANNEL")
|
||||||
|
return None
|
||||||
|
|
||||||
|
chan = str(cmd[1])
|
||||||
|
|
||||||
|
if not self.Base.Is_Channel(chan):
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :The channel must start with #")
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} #channel")
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} NAMES {chan}')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'invite':
|
||||||
|
try:
|
||||||
|
if len(cmd) < 3:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} #CHANNEL NICKNAME")
|
||||||
|
return None
|
||||||
|
|
||||||
|
chan = str(cmd[1])
|
||||||
|
nickname = str(cmd[2])
|
||||||
|
|
||||||
|
if not self.Base.Is_Channel(chan):
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :The channel must start with #")
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} #channel nickname")
|
||||||
|
return None
|
||||||
|
|
||||||
|
if self.User.get_nickname(nickname) is None:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :Nickname not found !")
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()} #channel NICKNAME")
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} INVITE {nickname} {chan}')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'inviteme':
|
||||||
|
try:
|
||||||
|
if len(cmd) == 0:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} :/msg {dnickname} {str(cmd[0]).upper()}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} INVITE {fromuser} {self.Config.SERVICE_CHANLOG}')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'map':
|
||||||
|
try:
|
||||||
|
self.user_to_notice = fromuser
|
||||||
|
self.Irc.send2socket(f':{dnickname} MAP')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
case 'umode':
|
case 'umode':
|
||||||
try:
|
try:
|
||||||
# .umode nickname +mode
|
# .umode nickname +mode
|
||||||
@@ -589,6 +855,39 @@ class Command():
|
|||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'mode':
|
||||||
|
# .mode #channel +/-mode
|
||||||
|
# .mode +/-mode
|
||||||
|
try:
|
||||||
|
|
||||||
|
if len(cmd) < 2:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#CHANNEL] [+/-]mode')
|
||||||
|
return None
|
||||||
|
|
||||||
|
if fromchannel is None:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#CHANNEL] [+/-]mode')
|
||||||
|
return None
|
||||||
|
|
||||||
|
if len(cmd) == 2:
|
||||||
|
channel_mode = cmd[1]
|
||||||
|
if self.Base.Is_Channel(fromchannel):
|
||||||
|
self.Irc.send2socket(f":{dnickname} MODE {fromchannel} {channel_mode}")
|
||||||
|
else:
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} : Right command : Channel [{fromchannel}] is not correct should start with #")
|
||||||
|
return None
|
||||||
|
|
||||||
|
if len(cmd) == 3:
|
||||||
|
provided_channel = cmd[1]
|
||||||
|
channel_mode = cmd[2]
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {provided_channel} {channel_mode}")
|
||||||
|
return None
|
||||||
|
|
||||||
|
except IndexError as e:
|
||||||
|
self.Logs.warning(f'_hcmd OP: {str(e)}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Right command : /msg {dnickname} {command.upper()} [#CHANNEL] [+/-]mode')
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
case 'svsjoin':
|
case 'svsjoin':
|
||||||
try:
|
try:
|
||||||
# .svsjoin nickname #channel
|
# .svsjoin nickname #channel
|
||||||
@@ -621,6 +920,38 @@ class Command():
|
|||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} SVSPART nickname #channel')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} SVSPART nickname #channel')
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'sajoin':
|
||||||
|
try:
|
||||||
|
# .sajoin nickname #channel
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
channel = str(cmd[2])
|
||||||
|
if len(cmd) < 3:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname #channel')
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{self.Config.SERVEUR_ID} SAJOIN {nickname} {channel}')
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname #channel')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'sapart':
|
||||||
|
try:
|
||||||
|
# .sapart nickname #channel
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
channel = str(cmd[2])
|
||||||
|
if len(cmd) < 3:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname #channel')
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{self.Config.SERVEUR_ID} SAPART {nickname} {channel}')
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname #channel')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
case 'svsnick':
|
case 'svsnick':
|
||||||
try:
|
try:
|
||||||
# .svsnick nickname newnickname
|
# .svsnick nickname newnickname
|
||||||
@@ -633,12 +964,203 @@ class Command():
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
if len(cmd) != 3:
|
if len(cmd) != 3:
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} SVSNICK nickname newnickname')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname newnickname')
|
||||||
return None
|
return None
|
||||||
|
|
||||||
self.Irc.send2socket(f':{self.Config.SERVEUR_ID} SVSNICK {nickname} {newnickname} {unixtime}')
|
self.Irc.send2socket(f':{self.Config.SERVEUR_ID} SVSNICK {nickname} {newnickname} {unixtime}')
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname newnickname')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'kill':
|
||||||
|
try:
|
||||||
|
# 'kill', 'gline', 'ungline', 'shun', 'unshun'
|
||||||
|
# .kill nickname reason
|
||||||
|
if len(cmd) < 3:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
kill_reason = ' '.join(cmd[2:])
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{service_id} KILL {nickname} {kill_reason} ({self.Config.COLORS.red}{dnickname}{self.Config.COLORS.nogc})")
|
||||||
except KeyError as ke:
|
except KeyError as ke:
|
||||||
self.Base.logs.error(ke)
|
self.Base.logs.error(ke)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} SVSNICK nickname newnickname')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} SVSNICK nickname newnickname')
|
||||||
self.Logs.warning(f'Unknown Error: {str(err)}')
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'gline':
|
||||||
|
try:
|
||||||
|
# TKL + G user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .gline [nickname] [host] [reason]
|
||||||
|
if len(cmd) < 4:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
set_at_timestamp = self.Base.get_unixtime()
|
||||||
|
expire_time = (60 * 60 * 24) + set_at_timestamp
|
||||||
|
gline_reason = ' '.join(cmd[3:])
|
||||||
|
|
||||||
|
if nickname == '*' and hostname == '*':
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : You want to close the server ? i would recommand ./unrealircd stop :)')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL + G {nickname} {hostname} {dnickname} {expire_time} {set_at_timestamp} :{gline_reason}")
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'ungline':
|
||||||
|
try:
|
||||||
|
# 'shun', 'unshun'
|
||||||
|
# TKL + G user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .ungline nickname host
|
||||||
|
if len(cmd) < 2:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL - G {nickname} {hostname} {dnickname}")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'kline':
|
||||||
|
try:
|
||||||
|
# TKL + k user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .gline [nickname] [host] [reason]
|
||||||
|
if len(cmd) < 4:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
set_at_timestamp = self.Base.get_unixtime()
|
||||||
|
expire_time = (60 * 60 * 24) + set_at_timestamp
|
||||||
|
gline_reason = ' '.join(cmd[3:])
|
||||||
|
|
||||||
|
if nickname == '*' and hostname == '*':
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : You want to close the server ? i would recommand ./unrealircd stop :)')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL + k {nickname} {hostname} {dnickname} {expire_time} {set_at_timestamp} :{gline_reason}")
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'unkline':
|
||||||
|
try:
|
||||||
|
# 'shun', 'unshun'
|
||||||
|
# TKL + G user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .ungline nickname host
|
||||||
|
if len(cmd) < 2:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL - k {nickname} {hostname} {dnickname}")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'shun':
|
||||||
|
try:
|
||||||
|
# TKL + G user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .shun [nickname] [host] [reason]
|
||||||
|
|
||||||
|
if len(cmd) < 4:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
set_at_timestamp = self.Base.get_unixtime()
|
||||||
|
expire_time = (60 * 60 * 24) + set_at_timestamp
|
||||||
|
shun_reason = ' '.join(cmd[3:])
|
||||||
|
|
||||||
|
if nickname == '*' and hostname == '*':
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : You want to close the server ? i would recommand ./unrealircd stop :)')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
return None
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL + s {nickname} {hostname} {dnickname} {expire_time} {set_at_timestamp} :{shun_reason}")
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname host reason')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'unshun':
|
||||||
|
try:
|
||||||
|
# 'shun', 'unshun'
|
||||||
|
# TKL + G user host set_by expire_timestamp set_at_timestamp :reason
|
||||||
|
# .unshun nickname host
|
||||||
|
if len(cmd) < 2:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
return None
|
||||||
|
|
||||||
|
nickname = str(cmd[1])
|
||||||
|
hostname = str(cmd[2])
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} TKL - s {nickname} {hostname} {dnickname}")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()} nickname hostname')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'glinelist':
|
||||||
|
try:
|
||||||
|
self.user_to_notice = fromuser
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_ID} STATS G")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()}')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'shunlist':
|
||||||
|
try:
|
||||||
|
self.user_to_notice = fromuser
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_ID} STATS s")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()}')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
|
|
||||||
|
case 'klinelist':
|
||||||
|
try:
|
||||||
|
self.user_to_notice = fromuser
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_ID} STATS k")
|
||||||
|
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(ke)
|
||||||
|
except Exception as err:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : /msg {dnickname} {command.upper()}')
|
||||||
|
self.Logs.warning(f'Unknown Error: {str(err)}')
|
||||||
@@ -1,7 +1,12 @@
|
|||||||
|
import socket
|
||||||
|
import json
|
||||||
|
import time
|
||||||
|
import re
|
||||||
|
import psutil
|
||||||
|
import requests
|
||||||
from dataclasses import dataclass, fields, field
|
from dataclasses import dataclass, fields, field
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from typing import Union
|
from typing import Union
|
||||||
import re, socket, psutil, requests, json, time
|
|
||||||
from sys import exit
|
from sys import exit
|
||||||
from core.irc import Irc
|
from core.irc import Irc
|
||||||
from core.Model import User
|
from core.Model import User
|
||||||
@@ -106,9 +111,6 @@ class Defender():
|
|||||||
self.__load_module_configuration()
|
self.__load_module_configuration()
|
||||||
# End of mandatory methods you can start your customization #
|
# End of mandatory methods you can start your customization #
|
||||||
|
|
||||||
# # Rejoindre les salons
|
|
||||||
# self.join_saved_channels()
|
|
||||||
|
|
||||||
self.timeout = self.Config.API_TIMEOUT
|
self.timeout = self.Config.API_TIMEOUT
|
||||||
|
|
||||||
# Listes qui vont contenir les ip a scanner avec les différentes API
|
# Listes qui vont contenir les ip a scanner avec les différentes API
|
||||||
@@ -145,7 +147,8 @@ class Defender():
|
|||||||
self.Base.create_thread(func=self.thread_reputation_timer)
|
self.Base.create_thread(func=self.thread_reputation_timer)
|
||||||
|
|
||||||
if self.ModConfig.reputation == 1:
|
if self.ModConfig.reputation == 1:
|
||||||
self.Irc.send2socket(f":{self.Config.SERVICE_ID} SAMODE {self.Config.SALON_JAIL} +{self.Config.SERVICE_UMODES} {self.Config.SERVICE_NICKNAME}")
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} SJOIN {self.Base.get_unixtime()} {self.Config.SALON_JAIL} + :{self.Config.SERVICE_NICKNAME}")
|
||||||
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} SAMODE {self.Config.SALON_JAIL} +o {self.Config.SERVICE_NICKNAME}")
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@@ -244,40 +247,6 @@ class Defender():
|
|||||||
self.reputationTimer_isRunning:bool = False
|
self.reputationTimer_isRunning:bool = False
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def add_defender_channel(self, channel:str) -> bool:
|
|
||||||
"""Cette fonction ajoute les salons de join de Defender
|
|
||||||
|
|
||||||
Args:
|
|
||||||
channel (str): le salon à enregistrer.
|
|
||||||
"""
|
|
||||||
mes_donnees = {'channel': channel}
|
|
||||||
response = self.Base.db_execute_query("SELECT id FROM def_channels WHERE channel = :channel", mes_donnees)
|
|
||||||
|
|
||||||
isChannelExist = response.fetchone()
|
|
||||||
|
|
||||||
if isChannelExist is None:
|
|
||||||
mes_donnees = {'datetime': self.Base.get_datetime(), 'channel': channel}
|
|
||||||
insert = self.Base.db_execute_query(f"INSERT INTO def_channels (datetime, channel) VALUES (:datetime, :channel)", mes_donnees)
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
def delete_defender_channel(self, channel:str) -> bool:
|
|
||||||
"""Cette fonction supprime les salons de join de Defender
|
|
||||||
|
|
||||||
Args:
|
|
||||||
channel (str): le salon à enregistrer.
|
|
||||||
"""
|
|
||||||
mes_donnes = {'channel': channel}
|
|
||||||
response = self.Base.db_execute_query("DELETE FROM def_channels WHERE channel = :channel", mes_donnes)
|
|
||||||
|
|
||||||
affected_row = response.rowcount
|
|
||||||
|
|
||||||
if affected_row > 0:
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
return False
|
|
||||||
|
|
||||||
def reputation_insert(self, reputationModel: ReputationModel) -> bool:
|
def reputation_insert(self, reputationModel: ReputationModel) -> bool:
|
||||||
|
|
||||||
response = False
|
response = False
|
||||||
@@ -367,7 +336,7 @@ class Defender():
|
|||||||
|
|
||||||
def join_saved_channels(self) -> None:
|
def join_saved_channels(self) -> None:
|
||||||
|
|
||||||
result = self.Base.db_execute_query("SELECT id, channel FROM def_channels")
|
result = self.Base.db_execute_query(f"SELECT distinct channel_name FROM {self.Config.table_channel}")
|
||||||
channels = result.fetchall()
|
channels = result.fetchall()
|
||||||
jail_chan = self.Config.SALON_JAIL
|
jail_chan = self.Config.SALON_JAIL
|
||||||
jail_chan_mode = self.Config.SALON_JAIL_MODES
|
jail_chan_mode = self.Config.SALON_JAIL_MODES
|
||||||
@@ -378,7 +347,7 @@ class Defender():
|
|||||||
unixtime = self.Base.get_unixtime()
|
unixtime = self.Base.get_unixtime()
|
||||||
|
|
||||||
for channel in channels:
|
for channel in channels:
|
||||||
id, chan = channel
|
chan = channel[0]
|
||||||
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} SJOIN {unixtime} {chan} + :{self.Config.SERVICE_ID}")
|
self.Irc.send2socket(f":{self.Config.SERVEUR_ID} SJOIN {unixtime} {chan} + :{self.Config.SERVICE_ID}")
|
||||||
if chan == jail_chan:
|
if chan == jail_chan:
|
||||||
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} +{dumodes} {dnickname}")
|
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} +{dumodes} {dnickname}")
|
||||||
@@ -441,9 +410,9 @@ class Defender():
|
|||||||
jailed_nickname = get_reputation.nickname
|
jailed_nickname = get_reputation.nickname
|
||||||
jailed_score = get_reputation.score
|
jailed_score = get_reputation.score
|
||||||
|
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
color_bold = self.Config.CONFIG_COLOR['gras']
|
color_bold = self.Config.COLORS.bold
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
service_prefix = self.Config.SERVICE_PREFIX
|
service_prefix = self.Config.SERVICE_PREFIX
|
||||||
reputation_ban_all_chan = self.ModConfig.reputation_ban_all_chan
|
reputation_ban_all_chan = self.ModConfig.reputation_ban_all_chan
|
||||||
@@ -478,8 +447,8 @@ class Defender():
|
|||||||
ban_all_chan = self.ModConfig.reputation_ban_all_chan
|
ban_all_chan = self.ModConfig.reputation_ban_all_chan
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
dchanlog = self.Config.SERVICE_CHANLOG
|
dchanlog = self.Config.SERVICE_CHANLOG
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
salon_jail = self.Config.SALON_JAIL
|
salon_jail = self.Config.SALON_JAIL
|
||||||
|
|
||||||
if reputation_flag == 0:
|
if reputation_flag == 0:
|
||||||
@@ -559,8 +528,8 @@ class Defender():
|
|||||||
flood_timer = self.ModConfig.flood_timer
|
flood_timer = self.ModConfig.flood_timer
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
dnickname = self.Config.SERVICE_NICKNAME
|
dnickname = self.Config.SERVICE_NICKNAME
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_bold = self.Config.CONFIG_COLOR['gras']
|
color_bold = self.Config.COLORS.bold
|
||||||
|
|
||||||
get_detected_uid = self.User.get_uid(detected_user)
|
get_detected_uid = self.User.get_uid(detected_user)
|
||||||
get_detected_nickname = self.User.get_nickname(detected_user)
|
get_detected_nickname = self.User.get_nickname(detected_user)
|
||||||
@@ -634,7 +603,7 @@ class Defender():
|
|||||||
connection = (remote_ip, self.Base.int_if_possible(port))
|
connection = (remote_ip, self.Base.int_if_possible(port))
|
||||||
newSocket.connect(connection)
|
newSocket.connect(connection)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.CONFIG_COLOR['rouge']}PROXY_SCAN{self.Config.CONFIG_COLOR['noire']} ] {fullname} ({remote_ip}) : Port [{str(port)}] ouvert sur l'adresse ip [{remote_ip}]")
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.COLORS.red}PROXY_SCAN{self.Config.COLORS.black} ] {fullname} ({remote_ip}) : Port [{str(port)}] ouvert sur l'adresse ip [{remote_ip}]")
|
||||||
# print(f"=======> Le port {str(port)} est ouvert !!")
|
# print(f"=======> Le port {str(port)} est ouvert !!")
|
||||||
self.Base.running_sockets.append(newSocket)
|
self.Base.running_sockets.append(newSocket)
|
||||||
# print(newSocket)
|
# print(newSocket)
|
||||||
@@ -697,7 +666,7 @@ class Defender():
|
|||||||
self.Logs.info(f"Connexion of {fullname} ({remote_ip}) using ports : {str(matching_ports)}")
|
self.Logs.info(f"Connexion of {fullname} ({remote_ip}) using ports : {str(matching_ports)}")
|
||||||
|
|
||||||
if matching_ports:
|
if matching_ports:
|
||||||
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.CONFIG_COLOR['rouge']}PSUTIL_SCAN{self.Config.CONFIG_COLOR['noire']} ] {fullname} ({remote_ip}) : is using ports {matching_ports}")
|
self.Irc.send2socket(f":{self.Config.SERVICE_NICKNAME} PRIVMSG {self.Config.SERVICE_CHANLOG} :[ {self.Config.COLORS.red}PSUTIL_SCAN{self.Config.COLORS.black} ] {fullname} ({remote_ip}) : is using ports {matching_ports}")
|
||||||
|
|
||||||
return matching_ports
|
return matching_ports
|
||||||
|
|
||||||
@@ -777,8 +746,8 @@ class Defender():
|
|||||||
|
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
service_chanlog = self.Config.SERVICE_CHANLOG
|
service_chanlog = self.Config.SERVICE_CHANLOG
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
|
|
||||||
# pseudo!ident@host
|
# pseudo!ident@host
|
||||||
fullname = f'{nickname}!{username}@{hostname}'
|
fullname = f'{nickname}!{username}@{hostname}'
|
||||||
@@ -845,8 +814,8 @@ class Defender():
|
|||||||
|
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
service_chanlog = self.Config.SERVICE_CHANLOG
|
service_chanlog = self.Config.SERVICE_CHANLOG
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
|
|
||||||
url = f'https://freeipapi.com/api/json/{remote_ip}'
|
url = f'https://freeipapi.com/api/json/{remote_ip}'
|
||||||
|
|
||||||
@@ -933,8 +902,8 @@ class Defender():
|
|||||||
|
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
service_chanlog = self.Config.SERVICE_CHANLOG
|
service_chanlog = self.Config.SERVICE_CHANLOG
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
|
|
||||||
url = f"https://developers18334.cloudfilt.com/"
|
url = f"https://developers18334.cloudfilt.com/"
|
||||||
|
|
||||||
@@ -995,190 +964,206 @@ class Defender():
|
|||||||
self.Logs.error(f"Thread_cloudfilt_scan Error : {ve}")
|
self.Logs.error(f"Thread_cloudfilt_scan Error : {ve}")
|
||||||
|
|
||||||
def cmd(self, data: list) -> None:
|
def cmd(self, data: list) -> None:
|
||||||
|
try:
|
||||||
|
service_id = self.Config.SERVICE_ID # Defender serveur id
|
||||||
|
cmd = list(data).copy()
|
||||||
|
|
||||||
service_id = self.Config.SERVICE_ID # Defender serveur id
|
match cmd[1]:
|
||||||
cmd = list(data).copy()
|
|
||||||
|
|
||||||
if len(cmd) < 2:
|
case 'REPUTATION':
|
||||||
return None
|
# :001 REPUTATION 91.168.141.239 118
|
||||||
|
try:
|
||||||
|
self.reputation_first_connexion['ip'] = cmd[2]
|
||||||
|
self.reputation_first_connexion['score'] = cmd[3]
|
||||||
|
if str(cmd[3]).find('*') != -1:
|
||||||
|
# If the reputation changed, we do not need to scan the IP
|
||||||
|
return None
|
||||||
|
|
||||||
match cmd[1]:
|
if not self.Base.is_valid_ip(cmd[2]):
|
||||||
|
return None
|
||||||
|
|
||||||
case 'EOS':
|
# Possibilité de déclancher les bans a ce niveau.
|
||||||
if self.Irc.INIT == 0:
|
except IndexError as ie:
|
||||||
self.Irc.send2socket(f":{service_id} SAMODE {self.Config.SALON_JAIL} +{self.Config.SERVICE_UMODES} {self.Config.SERVICE_NICKNAME}")
|
self.Logs.error(f'cmd reputation: index error: {ie}')
|
||||||
|
|
||||||
case 'REPUTATION':
|
if len(cmd) < 3:
|
||||||
# :001 REPUTATION 91.168.141.239 118
|
return None
|
||||||
try:
|
|
||||||
self.reputation_first_connexion['ip'] = cmd[2]
|
match cmd[2]:
|
||||||
self.reputation_first_connexion['score'] = cmd[3]
|
|
||||||
if str(cmd[3]).find('*') != -1:
|
case 'MODE':
|
||||||
# If the reputation changed, we do not need to scan the IP
|
# ['...', ':001XSCU0Q', 'MODE', '#jail', '+b', '~security-group:unknown-users']
|
||||||
|
channel = str(cmd[3])
|
||||||
|
mode = str(cmd[4])
|
||||||
|
group_to_check = str(cmd[5:])
|
||||||
|
group_to_unban = '~security-group:unknown-users'
|
||||||
|
|
||||||
|
if self.Config.SALON_JAIL == channel:
|
||||||
|
if mode == '+b' and group_to_unban in group_to_check:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {self.Config.SALON_JAIL} -b ~security-group:unknown-users")
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {self.Config.SALON_JAIL} -eee ~security-group:webirc-users ~security-group:known-users ~security-group:websocket-users")
|
||||||
|
|
||||||
|
case 'PRIVMSG':
|
||||||
|
cmd.pop(0)
|
||||||
|
user_trigger = str(cmd[0]).replace(':','')
|
||||||
|
channel = cmd[2]
|
||||||
|
find_nickname = self.User.get_nickname(user_trigger)
|
||||||
|
self.flood(find_nickname, channel)
|
||||||
|
|
||||||
|
case 'UID':
|
||||||
|
# If Init then do nothing
|
||||||
|
if self.Irc.INIT == 1:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
if not self.Base.is_valid_ip(cmd[2]):
|
# Supprimer la premiere valeur et finir le code normalement
|
||||||
return None
|
cmd.pop(0)
|
||||||
|
|
||||||
# Possibilité de déclancher les bans a ce niveau.
|
# Get User information
|
||||||
except IndexError as ie:
|
_User = self.User.get_User(str(cmd[7]))
|
||||||
self.Logs.error(f'cmd reputation: index error: {ie}')
|
|
||||||
|
|
||||||
match cmd[2]:
|
# If user is not service or IrcOp then scan them
|
||||||
|
|
||||||
case 'PRIVMSG':
|
|
||||||
cmd.pop(0)
|
|
||||||
user_trigger = str(cmd[0]).replace(':','')
|
|
||||||
channel = cmd[2]
|
|
||||||
find_nickname = self.User.get_nickname(user_trigger)
|
|
||||||
self.flood(find_nickname, channel)
|
|
||||||
|
|
||||||
case 'UID':
|
|
||||||
# If Init then do nothing
|
|
||||||
if self.Irc.INIT == 1:
|
|
||||||
return None
|
|
||||||
|
|
||||||
# Supprimer la premiere valeur et finir le code normalement
|
|
||||||
cmd.pop(0)
|
|
||||||
|
|
||||||
# Get User information
|
|
||||||
_User = self.User.get_User(str(cmd[7]))
|
|
||||||
|
|
||||||
# If user is not service or IrcOp then scan them
|
|
||||||
if not re.match(fr'^.*[S|o?].*$', _User.umodes):
|
|
||||||
self.abuseipdb_UserModel.append(_User) if self.ModConfig.abuseipdb_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
|
||||||
self.freeipapi_UserModel.append(_User) if self.ModConfig.freeipapi_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
|
||||||
self.cloudfilt_UserModel.append(_User) if self.ModConfig.cloudfilt_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
|
||||||
self.psutil_UserModel.append(_User) if self.ModConfig.psutil_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
|
||||||
self.localscan_UserModel.append(_User) if self.ModConfig.local_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
|
||||||
|
|
||||||
if _User is None:
|
|
||||||
self.Logs.critical(f'This UID: [{cmd[7]}] is not available please check why')
|
|
||||||
return None
|
|
||||||
|
|
||||||
reputation_flag = self.ModConfig.reputation
|
|
||||||
reputation_seuil = self.ModConfig.reputation_seuil
|
|
||||||
|
|
||||||
if self.Irc.INIT == 0:
|
|
||||||
# Si le user n'es pas un service ni un IrcOP
|
|
||||||
if not re.match(fr'^.*[S|o?].*$', _User.umodes):
|
if not re.match(fr'^.*[S|o?].*$', _User.umodes):
|
||||||
if reputation_flag == 1 and _User.score_connexion <= reputation_seuil:
|
self.abuseipdb_UserModel.append(_User) if self.ModConfig.abuseipdb_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
||||||
currentDateTime = self.Base.get_datetime()
|
self.freeipapi_UserModel.append(_User) if self.ModConfig.freeipapi_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
||||||
self.reputation_insert(
|
self.cloudfilt_UserModel.append(_User) if self.ModConfig.cloudfilt_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
||||||
self.ReputationModel(
|
self.psutil_UserModel.append(_User) if self.ModConfig.psutil_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
||||||
uid=_User.uid, nickname=_User.nickname, username=_User.username, realname=_User.realname,
|
self.localscan_UserModel.append(_User) if self.ModConfig.local_scan == 1 and not _User.remote_ip in self.Config.WHITELISTED_IP else None
|
||||||
hostname=_User.hostname, umodes=_User.umodes, vhost=_User.vhost, ip=_User.remote_ip, score=_User.score_connexion,
|
|
||||||
secret_code=self.Base.get_random(8), isWebirc=_User.isWebirc, isWebsocket=_User.isWebsocket, connected_datetime=currentDateTime,
|
if _User is None:
|
||||||
updated_datetime=currentDateTime
|
self.Logs.critical(f'This UID: [{cmd[7]}] is not available please check why')
|
||||||
|
return None
|
||||||
|
|
||||||
|
reputation_flag = self.ModConfig.reputation
|
||||||
|
reputation_seuil = self.ModConfig.reputation_seuil
|
||||||
|
|
||||||
|
if self.Irc.INIT == 0:
|
||||||
|
# Si le user n'es pas un service ni un IrcOP
|
||||||
|
if not re.match(fr'^.*[S|o?].*$', _User.umodes):
|
||||||
|
if reputation_flag == 1 and _User.score_connexion <= reputation_seuil:
|
||||||
|
currentDateTime = self.Base.get_datetime()
|
||||||
|
self.reputation_insert(
|
||||||
|
self.ReputationModel(
|
||||||
|
uid=_User.uid, nickname=_User.nickname, username=_User.username, realname=_User.realname,
|
||||||
|
hostname=_User.hostname, umodes=_User.umodes, vhost=_User.vhost, ip=_User.remote_ip, score=_User.score_connexion,
|
||||||
|
secret_code=self.Base.get_random(8), isWebirc=_User.isWebirc, isWebsocket=_User.isWebsocket, connected_datetime=currentDateTime,
|
||||||
|
updated_datetime=currentDateTime
|
||||||
|
)
|
||||||
)
|
)
|
||||||
)
|
# self.Irc.send2socket(f":{service_id} WHOIS {nickname}")
|
||||||
# self.Irc.send2socket(f":{service_id} WHOIS {nickname}")
|
if self.reputation_check(_User.uid):
|
||||||
if self.reputation_check(_User.uid):
|
if reputation_flag == 1 and _User.score_connexion <= reputation_seuil:
|
||||||
if reputation_flag == 1 and _User.score_connexion <= reputation_seuil:
|
self.system_reputation(_User.uid)
|
||||||
self.system_reputation(_User.uid)
|
self.Logs.info('Démarrer le systeme de reputation')
|
||||||
self.Logs.info('Démarrer le systeme de reputation')
|
|
||||||
|
|
||||||
case 'SJOIN':
|
case 'SJOIN':
|
||||||
# ['@msgid=F9B7JeHL5pj9nN57cJ5pEr;time=2023-12-28T20:47:24.305Z', ':001', 'SJOIN', '1702138958', '#welcome', ':0015L1AHL']
|
# ['@msgid=F9B7JeHL5pj9nN57cJ5pEr;time=2023-12-28T20:47:24.305Z', ':001', 'SJOIN', '1702138958', '#welcome', ':0015L1AHL']
|
||||||
try:
|
try:
|
||||||
|
cmd.pop(0)
|
||||||
|
parsed_chan = cmd[3]
|
||||||
|
|
||||||
|
if self.ModConfig.reputation == 1:
|
||||||
|
parsed_UID = cmd[4]
|
||||||
|
pattern = fr'^:[@|%|\+|~|\*]*'
|
||||||
|
parsed_UID = re.sub(pattern, '', parsed_UID)
|
||||||
|
|
||||||
|
get_reputation = self.reputation_get_Reputation(parsed_UID)
|
||||||
|
|
||||||
|
if parsed_chan != self.Config.SALON_JAIL:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +b ~security-group:unknown-users")
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +eee ~security-group:webirc-users ~security-group:known-users ~security-group:websocket-users")
|
||||||
|
|
||||||
|
if not get_reputation is None:
|
||||||
|
isWebirc = get_reputation.isWebirc
|
||||||
|
|
||||||
|
if not isWebirc:
|
||||||
|
if parsed_chan != self.Config.SALON_JAIL:
|
||||||
|
self.Irc.send2socket(f":{service_id} SAPART {get_reputation.nickname} {parsed_chan}")
|
||||||
|
|
||||||
|
if self.ModConfig.reputation_ban_all_chan == 1 and not isWebirc:
|
||||||
|
if parsed_chan != self.Config.SALON_JAIL:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +b {get_reputation.nickname}!*@*")
|
||||||
|
self.Irc.send2socket(f":{service_id} KICK {parsed_chan} {get_reputation.nickname}")
|
||||||
|
|
||||||
|
self.Logs.debug(f'SJOIN parsed_uid : {parsed_UID}')
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Logs.error(f"key error SJOIN : {ke}")
|
||||||
|
|
||||||
|
case 'SLOG':
|
||||||
|
# self.Base.scan_ports(cmd[7])
|
||||||
cmd.pop(0)
|
cmd.pop(0)
|
||||||
parsed_chan = cmd[3]
|
|
||||||
|
|
||||||
if self.ModConfig.reputation == 1:
|
if not self.Base.is_valid_ip(cmd[7]):
|
||||||
parsed_UID = cmd[4]
|
return None
|
||||||
pattern = fr'^:[@|%|\+|~|\*]*'
|
|
||||||
parsed_UID = re.sub(pattern, '', parsed_UID)
|
|
||||||
|
|
||||||
get_reputation = self.reputation_get_Reputation(parsed_UID)
|
# if self.ModConfig.local_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
||||||
|
# self.localscan_remote_ip.append(cmd[7])
|
||||||
|
|
||||||
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +b ~security-group:unknown-users")
|
# if self.ModConfig.psutil_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
||||||
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +eee ~security-group:webirc-users ~security-group:known-users ~security-group:websocket-users")
|
# self.psutil_remote_ip.append(cmd[7])
|
||||||
|
|
||||||
if not get_reputation is None:
|
# if self.ModConfig.abuseipdb_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
||||||
isWebirc = get_reputation.isWebirc
|
# self.abuseipdb_remote_ip.append(cmd[7])
|
||||||
|
|
||||||
if not isWebirc:
|
# if self.ModConfig.freeipapi_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
||||||
if parsed_chan != self.Config.SALON_JAIL:
|
# self.freeipapi_remote_ip.append(cmd[7])
|
||||||
self.Irc.send2socket(f":{service_id} SAPART {get_reputation.nickname} {parsed_chan}")
|
|
||||||
|
|
||||||
if self.ModConfig.reputation_ban_all_chan == 1 and not isWebirc:
|
# if self.ModConfig.cloudfilt_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
||||||
if parsed_chan != self.Config.SALON_JAIL:
|
# self.cloudfilt_remote_ip.append(cmd[7])
|
||||||
self.Irc.send2socket(f":{service_id} MODE {parsed_chan} +b {get_reputation.nickname}!*@*")
|
|
||||||
self.Irc.send2socket(f":{service_id} KICK {parsed_chan} {get_reputation.nickname}")
|
|
||||||
|
|
||||||
self.Logs.debug(f'SJOIN parsed_uid : {parsed_UID}')
|
case 'NICK':
|
||||||
except KeyError as ke:
|
# :0010BS24L NICK [NEWNICK] 1697917711
|
||||||
self.Logs.error(f"key error SJOIN : {ke}")
|
# Changement de nickname
|
||||||
|
try:
|
||||||
|
cmd.pop(0)
|
||||||
|
uid = str(cmd[0]).replace(':','')
|
||||||
|
get_Reputation = self.reputation_get_Reputation(uid)
|
||||||
|
jail_salon = self.Config.SALON_JAIL
|
||||||
|
service_id = self.Config.SERVICE_ID
|
||||||
|
|
||||||
case 'SLOG':
|
if get_Reputation is None:
|
||||||
# self.Base.scan_ports(cmd[7])
|
self.Logs.debug(f'This UID: {uid} is not listed in the reputation dataclass')
|
||||||
cmd.pop(0)
|
return None
|
||||||
|
|
||||||
if not self.Base.is_valid_ip(cmd[7]):
|
# Update the new nickname
|
||||||
return None
|
oldnick = get_Reputation.nickname
|
||||||
|
newnickname = cmd[2]
|
||||||
|
get_Reputation.nickname = newnickname
|
||||||
|
|
||||||
# if self.ModConfig.local_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
# If ban in all channel is ON then unban old nickname an ban the new nickname
|
||||||
# self.localscan_remote_ip.append(cmd[7])
|
if self.ModConfig.reputation_ban_all_chan == 1:
|
||||||
|
for chan in self.Channel.UID_CHANNEL_DB:
|
||||||
|
if chan.name != jail_salon:
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b {oldnick}!*@*")
|
||||||
|
self.Irc.send2socket(f":{service_id} MODE {chan.name} +b {newnickname}!*@*")
|
||||||
|
|
||||||
# if self.ModConfig.psutil_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
except KeyError as ke:
|
||||||
# self.psutil_remote_ip.append(cmd[7])
|
self.Logs.error(f'cmd - NICK - KeyError: {ke}')
|
||||||
|
|
||||||
# if self.ModConfig.abuseipdb_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
case 'QUIT':
|
||||||
# self.abuseipdb_remote_ip.append(cmd[7])
|
# :001N1WD7L QUIT :Quit: free_znc_1
|
||||||
|
|
||||||
# if self.ModConfig.freeipapi_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
|
||||||
# self.freeipapi_remote_ip.append(cmd[7])
|
|
||||||
|
|
||||||
# if self.ModConfig.cloudfilt_scan == 1 and not cmd[7] in self.Config.WHITELISTED_IP:
|
|
||||||
# self.cloudfilt_remote_ip.append(cmd[7])
|
|
||||||
|
|
||||||
case 'NICK':
|
|
||||||
# :0010BS24L NICK [NEWNICK] 1697917711
|
|
||||||
# Changement de nickname
|
|
||||||
try:
|
|
||||||
cmd.pop(0)
|
cmd.pop(0)
|
||||||
uid = str(cmd[0]).replace(':','')
|
ban_all_chan = self.Base.int_if_possible(self.ModConfig.reputation_ban_all_chan)
|
||||||
get_Reputation = self.reputation_get_Reputation(uid)
|
user_id = str(cmd[0]).replace(':','')
|
||||||
|
final_UID = user_id
|
||||||
|
|
||||||
jail_salon = self.Config.SALON_JAIL
|
jail_salon = self.Config.SALON_JAIL
|
||||||
service_id = self.Config.SERVICE_ID
|
service_id = self.Config.SERVICE_ID
|
||||||
|
|
||||||
if get_Reputation is None:
|
get_user_reputation = self.reputation_get_Reputation(final_UID)
|
||||||
self.Logs.debug(f'This UID: {uid} is not listed in the reputation dataclass')
|
|
||||||
return None
|
|
||||||
|
|
||||||
# Update the new nickname
|
if not get_user_reputation is None:
|
||||||
oldnick = get_Reputation.nickname
|
final_nickname = get_user_reputation.nickname
|
||||||
newnickname = cmd[2]
|
|
||||||
get_Reputation.nickname = newnickname
|
|
||||||
|
|
||||||
# If ban in all channel is ON then unban old nickname an ban the new nickname
|
|
||||||
if self.ModConfig.reputation_ban_all_chan == 1:
|
|
||||||
for chan in self.Channel.UID_CHANNEL_DB:
|
for chan in self.Channel.UID_CHANNEL_DB:
|
||||||
if chan.name != jail_salon:
|
if chan.name != jail_salon and ban_all_chan == 1:
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b {oldnick}!*@*")
|
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b {final_nickname}!*@*")
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} +b {newnickname}!*@*")
|
self.reputation_delete(final_UID)
|
||||||
|
|
||||||
except KeyError as ke:
|
except KeyError as ke:
|
||||||
self.Logs.error(f'cmd - NICK - KeyError: {ke}')
|
self.Logs.error(f"{ke} / {cmd} / length {str(len(cmd))}")
|
||||||
|
except IndexError as ie:
|
||||||
case 'QUIT':
|
self.Logs.error(f"{ie} / {cmd} / length {str(len(cmd))}")
|
||||||
# :001N1WD7L QUIT :Quit: free_znc_1
|
except Exception as err:
|
||||||
cmd.pop(0)
|
self.Logs.error(f"General Error: {err}")
|
||||||
ban_all_chan = self.Base.int_if_possible(self.ModConfig.reputation_ban_all_chan)
|
|
||||||
user_id = str(cmd[0]).replace(':','')
|
|
||||||
final_UID = user_id
|
|
||||||
|
|
||||||
jail_salon = self.Config.SALON_JAIL
|
|
||||||
service_id = self.Config.SERVICE_ID
|
|
||||||
|
|
||||||
get_user_reputation = self.reputation_get_Reputation(final_UID)
|
|
||||||
|
|
||||||
if not get_user_reputation is None:
|
|
||||||
final_nickname = get_user_reputation.nickname
|
|
||||||
for chan in self.Channel.UID_CHANNEL_DB:
|
|
||||||
if chan.name != jail_salon and ban_all_chan == 1:
|
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b {final_nickname}!*@*")
|
|
||||||
self.reputation_delete(final_UID)
|
|
||||||
|
|
||||||
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
||||||
|
|
||||||
@@ -1233,8 +1218,8 @@ class Defender():
|
|||||||
|
|
||||||
self.Logs.debug(f"IP de {jailed_nickname} : {jailed_IP}")
|
self.Logs.debug(f"IP de {jailed_nickname} : {jailed_IP}")
|
||||||
link = self.Config.SERVEUR_LINK
|
link = self.Config.SERVEUR_LINK
|
||||||
color_green = self.Config.CONFIG_COLOR['verte']
|
color_green = self.Config.COLORS.green
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
|
|
||||||
if release_code == get_reputation.secret_code:
|
if release_code == get_reputation.secret_code:
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {jailed_salon} : Bon mot de passe. Allez du vent !')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {jailed_salon} : Bon mot de passe. Allez du vent !')
|
||||||
@@ -1278,13 +1263,13 @@ class Defender():
|
|||||||
if activation == 'on':
|
if activation == 'on':
|
||||||
|
|
||||||
if self.ModConfig.reputation == 1:
|
if self.ModConfig.reputation == 1:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}REPUTATION{self.Config.CONFIG_COLOR['noire']} ] : Already activated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION{self.Config.COLORS.black} ] : Already activated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# self.update_db_configuration('reputation', 1)
|
# self.update_db_configuration('reputation', 1)
|
||||||
self.__update_configuration(key, 1)
|
self.__update_configuration(key, 1)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}REPUTATION{self.Config.CONFIG_COLOR['noire']} ] : Activated by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION{self.Config.COLORS.black} ] : Activated by {fromuser}")
|
||||||
self.Irc.send2socket(f":{service_id} JOIN {jail_chan}")
|
self.Irc.send2socket(f":{service_id} JOIN {jail_chan}")
|
||||||
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} +{dumodes} {dnickname}")
|
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} +{dumodes} {dnickname}")
|
||||||
self.Irc.send2socket(f":{service_id} MODE {jail_chan} +{jail_chan_mode}")
|
self.Irc.send2socket(f":{service_id} MODE {jail_chan} +{jail_chan_mode}")
|
||||||
@@ -1300,12 +1285,12 @@ class Defender():
|
|||||||
if activation == 'off':
|
if activation == 'off':
|
||||||
|
|
||||||
if self.ModConfig.reputation == 0:
|
if self.ModConfig.reputation == 0:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}REPUTATION{self.Config.CONFIG_COLOR['noire']} ] : Already deactivated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION{self.Config.COLORS.black} ] : Already deactivated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.__update_configuration(key, 0)
|
self.__update_configuration(key, 0)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}REPUTATION{self.Config.CONFIG_COLOR['noire']} ] : Deactivated by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}REPUTATION{self.Config.COLORS.black} ] : Deactivated by {fromuser}")
|
||||||
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} -{dumodes} {dnickname}")
|
self.Irc.send2socket(f":{service_id} SAMODE {jail_chan} -{dumodes} {dnickname}")
|
||||||
self.Irc.send2socket(f":{service_id} MODE {jail_chan} -sS")
|
self.Irc.send2socket(f":{service_id} MODE {jail_chan} -sS")
|
||||||
self.Irc.send2socket(f":{service_id} PART {jail_chan}")
|
self.Irc.send2socket(f":{service_id} PART {jail_chan}")
|
||||||
@@ -1313,9 +1298,7 @@ class Defender():
|
|||||||
for chan in self.Channel.UID_CHANNEL_DB:
|
for chan in self.Channel.UID_CHANNEL_DB:
|
||||||
if chan.name != jail_chan:
|
if chan.name != jail_chan:
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b ~security-group:unknown-users")
|
self.Irc.send2socket(f":{service_id} MODE {chan.name} -b ~security-group:unknown-users")
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -e ~security-group:webirc-users")
|
self.Irc.send2socket(f":{service_id} MODE {chan.name} -eee ~security-group:webirc-users ~security-group:known-users ~security-group:websocket-users")
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -e ~security-group:known-users")
|
|
||||||
self.Irc.send2socket(f":{service_id} MODE {chan.name} -e ~security-group:websocket-users")
|
|
||||||
|
|
||||||
self.Base.db_query_channel('del', self.module_name, jail_chan)
|
self.Base.db_query_channel('del', self.module_name, jail_chan)
|
||||||
|
|
||||||
@@ -1335,23 +1318,23 @@ class Defender():
|
|||||||
if get_value == 'on':
|
if get_value == 'on':
|
||||||
|
|
||||||
if self.ModConfig.reputation_ban_all_chan == 1:
|
if self.ModConfig.reputation_ban_all_chan == 1:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}BAN ON ALL CHANS{self.Config.CONFIG_COLOR['noire']} ] : Already activated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}BAN ON ALL CHANS{self.Config.COLORS.black} ] : Already activated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# self.update_db_configuration(key, 1)
|
# self.update_db_configuration(key, 1)
|
||||||
self.__update_configuration(key, 1)
|
self.__update_configuration(key, 1)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}BAN ON ALL CHANS{self.Config.CONFIG_COLOR["noire"]} ] : Activated by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}BAN ON ALL CHANS{self.Config.COLORS.black} ] : Activated by {fromuser}')
|
||||||
|
|
||||||
elif get_value == 'off':
|
elif get_value == 'off':
|
||||||
if self.ModConfig.reputation_ban_all_chan == 0:
|
if self.ModConfig.reputation_ban_all_chan == 0:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}BAN ON ALL CHANS{self.Config.CONFIG_COLOR['noire']} ] : Already deactivated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}BAN ON ALL CHANS{self.Config.COLORS.black} ] : Already deactivated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# self.update_db_configuration(key, 0)
|
# self.update_db_configuration(key, 0)
|
||||||
self.__update_configuration(key, 0)
|
self.__update_configuration(key, 0)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}BAN ON ALL CHANS{self.Config.CONFIG_COLOR["noire"]} ] : Deactivated by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}BAN ON ALL CHANS{self.Config.COLORS.black} ] : Deactivated by {fromuser}')
|
||||||
|
|
||||||
case 'limit':
|
case 'limit':
|
||||||
reputation_seuil = int(cmd[3])
|
reputation_seuil = int(cmd[3])
|
||||||
@@ -1360,7 +1343,7 @@ class Defender():
|
|||||||
# self.update_db_configuration(key, reputation_seuil)
|
# self.update_db_configuration(key, reputation_seuil)
|
||||||
self.__update_configuration(key, reputation_seuil)
|
self.__update_configuration(key, reputation_seuil)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}REPUTATION SEUIL{self.Config.CONFIG_COLOR["noire"]} ] : Limit set to {str(reputation_seuil)} by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION SEUIL{self.Config.COLORS.black} ] : Limit set to {str(reputation_seuil)} by {fromuser}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation set to {reputation_seuil}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation set to {reputation_seuil}')
|
||||||
|
|
||||||
case 'timer':
|
case 'timer':
|
||||||
@@ -1368,7 +1351,7 @@ class Defender():
|
|||||||
key = 'reputation_timer'
|
key = 'reputation_timer'
|
||||||
self.__update_configuration(key, reputation_timer)
|
self.__update_configuration(key, reputation_timer)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}REPUTATION TIMER{self.Config.CONFIG_COLOR["noire"]} ] : Timer set to {str(reputation_timer)} minute(s) by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION TIMER{self.Config.COLORS.black} ] : Timer set to {str(reputation_timer)} minute(s) by {fromuser}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation set to {reputation_timer}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation set to {reputation_timer}')
|
||||||
|
|
||||||
case 'score_after_release':
|
case 'score_after_release':
|
||||||
@@ -1376,7 +1359,7 @@ class Defender():
|
|||||||
key = 'reputation_score_after_release'
|
key = 'reputation_score_after_release'
|
||||||
self.__update_configuration(key, reputation_score_after_release)
|
self.__update_configuration(key, reputation_score_after_release)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}REPUTATION SCORE AFTER RELEASE{self.Config.CONFIG_COLOR["noire"]} ] : Reputation score after release set to {str(reputation_score_after_release)} by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION SCORE AFTER RELEASE{self.Config.COLORS.black} ] : Reputation score after release set to {str(reputation_score_after_release)} by {fromuser}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation score after release set to {reputation_score_after_release}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation score after release set to {reputation_score_after_release}')
|
||||||
|
|
||||||
case 'security_group':
|
case 'security_group':
|
||||||
@@ -1384,7 +1367,7 @@ class Defender():
|
|||||||
key = 'reputation_sg'
|
key = 'reputation_sg'
|
||||||
self.__update_configuration(key, reputation_sg)
|
self.__update_configuration(key, reputation_sg)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR["verte"]}REPUTATION SECURITY-GROUP{self.Config.CONFIG_COLOR["noire"]} ] : Reputation Security-group set to {str(reputation_sg)} by {fromuser}')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}REPUTATION SECURITY-GROUP{self.Config.COLORS.black} ] : Reputation Security-group set to {str(reputation_sg)} by {fromuser}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation score after release set to {reputation_sg}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : Reputation score after release set to {reputation_sg}')
|
||||||
|
|
||||||
case _:
|
case _:
|
||||||
@@ -1414,9 +1397,9 @@ class Defender():
|
|||||||
# .proxy_scan set psutil_scan on/off --> Active les informations de connexion a la machine locale
|
# .proxy_scan set psutil_scan on/off --> Active les informations de connexion a la machine locale
|
||||||
# .proxy_scan set abuseipdb_scan on/off --> Active le scan via l'api abuseipdb
|
# .proxy_scan set abuseipdb_scan on/off --> Active le scan via l'api abuseipdb
|
||||||
len_cmd = len(cmd)
|
len_cmd = len(cmd)
|
||||||
color_green = self.Config.CONFIG_COLOR['verte']
|
color_green = self.Config.COLORS.green
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
|
|
||||||
if len_cmd == 4:
|
if len_cmd == 4:
|
||||||
set_key = str(cmd[1]).lower()
|
set_key = str(cmd[1]).lower()
|
||||||
@@ -1548,21 +1531,21 @@ class Defender():
|
|||||||
key = 'flood'
|
key = 'flood'
|
||||||
if activation == 'on':
|
if activation == 'on':
|
||||||
if self.ModConfig.flood == 1:
|
if self.ModConfig.flood == 1:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Already activated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Already activated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.__update_configuration(key, 1)
|
self.__update_configuration(key, 1)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Activated by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Activated by {fromuser}")
|
||||||
|
|
||||||
if activation == 'off':
|
if activation == 'off':
|
||||||
if self.ModConfig.flood == 0:
|
if self.ModConfig.flood == 0:
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['rouge']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Already Deactivated")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.red}FLOOD{self.Config.COLORS.black} ] : Already Deactivated")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.__update_configuration(key, 0)
|
self.__update_configuration(key, 0)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Deactivated by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Deactivated by {fromuser}")
|
||||||
|
|
||||||
if len_cmd == 4:
|
if len_cmd == 4:
|
||||||
set_key = str(cmd[2]).lower()
|
set_key = str(cmd[2]).lower()
|
||||||
@@ -1574,21 +1557,21 @@ class Defender():
|
|||||||
set_value = int(cmd[3])
|
set_value = int(cmd[3])
|
||||||
self.__update_configuration(key, set_value)
|
self.__update_configuration(key, set_value)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Flood message set to {set_value} by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Flood message set to {set_value} by {fromuser}")
|
||||||
|
|
||||||
case 'flood_time':
|
case 'flood_time':
|
||||||
key = 'flood_time'
|
key = 'flood_time'
|
||||||
set_value = int(cmd[3])
|
set_value = int(cmd[3])
|
||||||
self.__update_configuration(key, set_value)
|
self.__update_configuration(key, set_value)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Flood time set to {set_value} by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Flood time set to {set_value} by {fromuser}")
|
||||||
|
|
||||||
case 'flood_timer':
|
case 'flood_timer':
|
||||||
key = 'flood_timer'
|
key = 'flood_timer'
|
||||||
set_value = int(cmd[3])
|
set_value = int(cmd[3])
|
||||||
self.__update_configuration(key, set_value)
|
self.__update_configuration(key, set_value)
|
||||||
|
|
||||||
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.CONFIG_COLOR['verte']}FLOOD{self.Config.CONFIG_COLOR['noire']} ] : Flood timer set to {set_value} by {fromuser}")
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {dchanlog} :[ {self.Config.COLORS.green}FLOOD{self.Config.COLORS.black} ] : Flood timer set to {set_value} by {fromuser}")
|
||||||
|
|
||||||
case _:
|
case _:
|
||||||
pass
|
pass
|
||||||
@@ -1597,9 +1580,9 @@ class Defender():
|
|||||||
self.Logs.error(f"{self.__class__.__name__} Value Error : {ve}")
|
self.Logs.error(f"{self.__class__.__name__} Value Error : {ve}")
|
||||||
|
|
||||||
case 'status':
|
case 'status':
|
||||||
color_green = self.Config.CONFIG_COLOR['verte']
|
color_green = self.Config.COLORS.green
|
||||||
color_red = self.Config.CONFIG_COLOR['rouge']
|
color_red = self.Config.COLORS.red
|
||||||
color_black = self.Config.CONFIG_COLOR['noire']
|
color_black = self.Config.COLORS.black
|
||||||
try:
|
try:
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : [{color_green if self.ModConfig.reputation == 1 else color_red}Reputation{color_black}] ==> {self.ModConfig.reputation}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : [{color_green if self.ModConfig.reputation == 1 else color_red}Reputation{color_black}] ==> {self.ModConfig.reputation}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : reputation_seuil ==> {self.ModConfig.reputation_seuil}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : reputation_seuil ==> {self.ModConfig.reputation_seuil}')
|
||||||
@@ -1626,6 +1609,12 @@ class Defender():
|
|||||||
UserObject = self.User.get_User(nickoruid)
|
UserObject = self.User.get_User(nickoruid)
|
||||||
|
|
||||||
if not UserObject is None:
|
if not UserObject is None:
|
||||||
|
channels: list = []
|
||||||
|
for chan in self.Channel.UID_CHANNEL_DB:
|
||||||
|
for uid_in_chan in chan.uids:
|
||||||
|
if self.Base.clean_uid(uid_in_chan) == UserObject.uid:
|
||||||
|
channels.append(chan.name)
|
||||||
|
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : UID : {UserObject.uid}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : UID : {UserObject.uid}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : NICKNAME : {UserObject.nickname}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : NICKNAME : {UserObject.nickname}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : USERNAME : {UserObject.username}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : USERNAME : {UserObject.username}')
|
||||||
@@ -1638,6 +1627,7 @@ class Defender():
|
|||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : WebWebsocket : {UserObject.isWebsocket}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : WebWebsocket : {UserObject.isWebsocket}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : REPUTATION : {UserObject.score_connexion}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : REPUTATION : {UserObject.score_connexion}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : MODES : {UserObject.umodes}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : MODES : {UserObject.umodes}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : CHANNELS : {channels}')
|
||||||
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : CONNECTION TIME : {UserObject.connexion_datetime}')
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} : CONNECTION TIME : {UserObject.connexion_datetime}')
|
||||||
else:
|
else:
|
||||||
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} : This user {nickoruid} doesn't exist")
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} : This user {nickoruid} doesn't exist")
|
||||||
|
|||||||
293
mods/mod_jsonrpc.py
Normal file
293
mods/mod_jsonrpc.py
Normal file
@@ -0,0 +1,293 @@
|
|||||||
|
from dataclasses import dataclass
|
||||||
|
from core.irc import Irc
|
||||||
|
from unrealircd_rpc_py.Live import Live
|
||||||
|
from unrealircd_rpc_py.Loader import Loader
|
||||||
|
|
||||||
|
|
||||||
|
class Jsonrpc():
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class ModConfModel:
|
||||||
|
"""The Model containing the module parameters
|
||||||
|
"""
|
||||||
|
param_exemple1: str
|
||||||
|
param_exemple2: int
|
||||||
|
|
||||||
|
def __init__(self, ircInstance:Irc) -> None:
|
||||||
|
|
||||||
|
# Module name (Mandatory)
|
||||||
|
self.module_name = 'mod_' + str(self.__class__.__name__).lower()
|
||||||
|
|
||||||
|
# Add Irc Object to the module (Mandatory)
|
||||||
|
self.Irc = ircInstance
|
||||||
|
|
||||||
|
# Add Global Configuration to the module (Mandatory)
|
||||||
|
self.Config = ircInstance.Config
|
||||||
|
|
||||||
|
# Add Base object to the module (Mandatory)
|
||||||
|
self.Base = ircInstance.Base
|
||||||
|
|
||||||
|
# Add logs object to the module (Mandatory)
|
||||||
|
self.Logs = ircInstance.Base.logs
|
||||||
|
|
||||||
|
# Add User object to the module (Mandatory)
|
||||||
|
self.User = ircInstance.User
|
||||||
|
|
||||||
|
# Add Channel object to the module (Mandatory)
|
||||||
|
self.Channel = ircInstance.Channel
|
||||||
|
|
||||||
|
# Create module commands (Mandatory)
|
||||||
|
self.commands_level = {
|
||||||
|
1: ['jsonrpc', 'jruser']
|
||||||
|
}
|
||||||
|
|
||||||
|
# Init the module
|
||||||
|
self.__init_module()
|
||||||
|
|
||||||
|
# Log the module
|
||||||
|
self.Logs.debug(f'Module {self.module_name} loaded ...')
|
||||||
|
|
||||||
|
def __init_module(self) -> None:
|
||||||
|
|
||||||
|
# Insert module commands into the core one (Mandatory)
|
||||||
|
self.__set_commands(self.commands_level)
|
||||||
|
|
||||||
|
# Create you own tables (Mandatory)
|
||||||
|
# self.__create_tables()
|
||||||
|
|
||||||
|
# Load module configuration and sync with core one (Mandatory)
|
||||||
|
self.__load_module_configuration()
|
||||||
|
# End of mandatory methods you can start your customization #
|
||||||
|
|
||||||
|
# self.UnrealIrcdRpcLive: Live = Live(
|
||||||
|
# req_method='unixsocket',
|
||||||
|
# path_to_socket_file=self.Config.JSONRPC_PATH_TO_SOCKET_FILE,
|
||||||
|
# callback_object_instance=self,
|
||||||
|
# callback_method_name='callback_sent_to_irc'
|
||||||
|
# )
|
||||||
|
|
||||||
|
self.UnrealIrcdRpcLive: Live = Live(
|
||||||
|
req_method='websocket',
|
||||||
|
url=self.Config.JSONRPC_URL,
|
||||||
|
username=self.Config.JSONRPC_USER,
|
||||||
|
password=self.Config.JSONRPC_PASSWORD,
|
||||||
|
callback_object_instance=self,
|
||||||
|
callback_method_name='callback_sent_to_irc'
|
||||||
|
)
|
||||||
|
|
||||||
|
self.Rpc: Loader = Loader(
|
||||||
|
req_method=self.Config.JSONRPC_METHOD,
|
||||||
|
url=self.Config.JSONRPC_URL,
|
||||||
|
username=self.Config.JSONRPC_USER,
|
||||||
|
password=self.Config.JSONRPC_PASSWORD
|
||||||
|
)
|
||||||
|
|
||||||
|
self.subscribed = False
|
||||||
|
|
||||||
|
if self.Rpc.Error.code != 0:
|
||||||
|
self.Irc.sendPrivMsg(f"[{self.Config.COLORS.red}ERROR{self.Config.COLORS.nogc}] {self.Rpc.Error.message}", self.Config.SERVICE_CHANLOG)
|
||||||
|
|
||||||
|
if self.UnrealIrcdRpcLive.Error.code != 0:
|
||||||
|
self.Irc.sendPrivMsg(f"[{self.Config.COLORS.red}ERROR{self.Config.COLORS.nogc}] {self.UnrealIrcdRpcLive.Error.message}", self.Config.SERVICE_CHANLOG)
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
def __set_commands(self, commands:dict[int, list[str]]) -> None:
|
||||||
|
"""### Rajoute les commandes du module au programme principal
|
||||||
|
|
||||||
|
Args:
|
||||||
|
commands (list): Liste des commandes du module
|
||||||
|
"""
|
||||||
|
for level, com in commands.items():
|
||||||
|
for c in commands[level]:
|
||||||
|
if not c in self.Irc.commands:
|
||||||
|
self.Irc.commands_level[level].append(c)
|
||||||
|
self.Irc.commands.append(c)
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
def __create_tables(self) -> None:
|
||||||
|
"""Methode qui va créer la base de donnée si elle n'existe pas.
|
||||||
|
Une Session unique pour cette classe sera crée, qui sera utilisé dans cette classe / module
|
||||||
|
Args:
|
||||||
|
database_name (str): Nom de la base de données ( pas d'espace dans le nom )
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
None: Aucun retour n'es attendu
|
||||||
|
"""
|
||||||
|
|
||||||
|
table_logs = '''CREATE TABLE IF NOT EXISTS test_logs (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
datetime TEXT,
|
||||||
|
server_msg TEXT
|
||||||
|
)
|
||||||
|
'''
|
||||||
|
|
||||||
|
self.Base.db_execute_query(table_logs)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def callback_sent_to_irc(self, json_response: str):
|
||||||
|
|
||||||
|
dnickname = self.Config.SERVICE_NICKNAME
|
||||||
|
dchanlog = self.Config.SERVICE_CHANLOG
|
||||||
|
green = self.Config.COLORS.green
|
||||||
|
nogc = self.Config.COLORS.nogc
|
||||||
|
bold = self.Config.COLORS.bold
|
||||||
|
red = self.Config.COLORS.red
|
||||||
|
|
||||||
|
if json_response.result == True:
|
||||||
|
self.Irc.sendPrivMsg(msg=f"[{bold}{green}JSONRPC{nogc}{bold}] Event activated", channel=dchanlog)
|
||||||
|
return None
|
||||||
|
|
||||||
|
level = json_response.result.level
|
||||||
|
subsystem = json_response.result.subsystem
|
||||||
|
event_id = json_response.result.event_id
|
||||||
|
log_source = json_response.result.log_source
|
||||||
|
msg = json_response.result.msg
|
||||||
|
|
||||||
|
build_msg = f"{green}{log_source}{nogc}: [{bold}{level}{bold}] {subsystem}.{event_id} - {msg}"
|
||||||
|
|
||||||
|
self.Irc.sendPrivMsg(msg=build_msg, channel=dchanlog)
|
||||||
|
|
||||||
|
def thread_start_jsonrpc(self):
|
||||||
|
|
||||||
|
if self.UnrealIrcdRpcLive.Error.code == 0:
|
||||||
|
self.UnrealIrcdRpcLive.subscribe(["all"])
|
||||||
|
self.subscribed = True
|
||||||
|
else:
|
||||||
|
self.Irc.sendPrivMsg(f"[{self.Config.COLORS.red}ERROR{self.Config.COLORS.nogc}] {self.UnrealIrcdRpcLive.Error.message}", self.Config.SERVICE_CHANLOG)
|
||||||
|
|
||||||
|
def __load_module_configuration(self) -> None:
|
||||||
|
"""### Load Module Configuration
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# Build the default configuration model (Mandatory)
|
||||||
|
self.ModConfig = self.ModConfModel(param_exemple1='param value 1', param_exemple2=1)
|
||||||
|
|
||||||
|
# Sync the configuration with core configuration (Mandatory)
|
||||||
|
#self.Base.db_sync_core_config(self.module_name, self.ModConfig)
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
except TypeError as te:
|
||||||
|
self.Logs.critical(te)
|
||||||
|
|
||||||
|
def __update_configuration(self, param_key: str, param_value: str):
|
||||||
|
"""Update the local and core configuration
|
||||||
|
|
||||||
|
Args:
|
||||||
|
param_key (str): The parameter key
|
||||||
|
param_value (str): The parameter value
|
||||||
|
"""
|
||||||
|
self.Base.db_update_core_config(self.module_name, self.ModConfig, param_key, param_value)
|
||||||
|
|
||||||
|
def unload(self) -> None:
|
||||||
|
if self.UnrealIrcdRpcLive.Error.code != -1:
|
||||||
|
self.UnrealIrcdRpcLive.unsubscribe()
|
||||||
|
return None
|
||||||
|
|
||||||
|
def cmd(self, data:list) -> None:
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
||||||
|
|
||||||
|
command = str(cmd[0]).lower()
|
||||||
|
dnickname = self.Config.SERVICE_NICKNAME
|
||||||
|
fromuser = user
|
||||||
|
fromchannel = str(channel) if not channel is None else None
|
||||||
|
|
||||||
|
match command:
|
||||||
|
|
||||||
|
case 'jsonrpc':
|
||||||
|
try:
|
||||||
|
option = str(cmd[1]).lower()
|
||||||
|
|
||||||
|
if len(command) == 1:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :/msg {dnickname} jsonrpc on')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :/msg {dnickname} jsonrpc off')
|
||||||
|
|
||||||
|
match option:
|
||||||
|
|
||||||
|
case 'on':
|
||||||
|
self.Base.create_thread(self.thread_start_jsonrpc, run_once=True)
|
||||||
|
|
||||||
|
case 'off':
|
||||||
|
self.UnrealIrcdRpcLive.unsubscribe()
|
||||||
|
|
||||||
|
except IndexError as ie:
|
||||||
|
self.Logs.error(ie)
|
||||||
|
|
||||||
|
case 'jruser':
|
||||||
|
try:
|
||||||
|
option = str(cmd[1]).lower()
|
||||||
|
|
||||||
|
if len(command) == 1:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :/msg {dnickname} jruser get nickname')
|
||||||
|
|
||||||
|
match option:
|
||||||
|
|
||||||
|
case 'get':
|
||||||
|
nickname = str(cmd[2])
|
||||||
|
uid_to_get = self.User.get_uid(nickname)
|
||||||
|
if uid_to_get is None:
|
||||||
|
return None
|
||||||
|
|
||||||
|
rpc = self.Rpc
|
||||||
|
|
||||||
|
UserInfo = rpc.User.get(uid_to_get)
|
||||||
|
if rpc.Error.code != 0:
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :{rpc.Error.message}')
|
||||||
|
return None
|
||||||
|
|
||||||
|
chan_list = []
|
||||||
|
for chan in UserInfo.user.channels:
|
||||||
|
chan_list.append(chan.name)
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :UID : {UserInfo.id}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :NICKNAME : {UserInfo.name}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :USERNAME : {UserInfo.user.username}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :REALNAME : {UserInfo.user.realname}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :MODES : {UserInfo.user.modes}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CHANNELS : {chan_list}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :SECURITY GROUP : {UserInfo.user.security_groups}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :REPUTATION : {UserInfo.user.reputation}')
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :IP : {UserInfo.ip}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :COUNTRY CODE : {UserInfo.geoip.country_code}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :ASN : {UserInfo.geoip.asn}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :ASNAME : {UserInfo.geoip.asname}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CLOAKED HOST : {UserInfo.user.cloakedhost}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :HOSTNAME : {UserInfo.hostname}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :VHOST : {UserInfo.user.vhost}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CLIENT PORT : {UserInfo.client_port}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :SERVER PORT : {UserInfo.server_port}')
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CERTFP : {UserInfo.tls.certfp}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CIPHER : {UserInfo.tls.cipher}')
|
||||||
|
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :IDLE SINCE : {UserInfo.idle_since}')
|
||||||
|
self.Irc.send2socket(f':{dnickname} NOTICE {fromuser} :CONNECTED SINCE : {UserInfo.connected_since}')
|
||||||
|
|
||||||
|
except IndexError as ie:
|
||||||
|
self.Logs.error(ie)
|
||||||
|
|
||||||
|
case 'ia':
|
||||||
|
try:
|
||||||
|
|
||||||
|
self.Base.create_thread(self.thread_ask_ia, ('',))
|
||||||
|
|
||||||
|
self.Irc.send2socket(f":{dnickname} NOTICE {fromuser} : This is a notice to the sender ...")
|
||||||
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {fromuser} : This is private message to the sender ...")
|
||||||
|
|
||||||
|
if not fromchannel is None:
|
||||||
|
self.Irc.send2socket(f":{dnickname} PRIVMSG {fromchannel} : This is channel message to the sender ...")
|
||||||
|
|
||||||
|
# How to update your module configuration
|
||||||
|
self.__update_configuration('param_exemple2', 7)
|
||||||
|
|
||||||
|
# Log if you want the result
|
||||||
|
self.Logs.debug(f"Test logs ready")
|
||||||
|
|
||||||
|
except Exception as err:
|
||||||
|
self.Logs.error(f"Unknown Error: {err}")
|
||||||
@@ -124,8 +124,16 @@ class Test():
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
def cmd(self, data:list) -> None:
|
def cmd(self, data:list) -> None:
|
||||||
|
try:
|
||||||
|
cmd = list(data).copy()
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
except KeyError as ke:
|
||||||
|
self.Base.logs.error(f"Key Error: {ke}")
|
||||||
|
except IndexError as ie:
|
||||||
|
self.Base.logs.error(f"{ie} / {cmd} / length {str(len(cmd))}")
|
||||||
|
except Exception as err:
|
||||||
|
self.Base.logs.error(f"General Error: {err}")
|
||||||
|
|
||||||
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
||||||
|
|
||||||
|
|||||||
@@ -229,11 +229,11 @@ class Votekick():
|
|||||||
if chan.channel_name == channel:
|
if chan.channel_name == channel:
|
||||||
target_user = self.User.get_nickname(chan.target_user)
|
target_user = self.User.get_nickname(chan.target_user)
|
||||||
if chan.vote_for > chan.vote_against:
|
if chan.vote_for > chan.vote_against:
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.CONFIG_COLOR["gras"]}{target_user}{self.Config.CONFIG_COLOR["nogc"]} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll be kicked from the channel')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.COLORS.bold}{target_user}{self.Config.COLORS.nogc} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll be kicked from the channel')
|
||||||
self.Irc.send2socket(f":{dnickname} KICK {channel} {target_user} Following the vote, you are not welcome in {channel}")
|
self.Irc.send2socket(f":{dnickname} KICK {channel} {target_user} Following the vote, you are not welcome in {channel}")
|
||||||
self.Channel.delete_user_from_channel(channel, self.User.get_uid(target_user))
|
self.Channel.delete_user_from_channel(channel, self.User.get_uid(target_user))
|
||||||
elif chan.vote_for <= chan.vote_against:
|
elif chan.vote_for <= chan.vote_against:
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.CONFIG_COLOR["gras"]}{target_user}{self.Config.CONFIG_COLOR["nogc"]} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll remain in the channel')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.COLORS.bold}{target_user}{self.Config.COLORS.nogc} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll remain in the channel')
|
||||||
|
|
||||||
# Init the system
|
# Init the system
|
||||||
if self.init_vote_system(channel):
|
if self.init_vote_system(channel):
|
||||||
@@ -242,15 +242,16 @@ class Votekick():
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
def cmd(self, data:list) -> None:
|
def cmd(self, data:list) -> None:
|
||||||
cmd = list(data).copy()
|
try:
|
||||||
|
cmd = list(data).copy()
|
||||||
|
return None
|
||||||
|
|
||||||
match cmd[2]:
|
except KeyError as ke:
|
||||||
case 'SJOIN':
|
self.Base.logs.error(f"Key Error: {ke}")
|
||||||
pass
|
except IndexError as ie:
|
||||||
case _:
|
self.Base.logs.error(f"{ie} / {cmd} / length {str(len(cmd))}")
|
||||||
pass
|
except Exception as err:
|
||||||
|
self.Base.logs.error(f"General Error: {err}")
|
||||||
return None
|
|
||||||
|
|
||||||
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
def _hcmds(self, user:str, channel: any, cmd: list, fullcmd: list = []) -> None:
|
||||||
# cmd is the command starting from the user command
|
# cmd is the command starting from the user command
|
||||||
@@ -471,10 +472,10 @@ class Votekick():
|
|||||||
if chan.channel_name == channel:
|
if chan.channel_name == channel:
|
||||||
target_user = self.User.get_nickname(chan.target_user)
|
target_user = self.User.get_nickname(chan.target_user)
|
||||||
if chan.vote_for > chan.vote_against:
|
if chan.vote_for > chan.vote_against:
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.CONFIG_COLOR["gras"]}{target_user}{self.Config.CONFIG_COLOR["nogc"]} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll be kicked from the channel')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.COLORS.bold}{target_user}{self.Config.COLORS.nogc} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll be kicked from the channel')
|
||||||
self.Irc.send2socket(f":{dnickname} KICK {channel} {target_user} Following the vote, you are not welcome in {channel}")
|
self.Irc.send2socket(f":{dnickname} KICK {channel} {target_user} Following the vote, you are not welcome in {channel}")
|
||||||
elif chan.vote_for <= chan.vote_against:
|
elif chan.vote_for <= chan.vote_against:
|
||||||
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.CONFIG_COLOR["gras"]}{target_user}{self.Config.CONFIG_COLOR["nogc"]} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll remain in the channel')
|
self.Irc.send2socket(f':{dnickname} PRIVMSG {channel} :User {self.Config.COLORS.bold}{target_user}{self.Config.COLORS.nogc} has {chan.vote_against} votes against and {chan.vote_for} votes for. For this reason, it\'ll remain in the channel')
|
||||||
|
|
||||||
# Init the system
|
# Init the system
|
||||||
if self.init_vote_system(channel):
|
if self.init_vote_system(channel):
|
||||||
|
|||||||
@@ -1,3 +1,9 @@
|
|||||||
{
|
{
|
||||||
"version": "5.2.9"
|
"version": "5.3.9",
|
||||||
|
|
||||||
|
"requests": "2.32.3",
|
||||||
|
"psutil": "6.0.0",
|
||||||
|
"unrealircd_rpc_py": "1.0.6",
|
||||||
|
"sqlalchemy": "2.0.35",
|
||||||
|
"faker": "30.1.0"
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user