fishyboteso/fishy/helper/helper.py

279 lines
7.5 KiB
Python
Raw Normal View History

import ctypes
import logging
import os
2021-03-24 16:58:35 +00:00
import shutil
import sys
import threading
import time
import traceback
import webbrowser
2021-05-09 07:05:51 +00:00
from hashlib import md5
2021-03-24 16:58:35 +00:00
from io import BytesIO
from threading import Thread
from uuid import uuid1
2021-05-09 07:05:51 +00:00
from zipfile import ZipFile
2021-05-09 07:05:51 +00:00
import requests
import winshell
from playsound import playsound
from win32com.client import Dispatch
from win32comext.shell import shell, shellcon
from win32gui import GetForegroundWindow, GetWindowText
import fishy
from fishy import web
from fishy.constants import libgps, lam2, fishyqr
from fishy.helper.config import config
def playsound_multiple(path, count=2):
if count < 1:
logging.debug("Please don't make me beep 0 times or less.")
return
def _ps_m():
2021-05-09 09:09:26 +00:00
for i in range(count - 1):
playsound(path, True)
playsound(path, False)
Thread(target=_ps_m).start()
def not_implemented():
logging.error("Not Implemented")
def empty_function():
pass
def wait_until(func):
while not func():
time.sleep(0.1)
def sign(x):
return -1 if x < 0 else 1
def open_web(website):
"""
Opens a website on browser,
uses multi-threading so that current thread doesnt get blocked
:param website: url
"""
logging.debug("opening web, please wait...")
Thread(target=lambda: webbrowser.open(website, new=2)).start()
2020-10-17 19:06:07 +00:00
def initialize_uid():
from .config import config
if config.get("uid") is not None:
return
new_uid = web.register_user()
if new_uid is not None:
config.set("uid", new_uid)
else:
logging.error("Couldn't register uid, some features might not work")
def _create_new_uid():
"""
Creates a unique id for user
"""
return md5(str(uuid1()).encode()).hexdigest()
def install_thread_excepthook():
"""
Workaround for sys.excepthook thread bug
https://bugs.python.org/issue1230540
(https://sourceforge.net/tracker/?func=detail&atid=105470&aid=1230540&group_id=5470).
Call once from __main__ before creating any threads.
If using psyco, call psycho.cannotcompile(threading.Thread.run)
since this replaces a new-style class method.
"""
run_old = threading.Thread.run
# noinspection PyBroadException
def run(*args, **kwargs):
try:
run_old(*args, **kwargs)
except (KeyboardInterrupt, SystemExit):
raise
except Exception:
sys.excepthook(*sys.exc_info())
threading.Thread.run = run
def unhandled_exception_logging(*exc_info):
text = "".join(traceback.format_exception(*exc_info))
logging.error("Unhandled exception: %s", text)
def manifest_file(rel_path):
"""
returns a file from the manifest files,
used to get the files which are installed along with the scripts
:param rel_path: relative path from `__init__.py`
:return: abs path of the file
"""
return os.path.join(os.path.dirname(fishy.__file__), rel_path)
2020-10-17 19:06:07 +00:00
def create_shortcut_first():
from .config import config
2020-10-17 19:06:07 +00:00
if not config.get("shortcut_created", False):
2020-10-13 18:23:27 +00:00
create_shortcut(False)
2020-10-17 19:06:07 +00:00
config.set("shortcut_created", True)
# noinspection PyBroadException
def create_shortcut(anti_ghosting: bool):
"""
creates a new shortcut on desktop
"""
try:
desktop = winshell.desktop()
path = os.path.join(desktop, "Fishybot ESO.lnk")
shell = Dispatch('WScript.Shell')
shortcut = shell.CreateShortCut(path)
if anti_ghosting:
shortcut.TargetPath = r"C:\Windows\System32\cmd.exe"
python_dir = os.path.join(os.path.dirname(sys.executable), "pythonw.exe")
shortcut.Arguments = f"/C start /affinity 1 /low {python_dir} -m fishy"
else:
shortcut.TargetPath = os.path.join(os.path.dirname(sys.executable), "python.exe")
shortcut.Arguments = "-m fishy"
shortcut.IconLocation = manifest_file("icon.ico")
shortcut.save()
logging.info("Shortcut created")
except Exception:
traceback.print_exc()
logging.error("Couldn't create shortcut")
2021-04-09 18:50:53 +00:00
def get_savedvarsdir():
# noinspection PyUnresolvedReferences
from win32com.shell import shell, shellcon
documents = shell.SHGetFolderPath(0, shellcon.CSIDL_PERSONAL, None, 0)
return os.path.join(documents, "Elder Scrolls Online", "live", "SavedVariables")
2021-03-24 16:58:35 +00:00
def get_addondir():
# noinspection PyUnresolvedReferences
from win32com.shell import shell, shellcon
documents = shell.SHGetFolderPath(0, shellcon.CSIDL_PERSONAL, None, 0)
2021-04-09 18:50:53 +00:00
return os.path.join(documents, "Elder Scrolls Online", "live", "Addons")
2021-03-24 16:58:35 +00:00
2021-03-26 09:34:01 +00:00
def addon_exists(name, url=None, v=None):
2021-03-24 16:58:35 +00:00
return os.path.exists(os.path.join(get_addondir(), name))
2021-03-26 09:34:01 +00:00
def get_addonversion(name, url=None, v=None):
if addon_exists(name):
txt = name + ".txt"
try:
with open(os.path.join(get_addondir(), name, txt)) as f:
for line in f:
if "AddOnVersion" in line:
return int(line.split(' ')[2])
except Exception:
pass
return 0
def install_required_addons(force=False):
addons_req = [libgps, lam2, fishyqr]
addon_version = config.get("addon_version", {})
installed = False
for addon in addons_req:
if force or (addon_exists(*addon) and
(addon[0] not in addon_version or (
addon[0] in addon_version and addon_version[addon[0]] < addon[2]))):
remove_addon(*addon)
install_addon(*addon)
addon_version[addon[0]] = addon[2]
installed = True
config.set("addon_version", addon_version)
if installed:
logging.info("Please make sure to enable \"Allow outdated addons\" in ESO")
# noinspection PyBroadException
2021-03-26 09:34:01 +00:00
def install_addon(name, url, v=None):
try:
2021-03-24 16:58:35 +00:00
r = requests.get(url, stream=True)
z = ZipFile(BytesIO(r.content))
z.extractall(path=get_addondir())
logging.info("Add-On " + name + " installed successfully!")
2021-03-26 09:35:00 +00:00
return 0
2021-05-09 09:48:35 +00:00
except Exception:
2021-05-09 09:09:26 +00:00
logging.error("Could not install Add-On " + name + ", try doing it manually")
traceback.print_exc()
2021-03-26 09:35:00 +00:00
return 1
2021-03-24 16:58:35 +00:00
2021-03-26 09:34:01 +00:00
def remove_addon(name, url=None, v=None):
2021-03-24 16:58:35 +00:00
try:
shutil.rmtree(os.path.join(get_addondir(), name))
2021-05-09 09:09:26 +00:00
logging.info("Add-On " + name + " removed!")
2021-03-24 16:58:35 +00:00
except FileNotFoundError:
pass
2021-05-09 09:48:35 +00:00
except PermissionError:
2021-05-09 09:09:26 +00:00
logging.error("Fishy has no permission to remove " + name + " Add-On")
2021-03-26 09:35:00 +00:00
return 1
return 0
def get_documents():
return shell.SHGetFolderPath(0, shellcon.CSIDL_PERSONAL, None, 0)
def restart():
os.execl(sys.executable, *([sys.executable] + sys.argv))
2021-02-14 18:10:11 +00:00
def log_raise(msg):
logging.error(msg)
raise Exception(msg)
2021-02-14 18:10:11 +00:00
def update():
from .config import config
config.delete("dont_ask_update")
restart()
def is_eso_active():
return GetWindowText(GetForegroundWindow()) == "Elder Scrolls Online"
# noinspection PyProtectedMember,PyUnresolvedReferences
def _get_id(thread):
# returns id of the respective thread
if hasattr(thread, '_thread_id'):
return thread._thread_id
for id, thread in threading._active.items():
if thread is thread:
return id
def kill_thread(thread):
thread_id = _get_id(thread)
res = ctypes.pythonapi.PyThreadState_SetAsyncExc(thread_id,
ctypes.py_object(SystemExit))
if res > 1:
ctypes.pythonapi.PyThreadState_SetAsyncExc(thread_id, 0)
print('Exception raise failure')