Wipe branch
This commit is contained in:
commit
553754cd77
162
.gitignore
vendored
Normal file
162
.gitignore
vendored
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
# ---> Python
|
||||||
|
# Byte-compiled / optimized / DLL files
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
||||||
|
|
||||||
|
# C extensions
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Distribution / packaging
|
||||||
|
.Python
|
||||||
|
build/
|
||||||
|
develop-eggs/
|
||||||
|
dist/
|
||||||
|
downloads/
|
||||||
|
eggs/
|
||||||
|
.eggs/
|
||||||
|
lib/
|
||||||
|
lib64/
|
||||||
|
parts/
|
||||||
|
sdist/
|
||||||
|
var/
|
||||||
|
wheels/
|
||||||
|
share/python-wheels/
|
||||||
|
*.egg-info/
|
||||||
|
.installed.cfg
|
||||||
|
*.egg
|
||||||
|
MANIFEST
|
||||||
|
|
||||||
|
# PyInstaller
|
||||||
|
# Usually these files are written by a python script from a template
|
||||||
|
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||||
|
*.manifest
|
||||||
|
*.spec
|
||||||
|
|
||||||
|
# Installer logs
|
||||||
|
pip-log.txt
|
||||||
|
pip-delete-this-directory.txt
|
||||||
|
|
||||||
|
# Unit test / coverage reports
|
||||||
|
htmlcov/
|
||||||
|
.tox/
|
||||||
|
.nox/
|
||||||
|
.coverage
|
||||||
|
.coverage.*
|
||||||
|
.cache
|
||||||
|
nosetests.xml
|
||||||
|
coverage.xml
|
||||||
|
*.cover
|
||||||
|
*.py,cover
|
||||||
|
.hypothesis/
|
||||||
|
.pytest_cache/
|
||||||
|
cover/
|
||||||
|
|
||||||
|
# Translations
|
||||||
|
*.mo
|
||||||
|
*.pot
|
||||||
|
|
||||||
|
# Django stuff:
|
||||||
|
*.log
|
||||||
|
local_settings.py
|
||||||
|
db.sqlite3
|
||||||
|
db.sqlite3-journal
|
||||||
|
|
||||||
|
# Flask stuff:
|
||||||
|
instance/
|
||||||
|
.webassets-cache
|
||||||
|
|
||||||
|
# Scrapy stuff:
|
||||||
|
.scrapy
|
||||||
|
|
||||||
|
# Sphinx documentation
|
||||||
|
docs/_build/
|
||||||
|
|
||||||
|
# PyBuilder
|
||||||
|
.pybuilder/
|
||||||
|
target/
|
||||||
|
|
||||||
|
# Jupyter Notebook
|
||||||
|
.ipynb_checkpoints
|
||||||
|
|
||||||
|
# IPython
|
||||||
|
profile_default/
|
||||||
|
ipython_config.py
|
||||||
|
|
||||||
|
# pyenv
|
||||||
|
# For a library or package, you might want to ignore these files since the code is
|
||||||
|
# intended to run in multiple environments; otherwise, check them in:
|
||||||
|
# .python-version
|
||||||
|
|
||||||
|
# pipenv
|
||||||
|
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
|
||||||
|
# However, in case of collaboration, if having platform-specific dependencies or dependencies
|
||||||
|
# having no cross-platform support, pipenv may install dependencies that don't work, or not
|
||||||
|
# install all needed dependencies.
|
||||||
|
#Pipfile.lock
|
||||||
|
|
||||||
|
# poetry
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
|
||||||
|
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
||||||
|
# commonly ignored for libraries.
|
||||||
|
# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
|
||||||
|
#poetry.lock
|
||||||
|
|
||||||
|
# pdm
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
|
||||||
|
#pdm.lock
|
||||||
|
# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
|
||||||
|
# in version control.
|
||||||
|
# https://pdm.fming.dev/#use-with-ide
|
||||||
|
.pdm.toml
|
||||||
|
|
||||||
|
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
||||||
|
__pypackages__/
|
||||||
|
|
||||||
|
# Celery stuff
|
||||||
|
celerybeat-schedule
|
||||||
|
celerybeat.pid
|
||||||
|
|
||||||
|
# SageMath parsed files
|
||||||
|
*.sage.py
|
||||||
|
|
||||||
|
# Environments
|
||||||
|
.env
|
||||||
|
.venv
|
||||||
|
env/
|
||||||
|
venv/
|
||||||
|
ENV/
|
||||||
|
env.bak/
|
||||||
|
venv.bak/
|
||||||
|
|
||||||
|
# Spyder project settings
|
||||||
|
.spyderproject
|
||||||
|
.spyproject
|
||||||
|
|
||||||
|
# Rope project settings
|
||||||
|
.ropeproject
|
||||||
|
|
||||||
|
# mkdocs documentation
|
||||||
|
/site
|
||||||
|
|
||||||
|
# mypy
|
||||||
|
.mypy_cache/
|
||||||
|
.dmypy.json
|
||||||
|
dmypy.json
|
||||||
|
|
||||||
|
# Pyre type checker
|
||||||
|
.pyre/
|
||||||
|
|
||||||
|
# pytype static type analyzer
|
||||||
|
.pytype/
|
||||||
|
|
||||||
|
# Cython debug symbols
|
||||||
|
cython_debug/
|
||||||
|
|
||||||
|
# PyCharm
|
||||||
|
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||||
|
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||||
|
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||||
|
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||||
|
#.idea/
|
||||||
|
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
BIT License
|
||||||
|
|
||||||
|
Copyright (c) 2023 Bit.Corp
|
||||||
|
|
||||||
|
Permission is deny, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
3
TODO
Normal file
3
TODO
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
MAKE BACKUPS FOR LICENSE SERVER!!!
|
||||||
|
Enable LED when WARN or CRIT
|
||||||
|
Switch debug remotely
|
27
asics.py
Normal file
27
asics.py
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
from enum import Enum, auto
|
||||||
|
|
||||||
|
class ASICs(Enum):
|
||||||
|
# Avalon 10-13
|
||||||
|
Avalon1066 = auto()
|
||||||
|
Avalon1126 = auto()
|
||||||
|
|
||||||
|
# Antminer S
|
||||||
|
AntminerS19 = auto()
|
||||||
|
AntminerS19J = auto()
|
||||||
|
AntminerS19Pro = auto()
|
||||||
|
AntminerS19JPro = auto()
|
||||||
|
|
||||||
|
# Antminer L
|
||||||
|
AntminerL3plus = auto()
|
||||||
|
AntminerL7 = auto()
|
||||||
|
|
||||||
|
# Antminer T
|
||||||
|
AntminerT17 = auto()
|
||||||
|
AntminerT17plus = auto()
|
||||||
|
|
||||||
|
# WhatsMiner
|
||||||
|
WhatsMinerM20s = auto()
|
||||||
|
WhatsMinerM21s = auto()
|
||||||
|
WhatsMinerM30 = auto()
|
||||||
|
WhatsMinerM31 = auto()
|
||||||
|
WhatsMinerM50 = auto()
|
4
build.sh
Executable file
4
build.sh
Executable file
@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
pyarmor-7 pack --clean -e "--onefile " main.py
|
||||||
|
#pyarmor-7 pack --clean -e "--onefile " main.py asics.py checker.py colors.py config.py license.py macros.py telegrambot.py updater.py webserver.py
|
334
checker.py
Normal file
334
checker.py
Normal file
@ -0,0 +1,334 @@
|
|||||||
|
from threading import Thread
|
||||||
|
from mysql.connector import connect, Error
|
||||||
|
from time import sleep, time
|
||||||
|
from asics import ASICs
|
||||||
|
from macros import *
|
||||||
|
import os, re, json, socket, requests
|
||||||
|
|
||||||
|
|
||||||
|
class CThread(Thread):
|
||||||
|
conn = None
|
||||||
|
|
||||||
|
def __init__(self, event):
|
||||||
|
super(CThread, self).__init__()
|
||||||
|
self.event = event
|
||||||
|
|
||||||
|
|
||||||
|
def getreg(self, reg, str):
|
||||||
|
res = re.findall(reg, str)
|
||||||
|
if res: return res[0][1]
|
||||||
|
else: return False
|
||||||
|
|
||||||
|
|
||||||
|
def getstat_AntminerL3(self, log):
|
||||||
|
status = 'ok'
|
||||||
|
|
||||||
|
return status
|
||||||
|
|
||||||
|
|
||||||
|
def getstat_Avalon1XXX(self, log, type):
|
||||||
|
status = 'ok'
|
||||||
|
|
||||||
|
rej = float(self.getreg('("Device Rejected%":([0-9.]+),)', log))
|
||||||
|
ghs = int(float(self.getreg('(GHSmm\[([0-9.:,]+)\])', log)))
|
||||||
|
avg = int(float(self.getreg('(GHSavg\[([0-9.: ]+)\])', log)))
|
||||||
|
# brd = self.getreg('(MGHS\[([0-9.:, ]+)\])', log)
|
||||||
|
tmp = int(self.getreg('(TMax\[([0-9]+)\])', log))
|
||||||
|
|
||||||
|
chash = CONF.get('asics', ASICs(type).name, 'crit', 'hash')
|
||||||
|
whash = CONF.get('asics', ASICs(type).name, 'warn', 'hash')
|
||||||
|
|
||||||
|
ctemp = CONF.get('asics', ASICs(type).name, 'crit', 'temp')
|
||||||
|
wtemp = CONF.get('asics', ASICs(type).name, 'warn', 'temp')
|
||||||
|
|
||||||
|
crej = CONF.get('asics', ASICs(type).name, 'crit', 'rej')
|
||||||
|
wrej = CONF.get('asics', ASICs(type).name, 'warn', 'rej')
|
||||||
|
|
||||||
|
if rej > crej or ghs < chash or avg < chash or tmp > ctemp: status = 'crit'
|
||||||
|
elif rej > wrej or ghs < whash or avg < whash or tmp > wtemp: status = 'warn'
|
||||||
|
|
||||||
|
DEBUG(f"{rej} | {ghs} | {avg} | {brd} | {tmp}")
|
||||||
|
|
||||||
|
return status
|
||||||
|
|
||||||
|
|
||||||
|
def getstat_AntminerS19(self, log):
|
||||||
|
status = 'ok'
|
||||||
|
|
||||||
|
rej = float(self.getreg('("Device Rejected%": ([0-9.]+),)', log))
|
||||||
|
ghs = int(float(self.getreg('("GHS 5s": ([0-9.]+),)', log)))
|
||||||
|
avg = int(float(self.getreg('("GHS av": ([0-9.]+),)', log)))
|
||||||
|
tmp = int(self.getreg('("TMax": ([0-9]+),)', log))
|
||||||
|
|
||||||
|
chash = CONF.get('asics', ASICs(type).name, 'crit', 'hash')
|
||||||
|
whash = CONF.get('asics', ASICs(type).name, 'warn', 'hash')
|
||||||
|
|
||||||
|
ctemp = CONF.get('asics', ASICs(type).name, 'crit', 'temp')
|
||||||
|
wtemp = CONF.get('asics', ASICs(type).name, 'warn', 'temp')
|
||||||
|
|
||||||
|
crej = CONF.get('asics', ASICs(type).name, 'crit', 'rej')
|
||||||
|
wrej = CONF.get('asics', ASICs(type).name, 'warn', 'rej')
|
||||||
|
|
||||||
|
if rej > crej or ghs < chash or avg < chash or tmp > ctemp: status = 'crit'
|
||||||
|
elif rej > wrej or ghs < whash or avg < whash or tmp > wtemp: status = 'warn'
|
||||||
|
|
||||||
|
DEBUG(f"{rej} | {ghs} | {avg} | {brd} | {tmp}")
|
||||||
|
|
||||||
|
return status
|
||||||
|
|
||||||
|
|
||||||
|
def gettype(self, ip):
|
||||||
|
info = self.rtcp(ip, 4028, '{"command": "version"}')
|
||||||
|
|
||||||
|
if not info:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if CONF.get('debug') and info:
|
||||||
|
info1 = self.rtcp(ip, 4028, '{"command": "stats"}')
|
||||||
|
info2 = self.rtcp(ip, 4028, '{"command": "version"}')
|
||||||
|
info3 = self.rtcp(ip, 4028, '{"command": "summary"}')
|
||||||
|
info4 = self.rtcp(ip, 4028, '{"command": "pools"}')
|
||||||
|
info5 = self.rtcp(ip, 4028, '{"command": "devdetails"}')
|
||||||
|
info6 = self.rtcp(ip, 4028, '{"command": "get_version"}')
|
||||||
|
info7 = self.rtcp(ip, 4028, '{"command": "status"}')
|
||||||
|
info8 = self.rtcp(ip, 4028, '{"command": "get_miner_info"}')
|
||||||
|
info9 = self.rtcp(ip, 4028, '{"command": "devs"}')
|
||||||
|
info10 = self.rtcp(ip, 4028, '{"command": "notify"}')
|
||||||
|
info11 = self.rtcp(ip, 4028, '{"command": "coin"}')
|
||||||
|
info12 = self.rtcp(ip, 4028, '{"command": "edevs"}')
|
||||||
|
info13 = self.rtcp(ip, 4028, '{"command": "estats"}')
|
||||||
|
|
||||||
|
DEBUG(f"[{ip}] {info1}")
|
||||||
|
DEBUG(f"[{ip}] {info2}")
|
||||||
|
DEBUG(f"[{ip}] {info3}")
|
||||||
|
DEBUG(f"[{ip}] {info4}")
|
||||||
|
DEBUG(f"[{ip}] {info5}")
|
||||||
|
DEBUG(f"[{ip}] {info6}")
|
||||||
|
DEBUG(f"[{ip}] {info7}")
|
||||||
|
DEBUG(f"[{ip}] {info8}")
|
||||||
|
DEBUG(f"[{ip}] {info9}")
|
||||||
|
DEBUG(f"[{ip}] {info10}")
|
||||||
|
DEBUG(f"[{ip}] {info11}")
|
||||||
|
DEBUG(f"[{ip}] {info12}")
|
||||||
|
DEBUG(f"[{ip}] {info13}")
|
||||||
|
|
||||||
|
info += self.rtcp(ip, 4028, '{"command": "devdetails"}')
|
||||||
|
|
||||||
|
# Bitmain
|
||||||
|
if "Antminer" in info:
|
||||||
|
if "S19J Pro" in info:
|
||||||
|
return ASICs.AntminerS19JPro.value
|
||||||
|
if "S19J" in info:
|
||||||
|
return ASICs.AntminerS19J.value
|
||||||
|
if "S19 Pro" in info:
|
||||||
|
return ASICs.AntminerS19Pro.value
|
||||||
|
if "S19" in info:
|
||||||
|
return ASICs.AntminerS19.value
|
||||||
|
|
||||||
|
if "L3+" in info:
|
||||||
|
return ASICs.AntminerL3plus.value
|
||||||
|
if "L7" in info:
|
||||||
|
return ASICs.AntminerL3plus.value
|
||||||
|
|
||||||
|
if "T17+" in info:
|
||||||
|
return ASICs.AntminerT17plus.value
|
||||||
|
if "T17" in info:
|
||||||
|
return ASICs.AntminerT17.value
|
||||||
|
|
||||||
|
# Avalon
|
||||||
|
if "Avalon" in info:
|
||||||
|
if "1066" in info:
|
||||||
|
return ASICs.Avalon1066.value
|
||||||
|
if "1126" in info:
|
||||||
|
return ASICs.Avalon1126.value
|
||||||
|
|
||||||
|
# WhatsMiner
|
||||||
|
if "bitmicro" in info:
|
||||||
|
if "M20s":
|
||||||
|
return ASICs.WhatsMinerM20s.value
|
||||||
|
if "M21s":
|
||||||
|
return ASICs.WhatsMinerM21s.value
|
||||||
|
if "M30":
|
||||||
|
return ASICs.WhatsMinerM30.value
|
||||||
|
if "M31":
|
||||||
|
return ASICs.WhatsMinerM31.value
|
||||||
|
if "M50":
|
||||||
|
return ASICs.WhatsMinerM50.value
|
||||||
|
|
||||||
|
# Innosilicon
|
||||||
|
if False:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def rtcp(self, ip, port, cmd):
|
||||||
|
cmd = cmd.encode('utf-8')
|
||||||
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
||||||
|
try:
|
||||||
|
s.connect((ip, port))
|
||||||
|
s.sendall(cmd)
|
||||||
|
tdata = []
|
||||||
|
while 1:
|
||||||
|
data = s.recv(1024*32)
|
||||||
|
if not data: break
|
||||||
|
tdata.append(data)
|
||||||
|
|
||||||
|
data = b''.join(tdata).decode("utf-8")
|
||||||
|
|
||||||
|
return data
|
||||||
|
except OSError as e:
|
||||||
|
# ERR(e)
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def check_AntminerL3(self, ip):
|
||||||
|
# f'http://{ip}/cgi-bin/get_miner_status.cgi'
|
||||||
|
# f'http://{ip}/cgi-bin/get_miner_conf.cgi'
|
||||||
|
summ = self.rtcp(ip, 4028, "summary")
|
||||||
|
stat = self.rtcp(ip, 4028, "estats")
|
||||||
|
info = self.rtcp(ip, 4028, "version")
|
||||||
|
|
||||||
|
if summ and stat:
|
||||||
|
mac = json.loads(requests.get(
|
||||||
|
f'http://{ip}/cgi-bin/get_system_info.cgi',
|
||||||
|
auth=requests.auth.HTTPDigestAuth('root', 'root')).text)['macaddr'].lower()
|
||||||
|
|
||||||
|
log = f"{info} ||| {summ} ||| {stat}"
|
||||||
|
|
||||||
|
return log, mac
|
||||||
|
|
||||||
|
return False, False
|
||||||
|
|
||||||
|
|
||||||
|
def check_AntminerS19(self, ip):
|
||||||
|
# f'http://{ip}/cgi-bin/get_miner_status.cgi'
|
||||||
|
# f'http://{ip}/cgi-bin/get_miner_conf.cgi'
|
||||||
|
summ = self.rtcp(ip, 4028, "summary")
|
||||||
|
stat = self.rtcp(ip, 4028, "stats")
|
||||||
|
info = self.rtcp(ip, 4028, "version")
|
||||||
|
|
||||||
|
if summ and stat:
|
||||||
|
mac = json.loads(requests.get(
|
||||||
|
f'http://{ip}/cgi-bin/get_network_info.cgi',
|
||||||
|
auth=requests.auth.HTTPDigestAuth('root', 'root')).text)['macaddr'].lower()
|
||||||
|
|
||||||
|
log = f"{info} ||| {summ} ||| {stat}"
|
||||||
|
|
||||||
|
return log, mac
|
||||||
|
|
||||||
|
return False, False
|
||||||
|
|
||||||
|
|
||||||
|
def check_Avalon1XXX(self, ip):
|
||||||
|
summ = self.rtcp(ip, 4028, "summary")
|
||||||
|
stat = self.rtcp(ip, 4028, "estats")
|
||||||
|
info = self.rtcp(ip, 4028, "version")
|
||||||
|
|
||||||
|
if summ and stat:
|
||||||
|
mac = self.getreg(r'(MAC=([A-z0-9]+))', info)
|
||||||
|
if mac: mac = ':'.join(mac[i:i+2] for i in range(0,12,2))
|
||||||
|
else: mac = ':'
|
||||||
|
|
||||||
|
log = f"{info} ||| {summ} ||| {stat}"
|
||||||
|
|
||||||
|
return log, mac
|
||||||
|
|
||||||
|
return False, False
|
||||||
|
|
||||||
|
|
||||||
|
def check(self, ip):
|
||||||
|
try:
|
||||||
|
type = self.gettype(ip)
|
||||||
|
|
||||||
|
if not type:
|
||||||
|
return (False, False, False, False, False), False
|
||||||
|
|
||||||
|
if type in [ASICs.AntminerL3plus.value]:
|
||||||
|
log, mac = self.check_AntminerL3(ip)
|
||||||
|
status = self.getstat_AntminerL3(log, type)
|
||||||
|
elif type in [ASICs.Avalon1066.value, ASICs.Avalon1126.value]:
|
||||||
|
log, mac = self.check_Avalon1XXX(ip)
|
||||||
|
status = self.getstat_Avalon1XXX(log, type)
|
||||||
|
elif type in [ASICs.AntminerS19.value, ASICs.AntminerS19J.value, ASICs.AntminerS19Pro.value, ASICs.AntminerS19JPro.value]:
|
||||||
|
log, mac = self.check_AntminerS19(ip)
|
||||||
|
status = self.getstat_AntminerS19(log, type)
|
||||||
|
|
||||||
|
return (ip, mac, type, int(time()), log), status
|
||||||
|
except Exception as e:
|
||||||
|
WARN('Minor error in <lambda>Thread: ' + str(e))
|
||||||
|
return (False, False, False, False, False), False
|
||||||
|
|
||||||
|
|
||||||
|
def checkall(self):
|
||||||
|
ips = []
|
||||||
|
|
||||||
|
with self.conn.cursor(dictionary=True) as c:
|
||||||
|
c.execute("SELECT * FROM `ips` WHERE `location` = %s", (CONF.get('location'),))
|
||||||
|
|
||||||
|
for ip in c.fetchall():
|
||||||
|
ip = ip['ip']
|
||||||
|
if ip.isalpha() or not ip.find('-'):
|
||||||
|
ips.append(ip)
|
||||||
|
else:
|
||||||
|
i_p = ip.split('.')
|
||||||
|
i_p_new = []
|
||||||
|
for i in i_p:
|
||||||
|
i_p_new.append(i.split('-'))
|
||||||
|
i_p_new[-1][0] = int(i_p_new[-1][0])
|
||||||
|
i_p_new[-1][-1] = int(i_p_new[-1][-1])
|
||||||
|
|
||||||
|
for i1 in range(i_p_new[0][0], i_p_new[0][-1] + 1):
|
||||||
|
for i2 in range(i_p_new[1][0], i_p_new[1][-1] + 1):
|
||||||
|
for i3 in range(i_p_new[2][0], i_p_new[2][-1] + 1):
|
||||||
|
for i4 in range(i_p_new[3][0], i_p_new[3][-1] + 1):
|
||||||
|
ips.append(f"{i1}.{i2}.{i3}.{i4}")
|
||||||
|
|
||||||
|
Tca = []
|
||||||
|
ca = []
|
||||||
|
for ip in ips:
|
||||||
|
T = Thread(target=lambda x: ca.append(self.check(x)), args=(ip,))
|
||||||
|
Tca.append(T)
|
||||||
|
for i in range(len(Tca)):
|
||||||
|
Tca[i].start()
|
||||||
|
for i in range(len(Tca)):
|
||||||
|
Tca[i].join()
|
||||||
|
|
||||||
|
with self.conn.cursor() as c:
|
||||||
|
c.execute(f"DELETE FROM `laststate` WHERE `location` = %s", (CONF.get('location'),))
|
||||||
|
|
||||||
|
for i in ca:
|
||||||
|
i, status = i
|
||||||
|
if i[4]:
|
||||||
|
c.execute(f"INSERT INTO `asiclogs` (`ip`, `mac`, `type`, `time`, `log`) VALUES (%s, %s, %s, %s, %s)", i)
|
||||||
|
|
||||||
|
if i[1] == ':':
|
||||||
|
continue
|
||||||
|
c.execute(
|
||||||
|
f"INSERT INTO `laststate` (`ip`, `mac`, `type`, `location`, `status`, `time`) VALUES (%s, %s, %s, %s, %s, %s)",
|
||||||
|
(i[0], i[1], i[2], CONF.get('location'), status, int(time())))
|
||||||
|
|
||||||
|
self.conn.commit()
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
SUCC("Checker started!")
|
||||||
|
sleep(5)
|
||||||
|
|
||||||
|
self.conn = connect(
|
||||||
|
host=CONF.get('db', 'host'),
|
||||||
|
user=CONF.get('db', 'user'),
|
||||||
|
password=CONF.get('db', 'password'),
|
||||||
|
database=CONF.get('db', 'name'))
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
if self.event.is_set():
|
||||||
|
SUCC("Checker stopped!")
|
||||||
|
break
|
||||||
|
|
||||||
|
self.checkall()
|
||||||
|
sleep(10)
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
3
checkmem
Executable file
3
checkmem
Executable file
@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
watch -n 0.1 'ps -o vsize,command,pcpu ax | sort -b -k3 -r | grep -E -i -w "[0-9]+ python"'
|
43
cleaner.py
Normal file
43
cleaner.py
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
from threading import Thread
|
||||||
|
from mysql.connector import connect, Error
|
||||||
|
from time import sleep, time
|
||||||
|
from macros import *
|
||||||
|
import os
|
||||||
|
|
||||||
|
|
||||||
|
class NThread(Thread):
|
||||||
|
conn = None
|
||||||
|
|
||||||
|
def __init__(self, event):
|
||||||
|
super(NThread, self).__init__()
|
||||||
|
self.event = event
|
||||||
|
|
||||||
|
|
||||||
|
def clean(self):
|
||||||
|
with self.conn.cursor() as c:
|
||||||
|
c.execute(f"DELETE FROM `asiclogs` WHERE `time` < UNIX_TIMESTAMP() - %s", (CONF.get('db-log-days'),))
|
||||||
|
c.execute(f"DELETE FROM `laststate` WHERE `time` < UNIX_TIMESTAMP() - 30")
|
||||||
|
|
||||||
|
self.conn.commit()
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
SUCC("Cleaner started!")
|
||||||
|
|
||||||
|
self.conn = connect(
|
||||||
|
host=CONF.get('db', 'host'),
|
||||||
|
user=CONF.get('db', 'user'),
|
||||||
|
password=CONF.get('db', 'password'),
|
||||||
|
database=CONF.get('db', 'name'))
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
if self.event.is_set():
|
||||||
|
SUCC('Cleaner stopped!')
|
||||||
|
break
|
||||||
|
|
||||||
|
self.clean()
|
||||||
|
sleep(5)
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
20
colors.py
Normal file
20
colors.py
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
RESET = "\033[0m"
|
||||||
|
BOLD = "\033[1m"
|
||||||
|
|
||||||
|
BLACK = "\033[30m"
|
||||||
|
RED = "\033[31m"
|
||||||
|
GREEN = "\033[32m"
|
||||||
|
YELLOW = "\033[33m"
|
||||||
|
BLUE = "\033[34m"
|
||||||
|
MAGENTA = "\033[35m"
|
||||||
|
CYAN = "\033[36m"
|
||||||
|
WHITE = "\033[37m"
|
||||||
|
|
||||||
|
BGBLACK = "\033[40m"
|
||||||
|
BGRED = "\033[41m"
|
||||||
|
BGGREEN = "\033[42m"
|
||||||
|
BGYELLOW = "\033[43m"
|
||||||
|
BGBLUE = "\033[44m"
|
||||||
|
BGMAGENTA = "\033[45m"
|
||||||
|
BGCYAN = "\033[46m"
|
||||||
|
BGWHITE = "\033[47m"
|
73
config.json
Normal file
73
config.json
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
{
|
||||||
|
"license": "00001111-2222-3333-4444-555566667777",
|
||||||
|
"location": "Underground",
|
||||||
|
"logging": true,
|
||||||
|
"db-log-days": 7,
|
||||||
|
"debug": true,
|
||||||
|
"updates": false,
|
||||||
|
"port": 9070,
|
||||||
|
"telegram": {
|
||||||
|
"enable": true,
|
||||||
|
"channel": -1001871940722,
|
||||||
|
"token": "5953600362:AAELgZr0Ldstf0omK43zKrAzGXOuMGowcf8",
|
||||||
|
"normal-notify": true,
|
||||||
|
"warn-notify": false,
|
||||||
|
"crit-notify": true
|
||||||
|
},
|
||||||
|
"db": {
|
||||||
|
"host": "127.0.0.1",
|
||||||
|
"user": "root",
|
||||||
|
"password": "0",
|
||||||
|
"name": "myasics"
|
||||||
|
},
|
||||||
|
"asics": {
|
||||||
|
"L3plus": {
|
||||||
|
"crit": {
|
||||||
|
"hash": 40000,
|
||||||
|
"temp": 90,
|
||||||
|
"rejects": 2.5
|
||||||
|
},
|
||||||
|
"warn": {
|
||||||
|
"hash": 45000,
|
||||||
|
"temp": 80,
|
||||||
|
"rejects": 1.5
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"S19": {
|
||||||
|
"crit": {
|
||||||
|
"hash": 80000,
|
||||||
|
"temp": 90,
|
||||||
|
"rejects": 2.5
|
||||||
|
},
|
||||||
|
"warn": {
|
||||||
|
"hash": 85000,
|
||||||
|
"temp": 80,
|
||||||
|
"rejects": 1.5
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Avalon1066": {
|
||||||
|
"crit": {
|
||||||
|
"hash": 42000,
|
||||||
|
"temp": 90,
|
||||||
|
"rejects": 2.5
|
||||||
|
},
|
||||||
|
"warn": {
|
||||||
|
"hash": 46000,
|
||||||
|
"temp": 80,
|
||||||
|
"rejects": 1.5
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"Avalon1126": {
|
||||||
|
"crit": {
|
||||||
|
"hash": 64000,
|
||||||
|
"temp": 90,
|
||||||
|
"rejects": 2.5
|
||||||
|
},
|
||||||
|
"warn": {
|
||||||
|
"hash": 66000,
|
||||||
|
"temp": 80,
|
||||||
|
"rejects": 1.5
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
3
config.py
Normal file
3
config.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
APPNAME='Bit.ASICmon'
|
||||||
|
VERSION='v0.1.1a'
|
||||||
|
DEBUGGING=True
|
12
docker-compose.yml
Normal file
12
docker-compose.yml
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
version: '3.1'
|
||||||
|
|
||||||
|
services:
|
||||||
|
|
||||||
|
db:
|
||||||
|
image: mysql
|
||||||
|
command: --default-authentication-plugin=mysql_native_password
|
||||||
|
restart: always
|
||||||
|
environment:
|
||||||
|
MYSQL_ROOT_PASSWORD: 0
|
||||||
|
ports:
|
||||||
|
- 3306:3306
|
223
index.js
Normal file
223
index.js
Normal file
@ -0,0 +1,223 @@
|
|||||||
|
const sprintf = (...[string, ...args]) => {
|
||||||
|
return string.replace(/{(\d+)}/g, (match, number) => args[number] ?? match);
|
||||||
|
}
|
||||||
|
|
||||||
|
const getJSON = async url => {
|
||||||
|
const resp = await fetch(url)
|
||||||
|
const json = await resp.json()
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
|
||||||
|
const createPopup = async id => {
|
||||||
|
const asics = await getJSON('/api/asictypes')
|
||||||
|
|
||||||
|
asicoptions = ''
|
||||||
|
asics.forEach(elem => {
|
||||||
|
asicoptions += sprintf('<option value="{0}">{1}</option>', elem['key'], elem['value'])
|
||||||
|
})
|
||||||
|
|
||||||
|
document.querySelector('body').innerHTML += sprintf(popup, id, asicoptions, 'content')
|
||||||
|
}
|
||||||
|
|
||||||
|
const deletePopup = id => {
|
||||||
|
document.querySelector('#popup-' + id).remove()
|
||||||
|
}
|
||||||
|
|
||||||
|
const update = async () => {
|
||||||
|
let inner = await getStats()
|
||||||
|
document.querySelector('.grid').innerHTML = inner
|
||||||
|
}
|
||||||
|
|
||||||
|
const datafn = async () => {
|
||||||
|
// return await getJSON('/api/webinit')
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = datafn()
|
||||||
|
|
||||||
|
console.log(data)
|
||||||
|
|
||||||
|
const cols = 40
|
||||||
|
const rows = 40
|
||||||
|
|
||||||
|
|
||||||
|
const css =
|
||||||
|
`<style>
|
||||||
|
@import url('https://fonts.googleapis.com/css2?family=Source+Code+Pro:wght@400;500;600;700&display=swap');
|
||||||
|
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
font-family: 'Source Code Pro', monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
input,
|
||||||
|
select {
|
||||||
|
border: 1px solid #fff;
|
||||||
|
background: #000;
|
||||||
|
color: #fff;
|
||||||
|
padding: 5px 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.green {
|
||||||
|
background: #0b0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.yellow {
|
||||||
|
background: #ff0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.red {
|
||||||
|
background: #f11 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.darkred {
|
||||||
|
background: #300 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
height: 60px;
|
||||||
|
background: #222;
|
||||||
|
text-align: center;
|
||||||
|
margin: 0 0 1px 0;
|
||||||
|
display: grid;
|
||||||
|
justify-items: center;
|
||||||
|
align-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-info {
|
||||||
|
color: #bbb;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(${cols}, 30px);
|
||||||
|
grid-template-rows: repeat(${rows}, 30px);
|
||||||
|
grid-gap: 1px;
|
||||||
|
margin: auto;
|
||||||
|
width: max-content;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cell {
|
||||||
|
position: relative;
|
||||||
|
background: #111;
|
||||||
|
font-size: 9px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cell:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cell:hover::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
pointer-events: none;
|
||||||
|
background: #ffffff33;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
display: grid;
|
||||||
|
justify-items: center;
|
||||||
|
align-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-bg {
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: #00000033;
|
||||||
|
z-index: 98;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-window {
|
||||||
|
position: relative;
|
||||||
|
height: max-content;
|
||||||
|
max-height: 90vh;
|
||||||
|
width: max-content;
|
||||||
|
max-width: 90vw;
|
||||||
|
padding: 40px;
|
||||||
|
background: #222;
|
||||||
|
z-index: 99;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
`
|
||||||
|
|
||||||
|
const header =
|
||||||
|
'<div class="header">'
|
||||||
|
+ '<div class="header-info">'
|
||||||
|
+ 'Всего устройств: {0}, Предупреждений, {1}, Ошибки: {2}, В сети: {3}, Не в сети: {4}'
|
||||||
|
+ '</div>'
|
||||||
|
+ '</div>'
|
||||||
|
|
||||||
|
|
||||||
|
const cell =
|
||||||
|
'<div class="cell {1}" id="cell-{0}" onclick="createPopup({0})">'
|
||||||
|
+ '{2}'
|
||||||
|
+ '</div>'
|
||||||
|
|
||||||
|
const grid =
|
||||||
|
'<div class="grid">'
|
||||||
|
+ '{0}'
|
||||||
|
+ '</div>'
|
||||||
|
|
||||||
|
const popup =
|
||||||
|
'<div class="popup" id="popup-{0}">'
|
||||||
|
+ '<div class="popup-bg" onclick="deletePopup({0})"></div>'
|
||||||
|
+ '<div class="popup-window">'
|
||||||
|
+ '<div class="row">'
|
||||||
|
+ '<select>'
|
||||||
|
+ '<option value="none">Свободно</option>'
|
||||||
|
+ '{1}'
|
||||||
|
+ '</select>'
|
||||||
|
+ '<input type="text" placeholder="Адрес...">'
|
||||||
|
+ '</div>'
|
||||||
|
+ '{2}'
|
||||||
|
+ '</div>'
|
||||||
|
+ '</div>'
|
||||||
|
|
||||||
|
const getStats = async () => {
|
||||||
|
let cells = ''
|
||||||
|
let info = await getJSON('/curstatus.json')
|
||||||
|
for(let i = 1; i <= cols * rows; i++) {
|
||||||
|
if(info.asics[i]?.status == "ok")
|
||||||
|
cells += sprintf(cell, i, 'green', info.asics[i]?.hashrate)
|
||||||
|
else if(info.asics[i]?.status == "warn")
|
||||||
|
cells += sprintf(cell, i, 'yellow', info.asics[i]?.hashrate)
|
||||||
|
else if(info.asics[i]?.status == "crit")
|
||||||
|
cells += sprintf(cell, i, 'red', info.asics[i]?.hashrate)
|
||||||
|
else if(info.asics[i]?.status == "off")
|
||||||
|
cells += sprintf(cell, i, 'darkred', '')
|
||||||
|
else
|
||||||
|
cells += sprintf(cell, i, '', '')
|
||||||
|
}
|
||||||
|
return cells
|
||||||
|
}
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
document.querySelector('body').innerHTML += css
|
||||||
|
document.querySelector('body').innerHTML += sprintf(header)
|
||||||
|
|
||||||
|
let inner = sprintf(grid, await getStats())
|
||||||
|
document.querySelector('body').innerHTML += inner
|
||||||
|
|
||||||
|
setInterval(async () => {
|
||||||
|
await update()
|
||||||
|
}, 10000)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
run()
|
76
license.py
Normal file
76
license.py
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
from threading import Thread
|
||||||
|
from time import sleep, time
|
||||||
|
from macros import *
|
||||||
|
from Crypto.Cipher import AES
|
||||||
|
from Crypto.Random import get_random_bytes
|
||||||
|
from hashlib import sha256
|
||||||
|
from base64 import b64decode
|
||||||
|
from datetime import datetime
|
||||||
|
from urllib import request
|
||||||
|
import os, json, requests
|
||||||
|
|
||||||
|
|
||||||
|
class LThread(Thread):
|
||||||
|
_FIRST = True
|
||||||
|
|
||||||
|
|
||||||
|
def __init__(self, event):
|
||||||
|
super(LThread, self).__init__()
|
||||||
|
self.event = event
|
||||||
|
|
||||||
|
|
||||||
|
def check(self):
|
||||||
|
url = 'https://license.bitheaven.ru/api/v1/license.check'
|
||||||
|
post = {
|
||||||
|
'key': CONF.get('license'),
|
||||||
|
'loc': CONF.get('location'),
|
||||||
|
'time': int(time()),
|
||||||
|
'data': get_random_bytes(16)
|
||||||
|
}
|
||||||
|
curtime = post['time']
|
||||||
|
|
||||||
|
try:
|
||||||
|
req = requests.post(url, data=post).json()
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if req['error']:
|
||||||
|
ERR(f"Server error! {req['msg']}")
|
||||||
|
return True
|
||||||
|
|
||||||
|
key = sha256(post['loc'].encode('utf-8')).hexdigest()[0:32].encode("utf-8")
|
||||||
|
iv = post['data']
|
||||||
|
aes = AES.new(key, AES.MODE_OFB, iv=iv)
|
||||||
|
msg = aes.decrypt(b64decode(req['data'])).decode('unicode_escape').strip()
|
||||||
|
|
||||||
|
until = datetime.fromtimestamp(req['until']).strftime('%d.%m.%Y')
|
||||||
|
|
||||||
|
if self._FIRST:
|
||||||
|
INFO(f'License paid until: {until}')
|
||||||
|
self._FIRST = False
|
||||||
|
|
||||||
|
return f"{CONF.get('license')}{curtime}" == msg
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
while 1:
|
||||||
|
if self.event.is_set():
|
||||||
|
break
|
||||||
|
|
||||||
|
cur = self.check()
|
||||||
|
|
||||||
|
if not cur:
|
||||||
|
try:
|
||||||
|
request.urlopen('http://1.1.1.1', timeout=2)
|
||||||
|
CRIT("License wrong or expired!")
|
||||||
|
WARN("Get it on https://bhev.ru/glfbam")
|
||||||
|
sleep(60)
|
||||||
|
except:
|
||||||
|
ERR("No access to network!")
|
||||||
|
# os._exit(1)
|
||||||
|
|
||||||
|
sleep(5)
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
BIN
log/.2023-05-10.log.swp
Normal file
BIN
log/.2023-05-10.log.swp
Normal file
Binary file not shown.
97
macros.py
Normal file
97
macros.py
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from colors import *
|
||||||
|
from platform import system
|
||||||
|
import inspect
|
||||||
|
import json, os, config
|
||||||
|
|
||||||
|
|
||||||
|
__DEBUGGING_ALERT = False
|
||||||
|
|
||||||
|
MUSTDIE = system().lower() == 'windows'
|
||||||
|
LINUX = system().lower() == 'linux'
|
||||||
|
MACOS = system().lower() == 'darwin'
|
||||||
|
|
||||||
|
|
||||||
|
class CONF():
|
||||||
|
def __init__(self):
|
||||||
|
CONF.config = json.load(open('config.json'))
|
||||||
|
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get(*params):
|
||||||
|
conf = CONF.config
|
||||||
|
|
||||||
|
for param in params:
|
||||||
|
if param in conf.keys():
|
||||||
|
conf = conf[param]
|
||||||
|
|
||||||
|
if conf:
|
||||||
|
return conf
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
CONF()
|
||||||
|
|
||||||
|
|
||||||
|
def __GET_TIME():
|
||||||
|
return datetime.now().strftime("%H:%M:%S.%f")
|
||||||
|
|
||||||
|
|
||||||
|
def __GET_DATE():
|
||||||
|
return datetime.now().strftime("%Y-%m-%d")
|
||||||
|
|
||||||
|
|
||||||
|
def __ADD_TO_LOG(str):
|
||||||
|
if CONF.get('logging'):
|
||||||
|
if not os.path.isdir('log'):
|
||||||
|
os.mkdir('log')
|
||||||
|
log = open(f'log/{__GET_DATE()}.log', 'a')
|
||||||
|
log.write(str + '\n')
|
||||||
|
log.close()
|
||||||
|
|
||||||
|
|
||||||
|
def __PRINT_LOG(str, log):
|
||||||
|
if CONF.get('debug') and config.DEBUGGING:
|
||||||
|
st = inspect.stack()[2]
|
||||||
|
caller = st.filename.split('/')[-1].split('.')[0]
|
||||||
|
callerline = st.lineno
|
||||||
|
str = f'[{__GET_TIME()}] [{caller}:{callerline}] {str}'
|
||||||
|
log = f'[{caller}:{callerline}] {log}'
|
||||||
|
else:
|
||||||
|
str = f'[{__GET_TIME()}] {str}'
|
||||||
|
|
||||||
|
print(str)
|
||||||
|
__ADD_TO_LOG(log)
|
||||||
|
|
||||||
|
|
||||||
|
def INFO(s = ''):
|
||||||
|
__PRINT_LOG(f"[INFO] {str(s)}{RESET}", str(s))
|
||||||
|
|
||||||
|
|
||||||
|
def DEBUG(s = ''):
|
||||||
|
global __DEBUGGING_ALERT
|
||||||
|
|
||||||
|
if not CONF.get('debug'):
|
||||||
|
return None
|
||||||
|
|
||||||
|
if config.DEBUGGING:
|
||||||
|
__PRINT_LOG(f"{MAGENTA}[DEBUG]{RESET} {BGBLUE}{YELLOW}{BOLD}{str(s)}{RESET}", str(s))
|
||||||
|
elif not __DEBUGGING_ALERT:
|
||||||
|
WARN('DEBUGGING DISABLED BY OWNER!')
|
||||||
|
__DEBUGGING_ALERT = True
|
||||||
|
|
||||||
|
|
||||||
|
def SUCC(s = ''):
|
||||||
|
__PRINT_LOG(f"{GREEN}[SUCCESS]{RESET} {GREEN}{str(s)}{RESET}", str(s))
|
||||||
|
|
||||||
|
|
||||||
|
def WARN(s = ''):
|
||||||
|
__PRINT_LOG(f"{YELLOW}[WARN]{RESET} {YELLOW}{str(s)}{RESET}", str(s))
|
||||||
|
|
||||||
|
|
||||||
|
def ERR(s = ''):
|
||||||
|
__PRINT_LOG(f"{RED}[ERROR]{RESET} {RED}{str(s)}{RESET}", str(s))
|
||||||
|
|
||||||
|
|
||||||
|
def CRIT(s = ''):
|
||||||
|
__PRINT_LOG(f"{RED}{BOLD}[CRITICAL]{RESET} {BGRED}{WHITE} {str(s)} {RESET}", str(s))
|
95
main.py
Executable file
95
main.py
Executable file
@ -0,0 +1,95 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
|
||||||
|
from updater import UThread
|
||||||
|
from license import LThread
|
||||||
|
from checker import CThread
|
||||||
|
from cleaner import NThread
|
||||||
|
from telegrambot import TGThread
|
||||||
|
from webserver import WSThread
|
||||||
|
from time import sleep
|
||||||
|
from macros import *
|
||||||
|
from colors import *
|
||||||
|
from threading import Event
|
||||||
|
import config, sys, os, asyncio
|
||||||
|
|
||||||
|
|
||||||
|
INFO('╔' + '═'*(len(config.APPNAME) + len(config.VERSION) + 3) + '╗')
|
||||||
|
INFO(f"║ {CYAN}{BOLD}{config.APPNAME}{RESET} {config.VERSION} ║")
|
||||||
|
INFO('╚' + '═'*(len(config.APPNAME) + len(config.VERSION) + 3) + '╝')
|
||||||
|
|
||||||
|
if CONF.get('debug'):
|
||||||
|
INFO('DEBUG TEST START ' + '*' * 20)
|
||||||
|
DEBUG(f"{RESET}{BLACK}*{RED}*{GREEN}*{YELLOW}*{BLUE}*{MAGENTA}*{CYAN}*{WHITE}*")
|
||||||
|
DEBUG(f"{RESET}{BOLD}{BLACK}*{RED}*{GREEN}*{YELLOW}*{BLUE}*{MAGENTA}*{CYAN}*{WHITE}*")
|
||||||
|
DEBUG(f"{RESET}{BGBLACK}*{BGRED}*{BGGREEN}*{BGYELLOW}*{BGBLUE}*{BGMAGENTA}*{BGCYAN}*{BGWHITE}*")
|
||||||
|
INFO('Test message')
|
||||||
|
DEBUG('Test message')
|
||||||
|
SUCC('Test message')
|
||||||
|
WARN('Test message')
|
||||||
|
ERR('Test message')
|
||||||
|
CRIT('Test message')
|
||||||
|
INFO('DEBUG TEST END ' + '*' * 22 + RESET)
|
||||||
|
|
||||||
|
|
||||||
|
eupdate = Event()
|
||||||
|
estop = Event()
|
||||||
|
|
||||||
|
lt = LThread(eupdate)
|
||||||
|
lt.daemon = True
|
||||||
|
lt.start()
|
||||||
|
|
||||||
|
wt = WSThread(eupdate)
|
||||||
|
wt.daemon = True
|
||||||
|
wt.start()
|
||||||
|
|
||||||
|
ct = CThread(eupdate)
|
||||||
|
ct.daemon = True
|
||||||
|
ct.start()
|
||||||
|
|
||||||
|
nt = NThread(eupdate)
|
||||||
|
nt.daemon = True
|
||||||
|
nt.start()
|
||||||
|
|
||||||
|
tt = TGThread(eupdate, estop)
|
||||||
|
tt.daemon = True
|
||||||
|
tt.start()
|
||||||
|
|
||||||
|
ut = UThread()
|
||||||
|
ut.daemon = True
|
||||||
|
ut.start()
|
||||||
|
|
||||||
|
updated = False
|
||||||
|
|
||||||
|
try:
|
||||||
|
while 1:
|
||||||
|
if lt.is_alive() and wt.is_alive() and ct.is_alive() and tt.is_alive() and ut.is_alive() and nt.is_alive():
|
||||||
|
sleep(1)
|
||||||
|
elif lt.is_alive() and wt.is_alive() and ct.is_alive() and tt.is_alive() and nt.is_alive():
|
||||||
|
eupdate.set()
|
||||||
|
if lt.is_alive(): lt.join()
|
||||||
|
if wt.is_alive(): wt.join()
|
||||||
|
if ct.is_alive(): ct.join()
|
||||||
|
if nt.is_alive(): nt.join()
|
||||||
|
if tt.is_alive(): tt.join()
|
||||||
|
updated = True
|
||||||
|
SUCC('Program updated, reboot!')
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
estop.set()
|
||||||
|
sleep(1)
|
||||||
|
CRIT('Program broken.')
|
||||||
|
os._exit(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
sys.stdout.flush()
|
||||||
|
print('\r', end='')
|
||||||
|
estop.set()
|
||||||
|
sleep(1)
|
||||||
|
CRIT('STOP')
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
estop.set()
|
||||||
|
sleep(1)
|
||||||
|
os._exit(1)
|
||||||
|
|
||||||
|
if updated:
|
||||||
|
os.execl(sys.argv[0], *sys.argv)
|
4
requirements.txt
Normal file
4
requirements.txt
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
flask
|
||||||
|
python-telegram-bot
|
||||||
|
pyinstaller
|
||||||
|
pyarmor<8.0.0
|
125
telegrambot.py
Normal file
125
telegrambot.py
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
from threading import Thread
|
||||||
|
from telegram import Bot, error
|
||||||
|
from asics import ASICs
|
||||||
|
from time import sleep
|
||||||
|
from mysql.connector import connect, Error
|
||||||
|
from macros import *
|
||||||
|
import os, json, asyncio
|
||||||
|
|
||||||
|
|
||||||
|
class TGThread(Thread):
|
||||||
|
conn = None
|
||||||
|
bot = None
|
||||||
|
asics = {}
|
||||||
|
|
||||||
|
|
||||||
|
def __init__(self, event, event2):
|
||||||
|
super(TGThread, self).__init__()
|
||||||
|
self.event = event
|
||||||
|
self.event2 = event2
|
||||||
|
|
||||||
|
|
||||||
|
async def sendmsg(self, msg):
|
||||||
|
return await self.bot.send_message(CONF.get('telegram', 'channel'), msg, 'MarkdownV2')
|
||||||
|
|
||||||
|
|
||||||
|
async def check(self):
|
||||||
|
msg = []
|
||||||
|
|
||||||
|
for i in self.asics.keys():
|
||||||
|
self.asics[i]['tooff'] -= 1
|
||||||
|
|
||||||
|
with self.conn.cursor(dictionary=True) as c:
|
||||||
|
c.execute(f"SELECT * FROM `laststate`")
|
||||||
|
|
||||||
|
for i in c.fetchall():
|
||||||
|
if not i['mac'] in self.asics.keys():
|
||||||
|
self.asics[i['mac']] = {}
|
||||||
|
self.asics[i['mac']]['alert'] = 0
|
||||||
|
|
||||||
|
msg.append(f"🟢 {i['location']} \| ASIC \"*{ASICs(i['type']).name}*\" `{i['ip']}` online\!")
|
||||||
|
|
||||||
|
self.asics[i['mac']]['tooff'] = 3
|
||||||
|
self.asics[i['mac']]['status'] = i['status']
|
||||||
|
|
||||||
|
if i['status'] == 'crit':
|
||||||
|
if CONF.get('telegram', 'crit-notify') and self.asics[i['mac']]['alert'] < 2:
|
||||||
|
msg.append(f"🔴 {i['location']} \| ASIC \"*{ASICs(i['type']).name}*\" `{i['ip']}` crit status\!")
|
||||||
|
self.asics[i['mac']]['alert'] = 2
|
||||||
|
elif i['status'] == 'warn':
|
||||||
|
if CONF.get('telegram', 'warn-notify') and self.asics[i['mac']]['alert'] < 1:
|
||||||
|
msg.append(f"🟡 {i['location']} \| ASIC \"*{ASICs(i['type']).name}*\" `{i['ip']}` warn status\!")
|
||||||
|
self.asics[i['mac']]['alert'] = 1
|
||||||
|
elif i['status'] == 'ok':
|
||||||
|
if CONF.get('telegram', 'normal-notify') and self.asics[i['mac']]['alert'] != 0:
|
||||||
|
msg.append(f"🟢 {i['location']} \| ASIC \"*{ASICs(i['type']).name}*\" `{i['ip']}` normal status\!")
|
||||||
|
self.asics[i['mac']]['alert'] = 0
|
||||||
|
|
||||||
|
if self.asics[i['mac']]['tooff'] <= 0:
|
||||||
|
msg.append(f"🔴 {i['location']} \| ASIC \"*{ASICs(i['type']).name}*\" `{i['ip']}` offline\!")
|
||||||
|
del self.asics[i['mac']]
|
||||||
|
|
||||||
|
if len(msg) > 0:
|
||||||
|
await self.sendmsg('\n'.join(msg))
|
||||||
|
|
||||||
|
|
||||||
|
async def runbot(self):
|
||||||
|
try:
|
||||||
|
res = await self.sendmsg('🟢 *Notify server online\!*')
|
||||||
|
INFO(f"Telegram channel ID: {res['chat']['id']}")
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
if self.event.is_set():
|
||||||
|
SUCC('Telegram stopped!')
|
||||||
|
await self.sendmsg('🟡 Updating software\! Recommend check all running instances.')
|
||||||
|
break
|
||||||
|
|
||||||
|
await self.check()
|
||||||
|
await asyncio.sleep(15)
|
||||||
|
|
||||||
|
except error.Forbidden as e:
|
||||||
|
WARN(str(e))
|
||||||
|
except error.TimedOut as e:
|
||||||
|
WARN(str(e))
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
# os._exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
async def stopalert(self):
|
||||||
|
while 1:
|
||||||
|
if self.event2.is_set():
|
||||||
|
await self.sendmsg('🔴 *Notify server offline\!*')
|
||||||
|
|
||||||
|
await asyncio.sleep(0.5)
|
||||||
|
|
||||||
|
|
||||||
|
async def main(self):
|
||||||
|
await asyncio.gather(self.stopalert(), self.runbot())
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
if CONF.get('telegram', 'enable'):
|
||||||
|
SUCC('Telegram started!')
|
||||||
|
|
||||||
|
self.conn = connect(
|
||||||
|
host=CONF.get('db', 'host'),
|
||||||
|
user=CONF.get('db', 'user'),
|
||||||
|
password=CONF.get('db', 'password'),
|
||||||
|
database=CONF.get('db', 'name'))
|
||||||
|
self.conn.autocommit = True
|
||||||
|
|
||||||
|
self.bot = Bot(CONF.get('telegram', 'token'))
|
||||||
|
|
||||||
|
asyncio.run(self.main())
|
||||||
|
else:
|
||||||
|
while 1:
|
||||||
|
if self.event.is_set():
|
||||||
|
SUCC('Telegram stopped!')
|
||||||
|
break
|
||||||
|
|
||||||
|
sleep(5)
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
54
updater.py
Normal file
54
updater.py
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
from threading import Thread
|
||||||
|
from macros import *
|
||||||
|
from config import *
|
||||||
|
from time import sleep
|
||||||
|
import urllib, os, requests
|
||||||
|
|
||||||
|
class UThread(Thread):
|
||||||
|
path = os.path.dirname(__file__)
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super(UThread, self).__init__()
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
global VERSION
|
||||||
|
|
||||||
|
try:
|
||||||
|
SUCC('Updater started!')
|
||||||
|
while 1:
|
||||||
|
if not CONF.get('updates'):
|
||||||
|
sleep(1)
|
||||||
|
continue
|
||||||
|
|
||||||
|
version = requests.get('https://mirror.bitheaven.ru/main/versions/Bit.ASICmon-a').text
|
||||||
|
|
||||||
|
if version == VERSION:
|
||||||
|
sleep(300)
|
||||||
|
continue
|
||||||
|
|
||||||
|
INFO('Update found! Downloading...')
|
||||||
|
if LINUX:
|
||||||
|
with urllib.request.urlopen("https://mirror.bitheaven.ru/main/archive/Bit.ASICmon-a_linux") as upd:
|
||||||
|
with open(self.path, "wb+") as f:
|
||||||
|
INFO('Installing update...')
|
||||||
|
f.write(upd.read())
|
||||||
|
INFO('Stopping process...')
|
||||||
|
break
|
||||||
|
elif MACOS:
|
||||||
|
with urllib.request.urlopen("https://mirror.bitheaven.ru/main/archive/Bit.ASICmon-a_macos") as upd:
|
||||||
|
with open(self.path, "wb+") as f:
|
||||||
|
INFO('Installing update...')
|
||||||
|
f.write(upd.read())
|
||||||
|
INFO('Stopping process...')
|
||||||
|
break
|
||||||
|
elif MUSTDIE:
|
||||||
|
with urllib.request.urlopen("https://mirror.bitheaven.ru/main/archive/Bit.ASICmon-a_mustdie.exe") as upd:
|
||||||
|
with open(self.path, "wb+") as f:
|
||||||
|
INFO('Installing update...')
|
||||||
|
f.write(upd.read())
|
||||||
|
INFO('Stopping process...')
|
||||||
|
break
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
107
webserver.py
Normal file
107
webserver.py
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
from flask import Flask, send_file, request
|
||||||
|
from time import sleep
|
||||||
|
from werkzeug.serving import make_server
|
||||||
|
from threading import Thread
|
||||||
|
from mysql.connector import connect, Error
|
||||||
|
from macros import *
|
||||||
|
import os, json, config, requests, logging
|
||||||
|
|
||||||
|
|
||||||
|
class WSThread(Thread):
|
||||||
|
app = Flask(__name__)
|
||||||
|
conn = None
|
||||||
|
|
||||||
|
def __init__(self, event):
|
||||||
|
super(WSThread, self).__init__()
|
||||||
|
self.event = event
|
||||||
|
|
||||||
|
log = logging.getLogger('werkzeug')
|
||||||
|
log.disabled = True
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/')
|
||||||
|
@app.route('/index')
|
||||||
|
def index():
|
||||||
|
return '<meta charset="UTF-8"><script defer src="/index.js"></script>'
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/index.js')
|
||||||
|
def indexjs():
|
||||||
|
return send_file('index.js')
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/curstatus.json')
|
||||||
|
def curstatus():
|
||||||
|
j = {}
|
||||||
|
j['info'] = {}
|
||||||
|
j['info']['asics'] = {}
|
||||||
|
|
||||||
|
return json.dumps(j), 200, {'ContentType':'application/json'}
|
||||||
|
|
||||||
|
|
||||||
|
def dbinit(self):
|
||||||
|
with self.conn.cursor() as c:
|
||||||
|
c.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS `ips` (
|
||||||
|
`id` INT AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
`ip` VARCHAR(64) NOT NULL,
|
||||||
|
`location` VARCHAR(64) NOT NULL
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
c.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS `laststate` (
|
||||||
|
`id` INT AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
`ip` VARCHAR(64) NOT NULL,
|
||||||
|
`mac` VARCHAR(32) NOT NULL,
|
||||||
|
`type` INT(11) NOT NULL,
|
||||||
|
`location` VARCHAR(64) NOT NULL,
|
||||||
|
`status` VARCHAR(32) NOT NULL,
|
||||||
|
`time` INT(11) NOT NULL
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
c.execute("""
|
||||||
|
CREATE TABLE IF NOT EXISTS `asiclogs` (
|
||||||
|
`id` INT AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
`ip` VARCHAR(64) NOT NULL,
|
||||||
|
`mac` VARCHAR(64) NOT NULL,
|
||||||
|
`type` VARCHAR(64) NOT NULL,
|
||||||
|
`time` INT(11) NOT NULL,
|
||||||
|
`log` TEXT NOT NULL
|
||||||
|
)
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.conn.commit()
|
||||||
|
|
||||||
|
|
||||||
|
def runweb(self):
|
||||||
|
self.server = make_server('0.0.0.0', CONF.get('port'), self.app)
|
||||||
|
self.server.serve_forever()
|
||||||
|
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
with connect(host=CONF.get('db', 'host'), user=CONF.get('db', 'user'), password=CONF.get('db', 'password')) as conn:
|
||||||
|
with conn.cursor() as c:
|
||||||
|
c.execute(f"CREATE DATABASE IF NOT EXISTS {CONF.get('db', 'name')}")
|
||||||
|
|
||||||
|
self.conn = connect(
|
||||||
|
host=CONF.get('db', 'host'),
|
||||||
|
user=CONF.get('db', 'user'),
|
||||||
|
password=CONF.get('db', 'password'),
|
||||||
|
database=CONF.get('db', 'name'))
|
||||||
|
self.dbinit()
|
||||||
|
|
||||||
|
web = Thread(target=self.runweb)
|
||||||
|
web.daemon = True
|
||||||
|
web.start()
|
||||||
|
|
||||||
|
SUCC(f"Web interface started at port {CONF.get('port')}!")
|
||||||
|
|
||||||
|
while not self.event.is_set():
|
||||||
|
sleep(1)
|
||||||
|
|
||||||
|
SUCC(f"Web server stopped!")
|
||||||
|
self.server.shutdown()
|
||||||
|
except Exception as e:
|
||||||
|
CRIT(str(e))
|
||||||
|
os._exit(1)
|
Loading…
Reference in New Issue
Block a user