Init
This commit is contained in:
21
.editorconfig
Normal file
21
.editorconfig
Normal file
@ -0,0 +1,21 @@
|
||||
# http://editorconfig.org
|
||||
|
||||
root = true
|
||||
|
||||
[*]
|
||||
indent_style = space
|
||||
indent_size = 4
|
||||
trim_trailing_whitespace = true
|
||||
insert_final_newline = true
|
||||
charset = utf-8
|
||||
end_of_line = lf
|
||||
|
||||
[*.bat]
|
||||
indent_style = tab
|
||||
end_of_line = crlf
|
||||
|
||||
[LICENSE]
|
||||
insert_final_newline = false
|
||||
|
||||
[Makefile]
|
||||
indent_style = tab
|
15
.github/ISSUE_TEMPLATE.md
vendored
Normal file
15
.github/ISSUE_TEMPLATE.md
vendored
Normal file
@ -0,0 +1,15 @@
|
||||
* eRepublik script version:
|
||||
* Python version:
|
||||
* Operating System:
|
||||
|
||||
### Description
|
||||
|
||||
Describe what you were trying to get done.
|
||||
Tell us what happened, what went wrong, and what you expected to happen.
|
||||
|
||||
### What I Did
|
||||
|
||||
```
|
||||
Paste the command(s) you ran and the output.
|
||||
If there was a crash, please include the traceback here.
|
||||
```
|
102
.gitignore
vendored
Normal file
102
.gitignore
vendored
Normal file
@ -0,0 +1,102 @@
|
||||
# Byte-compiled / optimized / DLL files
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
|
||||
# C extensions
|
||||
*.so
|
||||
|
||||
# Distribution / packaging
|
||||
.Python
|
||||
env/
|
||||
build/
|
||||
develop-eggs/
|
||||
dist/
|
||||
downloads/
|
||||
eggs/
|
||||
.eggs/
|
||||
lib/
|
||||
lib64/
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
wheels/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
|
||||
# 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/
|
||||
.coverage
|
||||
.coverage.*
|
||||
.cache
|
||||
nosetests.xml
|
||||
coverage.xml
|
||||
*.cover
|
||||
.hypothesis/
|
||||
.pytest_cache/
|
||||
|
||||
# Translations
|
||||
*.mo
|
||||
*.pot
|
||||
|
||||
# Django stuff:
|
||||
*.log
|
||||
local_settings.py
|
||||
|
||||
# Flask stuff:
|
||||
instance/
|
||||
.webassets-cache
|
||||
|
||||
# Scrapy stuff:
|
||||
.scrapy
|
||||
|
||||
# Sphinx documentation
|
||||
docs/_build/
|
||||
|
||||
# PyBuilder
|
||||
target/
|
||||
|
||||
# Jupyter Notebook
|
||||
.ipynb_checkpoints
|
||||
|
||||
# pyenv
|
||||
.python-version
|
||||
|
||||
# celery beat schedule file
|
||||
celerybeat-schedule
|
||||
|
||||
# SageMath parsed files
|
||||
*.sage.py
|
||||
|
||||
# dotenv
|
||||
.env
|
||||
|
||||
# virtualenv
|
||||
.venv
|
||||
venv/
|
||||
ENV/
|
||||
|
||||
# Spyder project settings
|
||||
.spyderproject
|
||||
.spyproject
|
||||
|
||||
# Rope project settings
|
||||
.ropeproject
|
||||
|
||||
# mkdocs documentation
|
||||
/site
|
||||
|
||||
# mypy
|
||||
.mypy_cache/
|
16
.travis.yml
Normal file
16
.travis.yml
Normal file
@ -0,0 +1,16 @@
|
||||
# Config file for automatic testing at travis-ci.org
|
||||
|
||||
language: python
|
||||
python:
|
||||
- 3.6
|
||||
- 3.5
|
||||
- 3.4
|
||||
- 2.7
|
||||
|
||||
# Command to install dependencies, e.g. pip install -r requirements.txt --use-mirrors
|
||||
install: pip install -U tox-travis
|
||||
|
||||
# Command to run tests, e.g. python setup.py test
|
||||
script: tox
|
||||
|
||||
|
13
AUTHORS.rst
Normal file
13
AUTHORS.rst
Normal file
@ -0,0 +1,13 @@
|
||||
=======
|
||||
Credits
|
||||
=======
|
||||
|
||||
Development Lead
|
||||
----------------
|
||||
|
||||
* Eriks Karls <eriks@72.lv>
|
||||
|
||||
Contributors
|
||||
------------
|
||||
|
||||
None yet. Why not be the first?
|
128
CONTRIBUTING.rst
Normal file
128
CONTRIBUTING.rst
Normal file
@ -0,0 +1,128 @@
|
||||
.. highlight:: shell
|
||||
|
||||
============
|
||||
Contributing
|
||||
============
|
||||
|
||||
Contributions are welcome, and they are greatly appreciated! Every little bit
|
||||
helps, and credit will always be given.
|
||||
|
||||
You can contribute in many ways:
|
||||
|
||||
Types of Contributions
|
||||
----------------------
|
||||
|
||||
Report Bugs
|
||||
~~~~~~~~~~~
|
||||
|
||||
Report bugs at https://github.com/eeriks/erepublik_script/issues.
|
||||
|
||||
If you are reporting a bug, please include:
|
||||
|
||||
* Your operating system name and version.
|
||||
* Any details about your local setup that might be helpful in troubleshooting.
|
||||
* Detailed steps to reproduce the bug.
|
||||
|
||||
Fix Bugs
|
||||
~~~~~~~~
|
||||
|
||||
Look through the GitHub issues for bugs. Anything tagged with "bug" and "help
|
||||
wanted" is open to whoever wants to implement it.
|
||||
|
||||
Implement Features
|
||||
~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Look through the GitHub issues for features. Anything tagged with "enhancement"
|
||||
and "help wanted" is open to whoever wants to implement it.
|
||||
|
||||
Write Documentation
|
||||
~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
eRepublik script could always use more documentation, whether as part of the
|
||||
official eRepublik script docs, in docstrings, or even on the web in blog posts,
|
||||
articles, and such.
|
||||
|
||||
Submit Feedback
|
||||
~~~~~~~~~~~~~~~
|
||||
|
||||
The best way to send feedback is to file an issue at https://github.com/eeriks/erepublik_script/issues.
|
||||
|
||||
If you are proposing a feature:
|
||||
|
||||
* Explain in detail how it would work.
|
||||
* Keep the scope as narrow as possible, to make it easier to implement.
|
||||
* Remember that this is a volunteer-driven project, and that contributions
|
||||
are welcome :)
|
||||
|
||||
Get Started!
|
||||
------------
|
||||
|
||||
Ready to contribute? Here's how to set up `erepublik_script` for local development.
|
||||
|
||||
1. Fork the `erepublik_script` repo on GitHub.
|
||||
2. Clone your fork locally::
|
||||
|
||||
$ git clone git@github.com:your_name_here/erepublik_script.git
|
||||
|
||||
3. Install your local copy into a virtualenv. Assuming you have virtualenvwrapper installed, this is how you set up your fork for local development::
|
||||
|
||||
$ mkvirtualenv erepublik_script
|
||||
$ cd erepublik_script/
|
||||
$ python setup.py develop
|
||||
|
||||
4. Create a branch for local development::
|
||||
|
||||
$ git checkout -b name-of-your-bugfix-or-feature
|
||||
|
||||
Now you can make your changes locally.
|
||||
|
||||
5. When you're done making changes, check that your changes pass flake8 and the
|
||||
tests, including testing other Python versions with tox::
|
||||
|
||||
$ flake8 erepublik_script tests
|
||||
$ python setup.py test or py.test
|
||||
$ tox
|
||||
|
||||
To get flake8 and tox, just pip install them into your virtualenv.
|
||||
|
||||
6. Commit your changes and push your branch to GitHub::
|
||||
|
||||
$ git add .
|
||||
$ git commit -m "Your detailed description of your changes."
|
||||
$ git push origin name-of-your-bugfix-or-feature
|
||||
|
||||
7. Submit a pull request through the GitHub website.
|
||||
|
||||
Pull Request Guidelines
|
||||
-----------------------
|
||||
|
||||
Before you submit a pull request, check that it meets these guidelines:
|
||||
|
||||
1. The pull request should include tests.
|
||||
2. If the pull request adds functionality, the docs should be updated. Put
|
||||
your new functionality into a function with a docstring, and add the
|
||||
feature to the list in README.rst.
|
||||
3. The pull request should work for Python 2.7, 3.4, 3.5 and 3.6, and for PyPy. Check
|
||||
https://travis-ci.org/eeriks/erepublik_script/pull_requests
|
||||
and make sure that the tests pass for all supported Python versions.
|
||||
|
||||
Tips
|
||||
----
|
||||
|
||||
To run a subset of tests::
|
||||
|
||||
|
||||
$ python -m unittest tests.test_erepublik_script
|
||||
|
||||
Deploying
|
||||
---------
|
||||
|
||||
A reminder for the maintainers on how to deploy.
|
||||
Make sure all your changes are committed (including an entry in HISTORY.rst).
|
||||
Then run::
|
||||
|
||||
$ bumpversion patch # possible: major / minor / patch
|
||||
$ git push
|
||||
$ git push --tags
|
||||
|
||||
Travis will then deploy to PyPI if tests pass.
|
8
HISTORY.rst
Normal file
8
HISTORY.rst
Normal file
@ -0,0 +1,8 @@
|
||||
=======
|
||||
History
|
||||
=======
|
||||
|
||||
0.1.0 (2019-07-19)
|
||||
------------------
|
||||
|
||||
* First release on PyPI.
|
22
LICENSE
Normal file
22
LICENSE
Normal file
@ -0,0 +1,22 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2019, Eriks Karls
|
||||
|
||||
Permission is hereby granted, 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.
|
||||
|
11
MANIFEST.in
Normal file
11
MANIFEST.in
Normal file
@ -0,0 +1,11 @@
|
||||
include AUTHORS.rst
|
||||
include CONTRIBUTING.rst
|
||||
include HISTORY.rst
|
||||
include LICENSE
|
||||
include README.rst
|
||||
|
||||
recursive-include tests *
|
||||
recursive-exclude * __pycache__
|
||||
recursive-exclude * *.py[co]
|
||||
|
||||
recursive-include docs *.rst conf.py Makefile make.bat *.jpg *.png *.gif
|
88
Makefile
Normal file
88
Makefile
Normal file
@ -0,0 +1,88 @@
|
||||
.PHONY: clean clean-test clean-pyc clean-build docs help
|
||||
.DEFAULT_GOAL := help
|
||||
|
||||
define BROWSER_PYSCRIPT
|
||||
import os, webbrowser, sys
|
||||
|
||||
try:
|
||||
from urllib import pathname2url
|
||||
except:
|
||||
from urllib.request import pathname2url
|
||||
|
||||
webbrowser.open("file://" + pathname2url(os.path.abspath(sys.argv[1])))
|
||||
endef
|
||||
export BROWSER_PYSCRIPT
|
||||
|
||||
define PRINT_HELP_PYSCRIPT
|
||||
import re, sys
|
||||
|
||||
for line in sys.stdin:
|
||||
match = re.match(r'^([a-zA-Z_-]+):.*?## (.*)$$', line)
|
||||
if match:
|
||||
target, help = match.groups()
|
||||
print("%-20s %s" % (target, help))
|
||||
endef
|
||||
export PRINT_HELP_PYSCRIPT
|
||||
|
||||
BROWSER := python -c "$$BROWSER_PYSCRIPT"
|
||||
|
||||
help:
|
||||
@python -c "$$PRINT_HELP_PYSCRIPT" < $(MAKEFILE_LIST)
|
||||
|
||||
clean: clean-build clean-pyc clean-test ## remove all build, test, coverage and Python artifacts
|
||||
|
||||
clean-build: ## remove build artifacts
|
||||
rm -fr build/
|
||||
rm -fr dist/
|
||||
rm -fr .eggs/
|
||||
find . -name '*.egg-info' -exec rm -fr {} +
|
||||
find . -name '*.egg' -exec rm -f {} +
|
||||
|
||||
clean-pyc: ## remove Python file artifacts
|
||||
find . -name '*.pyc' -exec rm -f {} +
|
||||
find . -name '*.pyo' -exec rm -f {} +
|
||||
find . -name '*~' -exec rm -f {} +
|
||||
find . -name '__pycache__' -exec rm -fr {} +
|
||||
|
||||
clean-test: ## remove test and coverage artifacts
|
||||
rm -fr .tox/
|
||||
rm -f .coverage
|
||||
rm -fr htmlcov/
|
||||
rm -fr .pytest_cache
|
||||
|
||||
lint: ## check style with flake8
|
||||
flake8 erepublik_script tests
|
||||
|
||||
test: ## run tests quickly with the default Python
|
||||
python setup.py test
|
||||
|
||||
test-all: ## run tests on every Python version with tox
|
||||
tox
|
||||
|
||||
coverage: ## check code coverage quickly with the default Python
|
||||
coverage run --source erepublik_script setup.py test
|
||||
coverage report -m
|
||||
coverage html
|
||||
$(BROWSER) htmlcov/index.html
|
||||
|
||||
docs: ## generate Sphinx HTML documentation, including API docs
|
||||
rm -f docs/erepublik_script.rst
|
||||
rm -f docs/modules.rst
|
||||
sphinx-apidoc -o docs/ erepublik_script
|
||||
$(MAKE) -C docs clean
|
||||
$(MAKE) -C docs html
|
||||
$(BROWSER) docs/_build/html/index.html
|
||||
|
||||
servedocs: docs ## compile the docs watching for changes
|
||||
watchmedo shell-command -p '*.rst' -c '$(MAKE) -C docs html' -R -D .
|
||||
|
||||
release: dist ## package and upload a release
|
||||
twine upload dist/*
|
||||
|
||||
dist: clean ## builds source and wheel package
|
||||
python setup.py sdist
|
||||
python setup.py bdist_wheel
|
||||
ls -l dist
|
||||
|
||||
install: clean ## install the package to the active Python's site-packages
|
||||
python setup.py install
|
41
README.rst
Normal file
41
README.rst
Normal file
@ -0,0 +1,41 @@
|
||||
================
|
||||
eRepublik script
|
||||
================
|
||||
|
||||
|
||||
.. image:: https://img.shields.io/pypi/v/erepublik_script.svg
|
||||
:target: https://pypi.python.org/pypi/erepublik_script
|
||||
|
||||
.. image:: https://img.shields.io/travis/eeriks/erepublik_script.svg
|
||||
:target: https://travis-ci.org/eeriks/erepublik_script
|
||||
|
||||
.. image:: https://readthedocs.org/projects/erepublik-script/badge/?version=latest
|
||||
:target: https://erepublik-script.readthedocs.io/en/latest/?badge=latest
|
||||
:alt: Documentation Status
|
||||
|
||||
|
||||
.. image:: https://pyup.io/repos/github/eeriks/erepublik_script/shield.svg
|
||||
:target: https://pyup.io/repos/github/eeriks/erepublik_script/
|
||||
:alt: Updates
|
||||
|
||||
|
||||
|
||||
Python package for eRepublik automated playing
|
||||
|
||||
|
||||
* Free software: MIT license
|
||||
* Documentation: https://erepublik-script.readthedocs.io.
|
||||
|
||||
|
||||
Features
|
||||
--------
|
||||
|
||||
* TODO
|
||||
|
||||
Credits
|
||||
-------
|
||||
|
||||
This package was created with Cookiecutter_ and the `audreyr/cookiecutter-pypackage`_ project template.
|
||||
|
||||
.. _Cookiecutter: https://github.com/audreyr/cookiecutter
|
||||
.. _`audreyr/cookiecutter-pypackage`: https://github.com/audreyr/cookiecutter-pypackage
|
597
debug/requests/2019-07-18_23-40-20_.html
Normal file
597
debug/requests/2019-07-18_23-40-20_.html
Normal file
File diff suppressed because one or more lines are too long
1655
debug/requests/2019-07-18_23-40-21_login.html
Normal file
1655
debug/requests/2019-07-18_23-40-21_login.html
Normal file
File diff suppressed because one or more lines are too long
1
debug/requests/2019-07-18_23-40-21_login_REDIRECT.html
Normal file
1
debug/requests/2019-07-18_23-40-21_login_REDIRECT.html
Normal file
@ -0,0 +1 @@
|
||||
<html><head><meta http-equiv="refresh" content="0;url=https://www.erepublik.com/en"/></head></html>
|
1655
debug/requests/2019-07-18_23-40-22_.html
Normal file
1655
debug/requests/2019-07-18_23-40-22_.html
Normal file
File diff suppressed because one or more lines are too long
1655
debug/requests/2019-07-18_23-40-23_.html
Normal file
1655
debug/requests/2019-07-18_23-40-23_.html
Normal file
File diff suppressed because one or more lines are too long
1655
debug/requests/2019-07-18_23-40-31_.html
Normal file
1655
debug/requests/2019-07-18_23-40-31_.html
Normal file
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
8403
debug/requests/2019-07-18_23-40-33_economy-mycompanies.html
Normal file
8403
debug/requests/2019-07-18_23-40-33_economy-mycompanies.html
Normal file
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -0,0 +1 @@
|
||||
{"error":false,"enabled":true,"type":{"anniversary":false,"flavorPacks":false,"springChallenge":false,"summerChallenge":false,"halloweenChallenge":false},"timeLeft":346765,"nextReward":{"maxReward":false,"type":"icon_energy_booster","text":"+1 Energy recovery until the end of Day 4,262"},"maxRewardId":0,"player":{"avatar":"//cdnt.erepublik.net/7efiav4XZ4SMvXAtgEk1NciUmAg=/55x55/smart/avatars/Citizens/2009/07/08/4b57b9ebb0232f0d6c3f6f2c21b8ab95.jpg?c022b6df6f643263dba839cb35b7a9ab","name":"inpoc1","prestigePoints":14170},"progress":0.9141935483871,"rewards":{"normal":[{"id":59,"collectedBefore":58,"percent":0.74193548387097,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":60,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_booster"},{"id":61,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":62,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_booster"},{"id":63,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":64,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_booster"},{"id":65,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":66,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_booster"},{"id":67,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":68,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_booster"},{"id":69,"collectedBefore":0,"percent":0.016129032258065,"label":"You have already collected this reward","tooltip":"","status":"rewarded","icon":"energy_bars"},{"id":70,"collectedBefore":0,"percent":0.016129032258065,"label":"Reach 14,250 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","tooltip":"Reach 14250 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","status":"","icon":"energy_booster"},{"id":71,"collectedBefore":0,"percent":0.016129032258065,"label":"Reach 14,500 Prestige Points to unlock the following reward: 10 Energy Bars","tooltip":"Reach 14500 Prestige Points to unlock the following reward: 10 Energy Bars","status":"","icon":"energy_bars"},{"id":72,"collectedBefore":0,"percent":0.032258064516129,"label":"Reach 15,000 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","tooltip":"Reach 15000 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","status":"","icon":"energy_booster"},{"id":73,"collectedBefore":0,"percent":0.016129032258065,"label":"Reach 15,250 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","tooltip":"Reach 15250 Prestige Points to unlock the following reward: +1 Energy recovery until the end of Day 4,262","status":"","icon":"energy_booster"},{"id":74,"collectedBefore":0,"percent":0.016129032258065,"label":"Reach 15,500 Prestige Points to unlock the following reward: 15 Energy Bars","tooltip":"Reach 15500 Prestige Points to unlock the following reward: 15 Energy Bars","status":"","icon":"energy_bars"}],"extra":[]}}
|
File diff suppressed because one or more lines are too long
20
docs/Makefile
Normal file
20
docs/Makefile
Normal file
@ -0,0 +1,20 @@
|
||||
# Minimal makefile for Sphinx documentation
|
||||
#
|
||||
|
||||
# You can set these variables from the command line.
|
||||
SPHINXOPTS =
|
||||
SPHINXBUILD = python -msphinx
|
||||
SPHINXPROJ = erepublik_script
|
||||
SOURCEDIR = .
|
||||
BUILDDIR = _build
|
||||
|
||||
# Put it first so that "make" without argument is like "make help".
|
||||
help:
|
||||
@$(SPHINXBUILD) -M help "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
|
||||
|
||||
.PHONY: help Makefile
|
||||
|
||||
# Catch-all target: route all unknown targets to Sphinx using the new
|
||||
# "make mode" option. $(O) is meant as a shortcut for $(SPHINXOPTS).
|
||||
%: Makefile
|
||||
@$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
|
1
docs/authors.rst
Normal file
1
docs/authors.rst
Normal file
@ -0,0 +1 @@
|
||||
.. include:: ../AUTHORS.rst
|
163
docs/conf.py
Executable file
163
docs/conf.py
Executable file
@ -0,0 +1,163 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
#
|
||||
# erepublik_script documentation build configuration file, created by
|
||||
# sphinx-quickstart on Fri Jun 9 13:47:02 2017.
|
||||
#
|
||||
# This file is execfile()d with the current directory set to its
|
||||
# containing dir.
|
||||
#
|
||||
# Note that not all possible configuration values are present in this
|
||||
# autogenerated file.
|
||||
#
|
||||
# All configuration values have a default; values that are commented out
|
||||
# serve to show the default.
|
||||
|
||||
# If extensions (or modules to document with autodoc) are in another
|
||||
# directory, add these directories to sys.path here. If the directory is
|
||||
# relative to the documentation root, use os.path.abspath to make it
|
||||
# absolute, like shown here.
|
||||
#
|
||||
import os
|
||||
import sys
|
||||
sys.path.insert(0, os.path.abspath('..'))
|
||||
|
||||
import erepublik_script
|
||||
|
||||
# -- General configuration ---------------------------------------------
|
||||
|
||||
# If your documentation needs a minimal Sphinx version, state it here.
|
||||
#
|
||||
# needs_sphinx = '1.0'
|
||||
|
||||
# Add any Sphinx extension module names here, as strings. They can be
|
||||
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom ones.
|
||||
extensions = ['sphinx.ext.autodoc', 'sphinx.ext.viewcode']
|
||||
|
||||
# Add any paths that contain templates here, relative to this directory.
|
||||
templates_path = ['_templates']
|
||||
|
||||
# The suffix(es) of source filenames.
|
||||
# You can specify multiple suffix as a list of string:
|
||||
#
|
||||
# source_suffix = ['.rst', '.md']
|
||||
source_suffix = '.rst'
|
||||
|
||||
# The master toctree document.
|
||||
master_doc = 'index'
|
||||
|
||||
# General information about the project.
|
||||
project = u'eRepublik script'
|
||||
copyright = u"2019, Eriks Karls"
|
||||
author = u"Eriks Karls"
|
||||
|
||||
# The version info for the project you're documenting, acts as replacement
|
||||
# for |version| and |release|, also used in various other places throughout
|
||||
# the built documents.
|
||||
#
|
||||
# The short X.Y version.
|
||||
version = erepublik_script.__version__
|
||||
# The full version, including alpha/beta/rc tags.
|
||||
release = erepublik_script.__version__
|
||||
|
||||
# The language for content autogenerated by Sphinx. Refer to documentation
|
||||
# for a list of supported languages.
|
||||
#
|
||||
# This is also used if you do content translation via gettext catalogs.
|
||||
# Usually you set "language" from the command line for these cases.
|
||||
language = None
|
||||
|
||||
# List of patterns, relative to source directory, that match files and
|
||||
# directories to ignore when looking for source files.
|
||||
# This patterns also effect to html_static_path and html_extra_path
|
||||
exclude_patterns = ['_build', 'Thumbs.db', '.DS_Store']
|
||||
|
||||
# The name of the Pygments (syntax highlighting) style to use.
|
||||
pygments_style = 'sphinx'
|
||||
|
||||
# If true, `todo` and `todoList` produce output, else they produce nothing.
|
||||
todo_include_todos = False
|
||||
|
||||
|
||||
# -- Options for HTML output -------------------------------------------
|
||||
|
||||
# The theme to use for HTML and HTML Help pages. See the documentation for
|
||||
# a list of builtin themes.
|
||||
#
|
||||
html_theme = 'alabaster'
|
||||
|
||||
# Theme options are theme-specific and customize the look and feel of a
|
||||
# theme further. For a list of options available for each theme, see the
|
||||
# documentation.
|
||||
#
|
||||
# html_theme_options = {}
|
||||
|
||||
# Add any paths that contain custom static files (such as style sheets) here,
|
||||
# relative to this directory. They are copied after the builtin static files,
|
||||
# so a file named "default.css" will overwrite the builtin "default.css".
|
||||
html_static_path = ['_static']
|
||||
|
||||
|
||||
# -- Options for HTMLHelp output ---------------------------------------
|
||||
|
||||
# Output file base name for HTML help builder.
|
||||
htmlhelp_basename = 'erepublik_scriptdoc'
|
||||
|
||||
|
||||
# -- Options for LaTeX output ------------------------------------------
|
||||
|
||||
latex_elements = {
|
||||
# The paper size ('letterpaper' or 'a4paper').
|
||||
#
|
||||
# 'papersize': 'letterpaper',
|
||||
|
||||
# The font size ('10pt', '11pt' or '12pt').
|
||||
#
|
||||
# 'pointsize': '10pt',
|
||||
|
||||
# Additional stuff for the LaTeX preamble.
|
||||
#
|
||||
# 'preamble': '',
|
||||
|
||||
# Latex figure (float) alignment
|
||||
#
|
||||
# 'figure_align': 'htbp',
|
||||
}
|
||||
|
||||
# Grouping the document tree into LaTeX files. List of tuples
|
||||
# (source start file, target name, title, author, documentclass
|
||||
# [howto, manual, or own class]).
|
||||
latex_documents = [
|
||||
(master_doc, 'erepublik_script.tex',
|
||||
u'eRepublik script Documentation',
|
||||
u'Eriks Karls', 'manual'),
|
||||
]
|
||||
|
||||
|
||||
# -- Options for manual page output ------------------------------------
|
||||
|
||||
# One entry per manual page. List of tuples
|
||||
# (source start file, name, description, authors, manual section).
|
||||
man_pages = [
|
||||
(master_doc, 'erepublik_script',
|
||||
u'eRepublik script Documentation',
|
||||
[author], 1)
|
||||
]
|
||||
|
||||
|
||||
# -- Options for Texinfo output ----------------------------------------
|
||||
|
||||
# Grouping the document tree into Texinfo files. List of tuples
|
||||
# (source start file, target name, title, author,
|
||||
# dir menu entry, description, category)
|
||||
texinfo_documents = [
|
||||
(master_doc, 'erepublik_script',
|
||||
u'eRepublik script Documentation',
|
||||
author,
|
||||
'erepublik_script',
|
||||
'One line description of project.',
|
||||
'Miscellaneous'),
|
||||
]
|
||||
|
||||
|
||||
|
1
docs/contributing.rst
Normal file
1
docs/contributing.rst
Normal file
@ -0,0 +1 @@
|
||||
.. include:: ../CONTRIBUTING.rst
|
1
docs/history.rst
Normal file
1
docs/history.rst
Normal file
@ -0,0 +1 @@
|
||||
.. include:: ../HISTORY.rst
|
20
docs/index.rst
Normal file
20
docs/index.rst
Normal file
@ -0,0 +1,20 @@
|
||||
Welcome to eRepublik script's documentation!
|
||||
======================================
|
||||
|
||||
.. toctree::
|
||||
:maxdepth: 2
|
||||
:caption: Contents:
|
||||
|
||||
readme
|
||||
installation
|
||||
usage
|
||||
modules
|
||||
contributing
|
||||
authors
|
||||
history
|
||||
|
||||
Indices and tables
|
||||
==================
|
||||
* :ref:`genindex`
|
||||
* :ref:`modindex`
|
||||
* :ref:`search`
|
51
docs/installation.rst
Normal file
51
docs/installation.rst
Normal file
@ -0,0 +1,51 @@
|
||||
.. highlight:: shell
|
||||
|
||||
============
|
||||
Installation
|
||||
============
|
||||
|
||||
|
||||
Stable release
|
||||
--------------
|
||||
|
||||
To install eRepublik script, run this command in your terminal:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
$ pip install erepublik_script
|
||||
|
||||
This is the preferred method to install eRepublik script, as it will always install the most recent stable release.
|
||||
|
||||
If you don't have `pip`_ installed, this `Python installation guide`_ can guide
|
||||
you through the process.
|
||||
|
||||
.. _pip: https://pip.pypa.io
|
||||
.. _Python installation guide: http://docs.python-guide.org/en/latest/starting/installation/
|
||||
|
||||
|
||||
From sources
|
||||
------------
|
||||
|
||||
The sources for eRepublik script can be downloaded from the `Github repo`_.
|
||||
|
||||
You can either clone the public repository:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
$ git clone git://github.com/eeriks/erepublik_script
|
||||
|
||||
Or download the `tarball`_:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
$ curl -OL https://github.com/eeriks/erepublik_script/tarball/master
|
||||
|
||||
Once you have a copy of the source, you can install it with:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
$ python setup.py install
|
||||
|
||||
|
||||
.. _Github repo: https://github.com/eeriks/erepublik_script
|
||||
.. _tarball: https://github.com/eeriks/erepublik_script/tarball/master
|
36
docs/make.bat
Normal file
36
docs/make.bat
Normal file
@ -0,0 +1,36 @@
|
||||
@ECHO OFF
|
||||
|
||||
pushd %~dp0
|
||||
|
||||
REM Command file for Sphinx documentation
|
||||
|
||||
if "%SPHINXBUILD%" == "" (
|
||||
set SPHINXBUILD=python -msphinx
|
||||
)
|
||||
set SOURCEDIR=.
|
||||
set BUILDDIR=_build
|
||||
set SPHINXPROJ=erepublik_script
|
||||
|
||||
if "%1" == "" goto help
|
||||
|
||||
%SPHINXBUILD% >NUL 2>NUL
|
||||
if errorlevel 9009 (
|
||||
echo.
|
||||
echo.The Sphinx module was not found. Make sure you have Sphinx installed,
|
||||
echo.then set the SPHINXBUILD environment variable to point to the full
|
||||
echo.path of the 'sphinx-build' executable. Alternatively you may add the
|
||||
echo.Sphinx directory to PATH.
|
||||
echo.
|
||||
echo.If you don't have Sphinx installed, grab it from
|
||||
echo.http://sphinx-doc.org/
|
||||
exit /b 1
|
||||
)
|
||||
|
||||
%SPHINXBUILD% -M %1 %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
|
||||
goto end
|
||||
|
||||
:help
|
||||
%SPHINXBUILD% -M help %SOURCEDIR% %BUILDDIR% %SPHINXOPTS%
|
||||
|
||||
:end
|
||||
popd
|
1
docs/readme.rst
Normal file
1
docs/readme.rst
Normal file
@ -0,0 +1 @@
|
||||
.. include:: ../README.rst
|
7
docs/usage.rst
Normal file
7
docs/usage.rst
Normal file
@ -0,0 +1,7 @@
|
||||
=====
|
||||
Usage
|
||||
=====
|
||||
|
||||
To use eRepublik script in a project::
|
||||
|
||||
import erepublik_script
|
416
erepublik_script/__init__.py
Normal file
416
erepublik_script/__init__.py
Normal file
@ -0,0 +1,416 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""Top-level package for eRepublik script."""
|
||||
|
||||
__author__ = """Eriks Karls"""
|
||||
__email__ = 'eriks@72.lv'
|
||||
__version__ = '0.1.0'
|
||||
|
||||
import json
|
||||
import os
|
||||
import random
|
||||
import sys
|
||||
import threading
|
||||
from collections import defaultdict
|
||||
from datetime import timedelta
|
||||
from typing import List, Tuple
|
||||
|
||||
from erepublik_script import classes, utils
|
||||
from erepublik_script.citizen import Citizen
|
||||
|
||||
__all__ = ["Citizen"]
|
||||
|
||||
INTERACTIVE = True
|
||||
CONFIG = defaultdict(bool)
|
||||
|
||||
|
||||
def main():
|
||||
player = None
|
||||
try: # If errors before player is initialized
|
||||
while True:
|
||||
player = Citizen(email=CONFIG['email'], password=CONFIG['password'])
|
||||
if player.logged_in:
|
||||
break
|
||||
utils.silent_sleep(2)
|
||||
player.config.work = CONFIG['work']
|
||||
player.config.train = CONFIG['train']
|
||||
player.config.ot = CONFIG['ot']
|
||||
player.config.wam = bool(CONFIG['wam'])
|
||||
player.config.employees = bool(CONFIG['employ'])
|
||||
player.config.auto_sell = CONFIG.get('auto_sell', [])
|
||||
player.config.auto_sell_all = CONFIG.get('auto_sell_all', False)
|
||||
player.config.auto_buy_raw = CONFIG.get('auto_buy_raw', False)
|
||||
player.config.force_wam = CONFIG.get('force_wam', False)
|
||||
player.config.fight = CONFIG['fight']
|
||||
player.config.air = CONFIG['air']
|
||||
player.config.ground = CONFIG['ground']
|
||||
player.config.all_in = CONFIG['all_in']
|
||||
player.config.next_energy = CONFIG['next_energy']
|
||||
player.config.boosters = CONFIG['boosters']
|
||||
player.config.travel_to_fight = CONFIG['travel_to_fight']
|
||||
player.config.always_travel = CONFIG.get('always_travel', False)
|
||||
player.config.epic_hunt = CONFIG['epic_hunt']
|
||||
player.config.epic_hunt_ebs = CONFIG['epic_hunt_ebs']
|
||||
player.config.rw_def_side = CONFIG['rw_def_side']
|
||||
player.config.random_sleep = CONFIG['random_sleep']
|
||||
player.config.continuous_fighting = CONFIG['continuous_fighting']
|
||||
player.config.interactive = CONFIG['interactive']
|
||||
player.reporter.allowed = not CONFIG.get('reporting_is_not_allowed')
|
||||
|
||||
player.set_debug(CONFIG.get('debug', False))
|
||||
while True:
|
||||
try:
|
||||
player.update_all()
|
||||
break
|
||||
except:
|
||||
utils.silent_sleep(2)
|
||||
|
||||
now = utils.now()
|
||||
dt_max = now.replace(year=9999)
|
||||
tasks = {
|
||||
'eat': now,
|
||||
}
|
||||
wam_hour = employ_hour = 14
|
||||
if player.config.work:
|
||||
tasks.update({'work': now})
|
||||
if player.config.train:
|
||||
tasks.update({'train': now})
|
||||
if player.config.ot:
|
||||
tasks.update({'ot': now})
|
||||
if player.config.fight:
|
||||
tasks.update({'fight': now})
|
||||
if player.config.wam:
|
||||
wam_hour = 14
|
||||
if not isinstance(CONFIG['wam'], bool):
|
||||
try:
|
||||
wam_hour = abs(int(CONFIG['wam'])) % 24
|
||||
except ValueError:
|
||||
pass
|
||||
tasks.update({'wam': now.replace(hour=wam_hour, minute=0, second=0, microsecond=0)})
|
||||
if player.config.employees:
|
||||
employ_hour = 8
|
||||
if not isinstance(CONFIG['employ'], bool):
|
||||
try:
|
||||
employ_hour = abs(int(CONFIG['employ'])) % 24
|
||||
except ValueError:
|
||||
pass
|
||||
tasks.update({'employ': now.replace(hour=employ_hour, minute=0, second=0, microsecond=0)})
|
||||
|
||||
if player.config.epic_hunt:
|
||||
tasks['epic_hunt'] = now
|
||||
|
||||
if CONFIG.get("renew_houses", True):
|
||||
tasks['renew_houses'] = now
|
||||
|
||||
if CONFIG.get('start_battles'):
|
||||
""" {'start_battle': {war_id: {'regions': [region_id, ],
|
||||
'timing': ['at', 'hh:mm' | 'before', 'hh:mm' (before autoattack) |
|
||||
'auto' (after round for citizenship country's oldest battle or at 00:00)
|
||||
'rw', (after first round of RW if you are occupying)]}} """
|
||||
player.allowed_battles = CONFIG.get('start_battles', dict())
|
||||
raise classes.ErepublikException("Battle starting is not implemented")
|
||||
|
||||
if player.reporter.allowed:
|
||||
report = dict(CONFIG)
|
||||
report.pop("email", None)
|
||||
report.pop("password", None)
|
||||
report.update(
|
||||
VERSION=utils.VERSION,
|
||||
COMMIT_ID=utils.COMMIT_ID
|
||||
)
|
||||
player.reporter.report_action("ACTIVE_CONFIG", json_val=report)
|
||||
# -1 because main thread is counted in
|
||||
name = "{}-state_updater-{}".format(player.name, threading.active_count() - 1)
|
||||
state_thread = threading.Thread(target=player.state_update_repeater, name=name)
|
||||
state_thread.start()
|
||||
|
||||
if CONFIG.get("congress", True):
|
||||
tasks['congress'] = now.replace(hour=1, minute=30, second=0)
|
||||
|
||||
if CONFIG.get("party_president", False):
|
||||
tasks['party_president'] = now.replace(hour=1, minute=30, second=0)
|
||||
|
||||
contribute_cc = int(CONFIG.get("contribute_cc", 0))
|
||||
if contribute_cc:
|
||||
tasks['contribute_cc'] = now.replace(hour=2, minute=0, second=0)
|
||||
|
||||
if CONFIG.get("gold_buy"):
|
||||
tasks['gold_buy'] = now.replace(hour=23, minute=57, second=0, microsecond=0)
|
||||
|
||||
error_count = 0
|
||||
while error_count < 3:
|
||||
try:
|
||||
now = utils.now()
|
||||
player.update_all()
|
||||
if tasks.get('work', dt_max) <= now:
|
||||
player.write_log("Doing task: work")
|
||||
player.update_citizen_info()
|
||||
player.work()
|
||||
if player.config.ot:
|
||||
tasks['ot'] = now
|
||||
player.collect_daily_task()
|
||||
next_time = now.replace(hour=0, minute=0, second=0) + timedelta(days=1)
|
||||
tasks.update({'work': next_time})
|
||||
|
||||
if tasks.get('train', dt_max) <= now:
|
||||
player.write_log("Doing task: train")
|
||||
player.update_citizen_info()
|
||||
player.train()
|
||||
player.collect_daily_task()
|
||||
next_time = now.replace(hour=0, minute=0, second=0) + timedelta(days=1)
|
||||
tasks.update({'train': next_time})
|
||||
|
||||
if tasks.get('wam', dt_max) <= now:
|
||||
player.write_log("Doing task: Work as manager")
|
||||
success = player.work_wam()
|
||||
player.eat()
|
||||
if success:
|
||||
next_time = now.replace(hour=wam_hour, minute=0, second=0, microsecond=0) + timedelta(days=1)
|
||||
else:
|
||||
next_time = now.replace(second=0, microsecond=0) + timedelta(minutes=30)
|
||||
|
||||
tasks.update({'wam': next_time})
|
||||
|
||||
if tasks.get('eat', dt_max) <= now:
|
||||
player.write_log("Doing task: eat")
|
||||
player.eat()
|
||||
|
||||
if player.energy.food_fights > player.energy.limit // 10:
|
||||
next_minutes = 12
|
||||
else:
|
||||
next_minutes = (player.energy.limit - 5 * player.energy.interval) // player.energy.interval * 6
|
||||
|
||||
next_time = player.energy.reference_time + timedelta(minutes=next_minutes)
|
||||
tasks.update({'eat': next_time})
|
||||
|
||||
if tasks.get('fight', dt_max) <= now or player.energy.is_energy_full:
|
||||
fight_energy_debug_log: List[Tuple[int, str]] = []
|
||||
player.write_log("Doing task: fight")
|
||||
player.write_log(player.health_info)
|
||||
|
||||
if player.should_fight():
|
||||
player.find_battle_and_fight()
|
||||
else:
|
||||
player.collect_weekly_reward()
|
||||
energy = classes.EnergyToFight(player.details.xp_till_level_up * 10 - player.energy.limit + 50)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i,
|
||||
f"Levelup reachable {player.details.xp_till_level_up} * 10 - {player.energy.limit} + 50"
|
||||
))
|
||||
|
||||
# Do levelup
|
||||
energy.check(player.details.xp_till_level_up * 10 + 50)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"Levelup {player.details.xp_till_level_up} * 10 + 50"
|
||||
))
|
||||
|
||||
# if levelup is close stop queueing other fighting
|
||||
if not player.is_levelup_close:
|
||||
|
||||
# Obligatory need 75pp
|
||||
if player.details.pp < 75:
|
||||
energy.check(75 - player.details.pp)
|
||||
fight_energy_debug_log.append((energy.i, f"Obligatory need 75pp: 75 - {player.details.pp}"))
|
||||
|
||||
if player.config.continuous_fighting and player.has_battle_contribution:
|
||||
energy.check(player.energy.interval)
|
||||
fight_energy_debug_log.append((energy.i, f"continuous_fighting: {player.energy.interval}"))
|
||||
|
||||
# All-in
|
||||
if player.config.all_in:
|
||||
energy.check(player.energy.limit * 2 - 3 * player.energy.interval)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"All-in: {player.energy.limit} * 2 - 3 * {player.energy.interval}"
|
||||
))
|
||||
elif player.energy.limit * 2 - 3 * player.energy.interval >= player.energy.recovered:
|
||||
# 1h worth of energy
|
||||
energy.check(player.energy.limit * 2 - 3 * player.energy.interval)
|
||||
fight_energy_debug_log.append(
|
||||
(energy.i, f"1h worth of energy: {player.energy.interval} * 10"
|
||||
))
|
||||
|
||||
# All-in for AIR battles
|
||||
if all([player.config.air, player.config.all_in,
|
||||
player.energy.available >= player.energy.limit]):
|
||||
energy.check(player.energy.limit)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"All-in for AIR battles: {player.energy.limit}"
|
||||
))
|
||||
|
||||
# Get to next Energy +1
|
||||
if player.next_reachable_energy and player.config.next_energy:
|
||||
energy.check(player.next_reachable_energy * 10)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"Get to next Energy +1: {player.next_reachable_energy} * 10"
|
||||
))
|
||||
|
||||
energy = energy.i - player.energy.available
|
||||
next_minutes = max([6, abs(energy) // player.energy.interval * 6])
|
||||
# utils.write_silent_log("\n".join([f"{energy} {info}" for energy, info in fight_energy_debug_log]))
|
||||
next_time = player.energy.reference_time + timedelta(minutes=next_minutes)
|
||||
tasks.update({'fight': next_time})
|
||||
|
||||
if tasks.get('ot', dt_max) <= now:
|
||||
player.write_log("Doing task: ot")
|
||||
if now > player.my_companies.next_ot_time:
|
||||
player.work_ot()
|
||||
next_time = now + timedelta(minutes=60)
|
||||
else:
|
||||
next_time = player.my_companies.next_ot_time
|
||||
tasks.update({'ot': next_time})
|
||||
|
||||
if tasks.get('employ', dt_max) <= now:
|
||||
player.write_log("Doing task: Employee work")
|
||||
next_time = utils.now().replace(hour=employ_hour, minute=0, second=0) + timedelta(days=1)
|
||||
next_time = next_time if player.work_employees() else tasks.get('employ') + timedelta(minutes=30)
|
||||
tasks.update({'employ': next_time})
|
||||
|
||||
if tasks.get('epic_hunt', dt_max) <= now:
|
||||
player.write_log("Doing task: EPIC check")
|
||||
player.check_epic_battles()
|
||||
if player.active_fs:
|
||||
next_time = now + timedelta(minutes=1)
|
||||
else:
|
||||
next_time = tasks.get('eat')
|
||||
tasks.update({'epic_hunt': next_time})
|
||||
|
||||
if tasks.get('gold_buy', dt_max) <= now:
|
||||
player.write_log("Doing task: auto buy 10g")
|
||||
for offer in player.get_monetary_offers():
|
||||
if offer['amount'] >= 10 and player.details.cc >= 20 * offer["price"]:
|
||||
# TODO: check allowed amount to buy
|
||||
if player.buy_monetary_market_offer(offer=offer['offer_id'], amount=10, currency=62):
|
||||
break
|
||||
|
||||
next_time = tasks.get('gold_buy') + timedelta(days=1)
|
||||
tasks.update({'gold_buy': next_time})
|
||||
|
||||
if tasks.get('congress', dt_max) <= now:
|
||||
if 1 <= now.day < 16:
|
||||
next_time = now.replace(day=16)
|
||||
elif 16 <= now.day < 24:
|
||||
player.write_log("Doing task: candidate for congress")
|
||||
player.candidate_for_congress()
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1, day=16)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1, day=16)
|
||||
else:
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1, day=16)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1, day=16)
|
||||
tasks.update({'congress': next_time.replace(hour=1, minute=30, second=0, microsecond=0)})
|
||||
|
||||
if tasks.get('party_president', dt_max) <= now:
|
||||
if not now.day == 15:
|
||||
player.write_log("Doing task: candidate for party president")
|
||||
player.candidate_for_party_presidency()
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1)
|
||||
else:
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1)
|
||||
tasks.update(party_president=next_time.replace(day=16, hour=0, minute=0, second=0, microsecond=0))
|
||||
|
||||
if tasks.get('contribute_cc', dt_max) <= now:
|
||||
if not now.weekday():
|
||||
player.update_money()
|
||||
cc = (player.details.cc // contribute_cc) * contribute_cc
|
||||
player.write_log("Doing task: Contribute {}cc to Latvia".format(cc))
|
||||
player.contribute_cc_to_country(cc)
|
||||
next_time = now + timedelta(days=7 - now.weekday())
|
||||
next_time = next_time.replace(hour=2, minute=0, second=0)
|
||||
tasks.update({'contribute_cc': next_time})
|
||||
|
||||
if tasks.get('renew_houses', dt_max) <= now:
|
||||
player.write_log("Doing task: Renew houses")
|
||||
end_times = player.renew_houses()
|
||||
if end_times:
|
||||
tasks.update(renew_houses=min(end_times.values()) - timedelta(hours=24))
|
||||
else:
|
||||
player.write_log("No houses found! Forcing q1 usage...")
|
||||
end_times = player.buy_and_activate_house(1)
|
||||
if not end_times:
|
||||
tasks.update(renew_houses=now + timedelta(hours=6))
|
||||
else:
|
||||
tasks.update(renew_houses=min(end_times.values()) - timedelta(hours=24))
|
||||
|
||||
closest_next_time = dt_max
|
||||
next_tasks = []
|
||||
for task, next_time in sorted(tasks.items(), key=lambda s: s[1]):
|
||||
next_tasks.append("{}: {}".format(next_time.strftime('%F %T'), task))
|
||||
if next_time < closest_next_time:
|
||||
closest_next_time = next_time
|
||||
random_seconds = random.randint(0, 121) if player.config.random_sleep else 0
|
||||
sleep_seconds = int(utils.get_sleep_seconds(closest_next_time))
|
||||
if sleep_seconds <= 0:
|
||||
raise classes.ErepublikException(f"Loop detected! Offending task: '{next_tasks[0]}'")
|
||||
closest_next_time += timedelta(seconds=random_seconds)
|
||||
player.write_log("My next Tasks and there time:\n" + "\n".join(sorted(next_tasks)))
|
||||
player.write_log("Sleeping until (eRep): {} (sleeping for {}s + random {}s)".format(
|
||||
closest_next_time.strftime("%F %T"), sleep_seconds, random_seconds))
|
||||
seconds_to_sleep = sleep_seconds + random_seconds if sleep_seconds > 0 else 0
|
||||
player.sleep(seconds_to_sleep)
|
||||
|
||||
except classes.ErepublikNetworkException:
|
||||
player.write_log('Network ERROR detected. Sleeping for 1min...')
|
||||
player.sleep(60)
|
||||
except (KeyboardInterrupt, SystemExit):
|
||||
sys.exit(1)
|
||||
except classes.ErepublikException as e:
|
||||
utils.process_error(f"Known error detected! {e}", player.name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
except:
|
||||
utils.process_error("Unknown error!", player.name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
error_count += 1
|
||||
if error_count < 3:
|
||||
player.sleep(60)
|
||||
player.stop_threads.set()
|
||||
player.write_log('Too many errors.')
|
||||
except (KeyboardInterrupt, SystemExit):
|
||||
sys.exit(1)
|
||||
except classes.ErepublikException:
|
||||
utils.process_error("[{}] Fatal error.".format(utils.COMMIT_ID), player.name, sys.exc_info(), player,
|
||||
utils.COMMIT_ID)
|
||||
except:
|
||||
if isinstance(player, Citizen):
|
||||
name = player.name
|
||||
elif CONFIG.get('email', None):
|
||||
name = CONFIG['email']
|
||||
else:
|
||||
name = "Uninitialized"
|
||||
utils.process_error("[{}] Fatal error.".format(utils.COMMIT_ID), name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
sys.exit(1)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
assert sys.version_info >= (3, 7, 1)
|
||||
|
||||
write_log = utils.write_silent_log
|
||||
|
||||
try:
|
||||
with open('config.json', 'r') as f:
|
||||
CONFIG = json.load(f)
|
||||
|
||||
write_log('Config file found. Checking...')
|
||||
CONFIG = utils.parse_config(CONFIG)
|
||||
except:
|
||||
CONFIG = utils.parse_config()
|
||||
|
||||
with open('config.json', 'w') as f:
|
||||
json.dump(CONFIG, f, indent=True, sort_keys=True)
|
||||
if CONFIG['interactive']:
|
||||
write_log = utils.write_interactive_log
|
||||
else:
|
||||
write_log = utils.write_silent_log
|
||||
write_log('\nTo quit press [ctrl] + [c]', False)
|
||||
os.chdir(os.path.dirname(os.path.realpath(__file__)))
|
||||
write_log('Version: ' + utils.VERSION)
|
||||
while True:
|
||||
main()
|
||||
write_log("Restarting after 1h")
|
||||
utils.interactive_sleep(60 * 60)
|
1802
erepublik_script/citizen.py
Normal file
1802
erepublik_script/citizen.py
Normal file
File diff suppressed because it is too large
Load Diff
1046
erepublik_script/classes.py
Normal file
1046
erepublik_script/classes.py
Normal file
File diff suppressed because it is too large
Load Diff
423
erepublik_script/cli.py
Normal file
423
erepublik_script/cli.py
Normal file
@ -0,0 +1,423 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""Console script for erepublik_script."""
|
||||
|
||||
__author__ = """Eriks Karls"""
|
||||
__email__ = 'eriks@72.lv'
|
||||
__version__ = '0.1.0'
|
||||
|
||||
import json
|
||||
import os
|
||||
import random
|
||||
import sys
|
||||
import threading
|
||||
from collections import defaultdict
|
||||
from datetime import timedelta
|
||||
from typing import List, Tuple
|
||||
|
||||
import click
|
||||
|
||||
from erepublik_script import classes, utils
|
||||
from erepublik_script.citizen import Citizen
|
||||
|
||||
|
||||
__all__ = ["Citizen"]
|
||||
|
||||
INTERACTIVE = True
|
||||
CONFIG = defaultdict(bool)
|
||||
|
||||
|
||||
@click.command()
|
||||
def main(args=None):
|
||||
player = None
|
||||
try: # If errors before player is initialized
|
||||
while True:
|
||||
player = Citizen(email=CONFIG['email'], password=CONFIG['password'])
|
||||
if player.logged_in:
|
||||
break
|
||||
utils.silent_sleep(2)
|
||||
player.config.work = CONFIG['work']
|
||||
player.config.train = CONFIG['train']
|
||||
player.config.ot = CONFIG['ot']
|
||||
player.config.wam = bool(CONFIG['wam'])
|
||||
player.config.employees = bool(CONFIG['employ'])
|
||||
player.config.auto_sell = CONFIG.get('auto_sell', [])
|
||||
player.config.auto_sell_all = CONFIG.get('auto_sell_all', False)
|
||||
player.config.auto_buy_raw = CONFIG.get('auto_buy_raw', False)
|
||||
player.config.force_wam = CONFIG.get('force_wam', False)
|
||||
player.config.fight = CONFIG['fight']
|
||||
player.config.air = CONFIG['air']
|
||||
player.config.ground = CONFIG['ground']
|
||||
player.config.all_in = CONFIG['all_in']
|
||||
player.config.next_energy = CONFIG['next_energy']
|
||||
player.config.boosters = CONFIG['boosters']
|
||||
player.config.travel_to_fight = CONFIG['travel_to_fight']
|
||||
player.config.always_travel = CONFIG.get('always_travel', False)
|
||||
player.config.epic_hunt = CONFIG['epic_hunt']
|
||||
player.config.epic_hunt_ebs = CONFIG['epic_hunt_ebs']
|
||||
player.config.rw_def_side = CONFIG['rw_def_side']
|
||||
player.config.random_sleep = CONFIG['random_sleep']
|
||||
player.config.continuous_fighting = CONFIG['continuous_fighting']
|
||||
player.config.interactive = CONFIG['interactive']
|
||||
player.reporter.allowed = not CONFIG.get('reporting_is_not_allowed')
|
||||
|
||||
player.set_debug(CONFIG.get('debug', False))
|
||||
while True:
|
||||
try:
|
||||
player.update_all()
|
||||
break
|
||||
except:
|
||||
utils.silent_sleep(2)
|
||||
|
||||
now = utils.now()
|
||||
dt_max = now.replace(year=9999)
|
||||
tasks = {
|
||||
'eat': now,
|
||||
}
|
||||
wam_hour = employ_hour = 14
|
||||
if player.config.work:
|
||||
tasks.update({'work': now})
|
||||
if player.config.train:
|
||||
tasks.update({'train': now})
|
||||
if player.config.ot:
|
||||
tasks.update({'ot': now})
|
||||
if player.config.fight:
|
||||
tasks.update({'fight': now})
|
||||
if player.config.wam:
|
||||
wam_hour = 14
|
||||
if not isinstance(CONFIG['wam'], bool):
|
||||
try:
|
||||
wam_hour = abs(int(CONFIG['wam'])) % 24
|
||||
except ValueError:
|
||||
pass
|
||||
tasks.update({'wam': now.replace(hour=wam_hour, minute=0, second=0, microsecond=0)})
|
||||
if player.config.employees:
|
||||
employ_hour = 8
|
||||
if not isinstance(CONFIG['employ'], bool):
|
||||
try:
|
||||
employ_hour = abs(int(CONFIG['employ'])) % 24
|
||||
except ValueError:
|
||||
pass
|
||||
tasks.update({'employ': now.replace(hour=employ_hour, minute=0, second=0, microsecond=0)})
|
||||
|
||||
if player.config.epic_hunt:
|
||||
tasks['epic_hunt'] = now
|
||||
|
||||
if CONFIG.get("renew_houses", True):
|
||||
tasks['renew_houses'] = now
|
||||
|
||||
if CONFIG.get('start_battles'):
|
||||
""" {'start_battle': {war_id: {'regions': [region_id, ],
|
||||
'timing': ['at', 'hh:mm' | 'before', 'hh:mm' (before autoattack) |
|
||||
'auto' (after round for citizenship country's oldest battle or at 00:00)
|
||||
'rw', (after first round of RW if you are occupying)]}} """
|
||||
player.allowed_battles = CONFIG.get('start_battles', dict())
|
||||
raise classes.ErepublikException("Battle starting is not implemented")
|
||||
|
||||
if player.reporter.allowed:
|
||||
report = dict(CONFIG)
|
||||
report.pop("email", None)
|
||||
report.pop("password", None)
|
||||
report.update(
|
||||
VERSION=utils.VERSION,
|
||||
COMMIT_ID=utils.COMMIT_ID
|
||||
)
|
||||
player.reporter.report_action("ACTIVE_CONFIG", json_val=report)
|
||||
# -1 because main thread is counted in
|
||||
name = "{}-state_updater-{}".format(player.name, threading.active_count() - 1)
|
||||
state_thread = threading.Thread(target=player.state_update_repeater, name=name)
|
||||
state_thread.start()
|
||||
|
||||
if CONFIG.get("congress", True):
|
||||
tasks['congress'] = now.replace(hour=1, minute=30, second=0)
|
||||
|
||||
if CONFIG.get("party_president", False):
|
||||
tasks['party_president'] = now.replace(hour=1, minute=30, second=0)
|
||||
|
||||
contribute_cc = int(CONFIG.get("contribute_cc", 0))
|
||||
if contribute_cc:
|
||||
tasks['contribute_cc'] = now.replace(hour=2, minute=0, second=0)
|
||||
|
||||
if CONFIG.get("gold_buy"):
|
||||
tasks['gold_buy'] = now.replace(hour=23, minute=57, second=0, microsecond=0)
|
||||
|
||||
error_count = 0
|
||||
while error_count < 3:
|
||||
try:
|
||||
now = utils.now()
|
||||
player.update_all()
|
||||
if tasks.get('work', dt_max) <= now:
|
||||
player.write_log("Doing task: work")
|
||||
player.update_citizen_info()
|
||||
player.work()
|
||||
if player.config.ot:
|
||||
tasks['ot'] = now
|
||||
player.collect_daily_task()
|
||||
next_time = now.replace(hour=0, minute=0, second=0) + timedelta(days=1)
|
||||
tasks.update({'work': next_time})
|
||||
|
||||
if tasks.get('train', dt_max) <= now:
|
||||
player.write_log("Doing task: train")
|
||||
player.update_citizen_info()
|
||||
player.train()
|
||||
player.collect_daily_task()
|
||||
next_time = now.replace(hour=0, minute=0, second=0) + timedelta(days=1)
|
||||
tasks.update({'train': next_time})
|
||||
|
||||
if tasks.get('wam', dt_max) <= now:
|
||||
player.write_log("Doing task: Work as manager")
|
||||
success = player.work_wam()
|
||||
player.eat()
|
||||
if success:
|
||||
next_time = now.replace(hour=wam_hour, minute=0, second=0, microsecond=0) + timedelta(days=1)
|
||||
else:
|
||||
next_time = now.replace(second=0, microsecond=0) + timedelta(minutes=30)
|
||||
|
||||
tasks.update({'wam': next_time})
|
||||
|
||||
if tasks.get('eat', dt_max) <= now:
|
||||
player.write_log("Doing task: eat")
|
||||
player.eat()
|
||||
|
||||
if player.energy.food_fights > player.energy.limit // 10:
|
||||
next_minutes = 12
|
||||
else:
|
||||
next_minutes = (player.energy.limit - 5 * player.energy.interval) // player.energy.interval * 6
|
||||
|
||||
next_time = player.energy.reference_time + timedelta(minutes=next_minutes)
|
||||
tasks.update({'eat': next_time})
|
||||
|
||||
if tasks.get('fight', dt_max) <= now or player.energy.is_energy_full:
|
||||
fight_energy_debug_log: List[Tuple[int, str]] = []
|
||||
player.write_log("Doing task: fight")
|
||||
player.write_log(player.health_info)
|
||||
|
||||
if player.should_fight():
|
||||
player.find_battle_and_fight()
|
||||
else:
|
||||
player.collect_weekly_reward()
|
||||
energy = classes.EnergyToFight(player.details.xp_till_level_up * 10 - player.energy.limit + 50)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i,
|
||||
f"Levelup reachable {player.details.xp_till_level_up} * 10 - {player.energy.limit} + 50"
|
||||
))
|
||||
|
||||
# Do levelup
|
||||
energy.check(player.details.xp_till_level_up * 10 + 50)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"Levelup {player.details.xp_till_level_up} * 10 + 50"
|
||||
))
|
||||
|
||||
# if levelup is close stop queueing other fighting
|
||||
if not player.is_levelup_close:
|
||||
|
||||
# Obligatory need 75pp
|
||||
if player.details.pp < 75:
|
||||
energy.check(75 - player.details.pp)
|
||||
fight_energy_debug_log.append((energy.i, f"Obligatory need 75pp: 75 - {player.details.pp}"))
|
||||
|
||||
if player.config.continuous_fighting and player.has_battle_contribution:
|
||||
energy.check(player.energy.interval)
|
||||
fight_energy_debug_log.append((energy.i, f"continuous_fighting: {player.energy.interval}"))
|
||||
|
||||
# All-in
|
||||
if player.config.all_in:
|
||||
energy.check(player.energy.limit * 2 - 3 * player.energy.interval)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"All-in: {player.energy.limit} * 2 - 3 * {player.energy.interval}"
|
||||
))
|
||||
elif player.energy.limit * 2 - 3 * player.energy.interval >= player.energy.recovered:
|
||||
# 1h worth of energy
|
||||
energy.check(player.energy.limit * 2 - 3 * player.energy.interval)
|
||||
fight_energy_debug_log.append(
|
||||
(energy.i, f"1h worth of energy: {player.energy.interval} * 10"
|
||||
))
|
||||
|
||||
# All-in for AIR battles
|
||||
if all([player.config.air, player.config.all_in,
|
||||
player.energy.available >= player.energy.limit]):
|
||||
energy.check(player.energy.limit)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"All-in for AIR battles: {player.energy.limit}"
|
||||
))
|
||||
|
||||
# Get to next Energy +1
|
||||
if player.next_reachable_energy and player.config.next_energy:
|
||||
energy.check(player.next_reachable_energy * 10)
|
||||
fight_energy_debug_log.append((
|
||||
energy.i, f"Get to next Energy +1: {player.next_reachable_energy} * 10"
|
||||
))
|
||||
|
||||
energy = energy.i - player.energy.available
|
||||
next_minutes = max([6, abs(energy) // player.energy.interval * 6])
|
||||
# utils.write_silent_log("\n".join([f"{energy} {info}" for energy, info in fight_energy_debug_log]))
|
||||
next_time = player.energy.reference_time + timedelta(minutes=next_minutes)
|
||||
tasks.update({'fight': next_time})
|
||||
|
||||
if tasks.get('ot', dt_max) <= now:
|
||||
player.write_log("Doing task: ot")
|
||||
if now > player.my_companies.next_ot_time:
|
||||
player.work_ot()
|
||||
next_time = now + timedelta(minutes=60)
|
||||
else:
|
||||
next_time = player.my_companies.next_ot_time
|
||||
tasks.update({'ot': next_time})
|
||||
|
||||
if tasks.get('employ', dt_max) <= now:
|
||||
player.write_log("Doing task: Employee work")
|
||||
next_time = utils.now().replace(hour=employ_hour, minute=0, second=0) + timedelta(days=1)
|
||||
next_time = next_time if player.work_employees() else tasks.get('employ') + timedelta(minutes=30)
|
||||
tasks.update({'employ': next_time})
|
||||
|
||||
if tasks.get('epic_hunt', dt_max) <= now:
|
||||
player.write_log("Doing task: EPIC check")
|
||||
player.check_epic_battles()
|
||||
if player.active_fs:
|
||||
next_time = now + timedelta(minutes=1)
|
||||
else:
|
||||
next_time = tasks.get('eat')
|
||||
tasks.update({'epic_hunt': next_time})
|
||||
|
||||
if tasks.get('gold_buy', dt_max) <= now:
|
||||
player.write_log("Doing task: auto buy 10g")
|
||||
for offer in player.get_monetary_offers():
|
||||
if offer['amount'] >= 10 and player.details.cc >= 20 * offer["price"]:
|
||||
# TODO: check allowed amount to buy
|
||||
if player.buy_monetary_market_offer(offer=offer['offer_id'], amount=10, currency=62):
|
||||
break
|
||||
|
||||
next_time = tasks.get('gold_buy') + timedelta(days=1)
|
||||
tasks.update({'gold_buy': next_time})
|
||||
|
||||
if tasks.get('congress', dt_max) <= now:
|
||||
if 1 <= now.day < 16:
|
||||
next_time = now.replace(day=16)
|
||||
elif 16 <= now.day < 24:
|
||||
player.write_log("Doing task: candidate for congress")
|
||||
player.candidate_for_congress()
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1, day=16)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1, day=16)
|
||||
else:
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1, day=16)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1, day=16)
|
||||
tasks.update({'congress': next_time.replace(hour=1, minute=30, second=0, microsecond=0)})
|
||||
|
||||
if tasks.get('party_president', dt_max) <= now:
|
||||
if not now.day == 15:
|
||||
player.write_log("Doing task: candidate for party president")
|
||||
player.candidate_for_party_presidency()
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1)
|
||||
else:
|
||||
if not now.month == 12:
|
||||
next_time = now.replace(month=now.month + 1)
|
||||
else:
|
||||
next_time = now.replace(year=now.year + 1, month=1)
|
||||
tasks.update(party_president=next_time.replace(day=16, hour=0, minute=0, second=0, microsecond=0))
|
||||
|
||||
if tasks.get('contribute_cc', dt_max) <= now:
|
||||
if not now.weekday():
|
||||
player.update_money()
|
||||
cc = (player.details.cc // contribute_cc) * contribute_cc
|
||||
player.write_log("Doing task: Contribute {}cc to Latvia".format(cc))
|
||||
player.contribute_cc_to_country(cc)
|
||||
next_time = now + timedelta(days=7 - now.weekday())
|
||||
next_time = next_time.replace(hour=2, minute=0, second=0)
|
||||
tasks.update({'contribute_cc': next_time})
|
||||
|
||||
if tasks.get('renew_houses', dt_max) <= now:
|
||||
player.write_log("Doing task: Renew houses")
|
||||
end_times = player.renew_houses()
|
||||
if end_times:
|
||||
tasks.update(renew_houses=min(end_times.values()) - timedelta(hours=24))
|
||||
else:
|
||||
player.write_log("No houses found! Forcing q1 usage...")
|
||||
end_times = player.buy_and_activate_house(1)
|
||||
if not end_times:
|
||||
tasks.update(renew_houses=now + timedelta(hours=6))
|
||||
else:
|
||||
tasks.update(renew_houses=min(end_times.values()) - timedelta(hours=24))
|
||||
|
||||
closest_next_time = dt_max
|
||||
next_tasks = []
|
||||
for task, next_time in sorted(tasks.items(), key=lambda s: s[1]):
|
||||
next_tasks.append("{}: {}".format(next_time.strftime('%F %T'), task))
|
||||
if next_time < closest_next_time:
|
||||
closest_next_time = next_time
|
||||
random_seconds = random.randint(0, 121) if player.config.random_sleep else 0
|
||||
sleep_seconds = int(utils.get_sleep_seconds(closest_next_time))
|
||||
if sleep_seconds <= 0:
|
||||
raise classes.ErepublikException(f"Loop detected! Offending task: '{next_tasks[0]}'")
|
||||
closest_next_time += timedelta(seconds=random_seconds)
|
||||
player.write_log("My next Tasks and there time:\n" + "\n".join(sorted(next_tasks)))
|
||||
player.write_log("Sleeping until (eRep): {} (sleeping for {}s + random {}s)".format(
|
||||
closest_next_time.strftime("%F %T"), sleep_seconds, random_seconds))
|
||||
seconds_to_sleep = sleep_seconds + random_seconds if sleep_seconds > 0 else 0
|
||||
player.sleep(seconds_to_sleep)
|
||||
|
||||
except classes.ErepublikNetworkException:
|
||||
player.write_log('Network ERROR detected. Sleeping for 1min...')
|
||||
player.sleep(60)
|
||||
except (KeyboardInterrupt, SystemExit):
|
||||
sys.exit(1)
|
||||
except classes.ErepublikException as e:
|
||||
utils.process_error(f"Known error detected! {e}", player.name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
except:
|
||||
utils.process_error("Unknown error!", player.name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
error_count += 1
|
||||
if error_count < 3:
|
||||
player.sleep(60)
|
||||
finally:
|
||||
if error_count >= 3:
|
||||
player.stop_threads.set()
|
||||
player.stop_threads.set()
|
||||
player.write_log('Too many errors.')
|
||||
except (KeyboardInterrupt, SystemExit):
|
||||
sys.exit(1)
|
||||
except classes.ErepublikException:
|
||||
utils.process_error("[{}] To many errors.".format(utils.COMMIT_ID), player.name, sys.exc_info(), player,
|
||||
utils.COMMIT_ID)
|
||||
except:
|
||||
if isinstance(player, Citizen):
|
||||
name = player.name
|
||||
elif CONFIG.get('email', None):
|
||||
name = CONFIG['email']
|
||||
else:
|
||||
name = "Uninitialized"
|
||||
utils.process_error("[{}] Fatal error.".format(utils.COMMIT_ID), name, sys.exc_info(), player, utils.COMMIT_ID)
|
||||
sys.exit(1)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
assert sys.version_info >= (3, 7, 1)
|
||||
|
||||
write_log = utils.write_silent_log
|
||||
|
||||
try:
|
||||
with open('config.json', 'r') as f:
|
||||
CONFIG = json.load(f)
|
||||
|
||||
write_log('Config file found. Checking...')
|
||||
CONFIG = utils.parse_config(CONFIG)
|
||||
except:
|
||||
CONFIG = utils.parse_config()
|
||||
|
||||
with open('config.json', 'w') as f:
|
||||
json.dump(CONFIG, f, indent=True, sort_keys=True)
|
||||
if CONFIG['interactive']:
|
||||
write_log = utils.write_interactive_log
|
||||
else:
|
||||
write_log = utils.write_silent_log
|
||||
write_log('\nTo quit press [ctrl] + [c]', False)
|
||||
os.chdir(os.path.dirname(os.path.realpath(__file__)))
|
||||
write_log('Version: ' + utils.VERSION)
|
||||
while True:
|
||||
main()
|
||||
write_log("Restarting after 1h")
|
||||
utils.interactive_sleep(60 * 60)
|
730
erepublik_script/utils.py
Normal file
730
erepublik_script/utils.py
Normal file
@ -0,0 +1,730 @@
|
||||
import datetime
|
||||
import inspect
|
||||
import json
|
||||
import os
|
||||
import re
|
||||
import sys
|
||||
import time
|
||||
import traceback
|
||||
from collections import deque
|
||||
from decimal import Decimal
|
||||
from json import JSONEncoder
|
||||
from pathlib import Path
|
||||
from typing import Union
|
||||
|
||||
import pytz
|
||||
import requests
|
||||
from requests import Response
|
||||
from slugify import slugify
|
||||
|
||||
|
||||
__all__ = ["FOOD_ENERGY", "VERSION", "COMMIT_ID", "COUNTRIES", "erep_tz",
|
||||
"now", "localize_dt", "localize_timestamp", "good_timedelta", "eday_from_date", "date_from_eday",
|
||||
"get_sleep_seconds", "interactive_sleep", "silent_sleep",
|
||||
"write_silent_log", "write_interactive_log", "get_file", "write_file",
|
||||
"send_email", "normalize_html_json", "process_error", ]
|
||||
|
||||
|
||||
FOOD_ENERGY = dict(q1=2, q2=4, q3=6, q4=8, q5=10, q6=12, q7=20)
|
||||
VERSION = "v0.14.1"
|
||||
COMMIT_ID = "7b92e19"
|
||||
|
||||
erep_tz = pytz.timezone('US/Pacific')
|
||||
AIR_RANKS = {1: "Airman", 2: "Airman 1st Class", 3: "Airman 1st Class*", 4: "Airman 1st Class**",
|
||||
5: "Airman 1st Class***", 6: "Airman 1st Class****", 7: "Airman 1st Class*****",
|
||||
8: "Senior Airman", 9: "Senior Airman*", 10: "Senior Airman**", 11: "Senior Airman***",
|
||||
12: "Senior Airman****", 13: "Senior Airman*****",
|
||||
14: "Staff Sergeant", 15: "Staff Sergeant*", 16: "Staff Sergeant**", 17: "Staff Sergeant***",
|
||||
18: "Staff Sergeant****", 19: "Staff Sergeant*****",
|
||||
20: "Aviator", 21: "Aviator*", 22: "Aviator**", 23: "Aviator***", 24: "Aviator****", 25: "Aviator*****",
|
||||
26: "Flight Lieutenant", 27: "Flight Lieutenant*", 28: "Flight Lieutenant**", 29: "Flight Lieutenant***",
|
||||
30: "Flight Lieutenant****", 31: "Flight Lieutenant*****",
|
||||
32: "Squadron Leader", 33: "Squadron Leader*", 34: "Squadron Leader**", 35: "Squadron Leader***",
|
||||
36: "Squadron Leader****", 37: "Squadron Leader*****",
|
||||
38: "Chief Master Sergeant", 39: "Chief Master Sergeant*", 40: "Chief Master Sergeant**",
|
||||
41: "Chief Master Sergeant***", 42: "Chief Master Sergeant****", 43: "Chief Master Sergeant*****",
|
||||
44: "Wing Commander", 45: "Wing Commander*", 46: "Wing Commander**", 47: "Wing Commander***",
|
||||
48: "Wing Commander****", 49: "Wing Commander*****",
|
||||
50: "Group Captain", 51: "Group Captain*", 52: "Group Captain**", 53: "Group Captain***",
|
||||
54: "Group Captain****", 55: "Group Captain*****",
|
||||
56: "Air Commodore", 57: "Air Commodore*", 58: "Air Commodore**", 59: "Air Commodore***",
|
||||
60: "Air Commodore****", 61: "Air Commodore*****", }
|
||||
|
||||
GROUND_RANKS = {1: "Recruit", 2: "Private", 3: "Private*", 4: "Private**", 5: "Private***", 6: "Corporal",
|
||||
7: "Corporal*", 8: "Corporal**", 9: "Corporal***",
|
||||
10: "Sergeant", 11: "Sergeant*", 12: "Sergeant**", 13: "Sergeant***", 14: "Lieutenant",
|
||||
15: "Lieutenant*", 16: "Lieutenant**", 17: "Lieutenant***",
|
||||
18: "Captain", 19: "Captain*", 20: "Captain**", 21: "Captain***", 22: "Major", 23: "Major*",
|
||||
24: "Major**", 25: "Major***",
|
||||
26: "Commander", 27: "Commander*", 28: "Commander**", 29: "Commander***", 30: "Lt Colonel",
|
||||
31: "Lt Colonel*", 32: "Lt Colonel**", 33: "Lt Colonel***",
|
||||
34: "Colonel", 35: "Colonel*", 36: "Colonel**", 37: "Colonel***", 38: "General", 39: "General*",
|
||||
40: "General**", 41: "General***",
|
||||
42: "Field Marshal", 43: "Field Marshal*", 44: "Field Marshal**", 45: "Field Marshal***",
|
||||
46: "Supreme Marshal", 47: "Supreme Marshal*", 48: "Supreme Marshal**", 49: "Supreme Marshal***",
|
||||
50: "National Force", 51: "National Force*", 52: "National Force**", 53: "National Force***",
|
||||
54: "World Class Force", 55: "World Class Force*", 56: "World Class Force**",
|
||||
57: "World Class Force***", 58: "Legendary Force", 59: "Legendary Force*", 60: "Legendary Force**",
|
||||
61: "Legendary Force***",
|
||||
62: "God of War", 63: "God of War*", 64: "God of War**", 65: "God of War***", 66: "Titan", 67: "Titan*",
|
||||
68: "Titan**", 69: "Titan***",
|
||||
70: "Legends I", 71: "Legends II", 72: "Legends III", 73: "Legends IV", 74: "Legends V",
|
||||
75: "Legends VI", 76: "Legends VII", 77: "Legends VIII", 78: "Legends IX", 79: "Legends X",
|
||||
80: "Legends XI", 81: "Legends XII", 82: "Legends XIII", 83: "Legends XIV", 84: "Legends XV",
|
||||
85: "Legends XVI", 86: "Legends XVII", 87: "Legends XVIII", 88: "Legends XIX", 89: "Legends XX", }
|
||||
|
||||
COUNTRIES = {1: 'Romania', 9: 'Brazil', 10: 'Italy', 11: 'France', 12: 'Germany', 13: 'Hungary', 14: 'China',
|
||||
15: 'Spain', 23: 'Canada', 24: 'USA', 26: 'Mexico', 27: 'Argentina', 28: 'Venezuela', 29: 'United Kingdom',
|
||||
30: 'Switzerland', 31: 'Netherlands', 32: 'Belgium', 33: 'Austria', 34: 'Czech Republic', 35: 'Poland',
|
||||
36: 'Slovakia', 37: 'Norway', 38: 'Sweden', 39: 'Finland', 40: 'Ukraine', 41: 'Russia', 42: 'Bulgaria',
|
||||
43: 'Turkey', 44: 'Greece', 45: 'Japan', 47: 'South Korea', 48: 'India', 49: 'Indonesia', 50: 'Australia',
|
||||
51: 'South Africa', 52: 'Republic of Moldova', 53: 'Portugal', 54: 'Ireland', 55: 'Denmark', 56: 'Iran',
|
||||
57: 'Pakistan', 58: 'Israel', 59: 'Thailand', 61: 'Slovenia', 63: 'Croatia', 64: 'Chile', 65: 'Serbia',
|
||||
66: 'Malaysia', 67: 'Philippines', 68: 'Singapore', 69: 'Bosnia and Herzegovina', 70: 'Estonia',
|
||||
71: 'Latvia', 72: 'Lithuania', 73: 'North Korea', 74: 'Uruguay', 75: 'Paraguay', 76: 'Bolivia', 77: 'Peru',
|
||||
78: 'Colombia', 79: 'Republic of Macedonia (FYROM)', 80: 'Montenegro', 81: 'Republic of China (Taiwan)',
|
||||
82: 'Cyprus', 83: 'Belarus', 84: 'New Zealand', 164: 'Saudi Arabia', 165: 'Egypt',
|
||||
166: 'United Arab Emirates', 167: 'Albania', 168: 'Georgia', 169: 'Armenia', 170: 'Nigeria', 171: 'Cuba'}
|
||||
|
||||
|
||||
class MyJSONEncoder(JSONEncoder):
|
||||
def default(self, o):
|
||||
if isinstance(o, Decimal):
|
||||
return float("{:.02f}".format(o))
|
||||
elif isinstance(o, datetime.datetime):
|
||||
return dict(__type__='datetime', year=o.year, month=o.month, day=o.day, hour=o.hour, minute=o.minute,
|
||||
second=o.second, microsecond=o.microsecond)
|
||||
elif isinstance(o, datetime.date):
|
||||
return dict(__type__='date', year=o.year, month=o.month, day=o.day)
|
||||
elif isinstance(o, datetime.timedelta):
|
||||
return dict(__type__='timedelta', days=o.days, seconds=o.seconds,
|
||||
microseconds=o.microseconds, total_seconds=o.total_seconds())
|
||||
elif isinstance(o, Response):
|
||||
return dict(headers=o.headers.__dict__, url=o.url, text=o.text)
|
||||
elif hasattr(o, '__dict__'):
|
||||
return o.__dict__
|
||||
elif isinstance(o, deque):
|
||||
return list(o)
|
||||
return super().default(o)
|
||||
|
||||
|
||||
def now():
|
||||
return datetime.datetime.now(erep_tz).replace(microsecond=0)
|
||||
|
||||
|
||||
def localize_timestamp(timestamp: int):
|
||||
return datetime.datetime.fromtimestamp(timestamp, erep_tz)
|
||||
|
||||
|
||||
def localize_dt(dt: Union[datetime.date, datetime.datetime]):
|
||||
if isinstance(dt, datetime.date):
|
||||
dt = datetime.datetime.combine(dt, datetime.time(0, 0, 0))
|
||||
return erep_tz.localize(dt)
|
||||
|
||||
|
||||
def good_timedelta(dt: datetime.datetime, td: datetime.timedelta) -> datetime.datetime:
|
||||
return erep_tz.normalize(dt + td)
|
||||
|
||||
|
||||
def eday_from_date(date: Union[datetime.date, datetime.datetime] = now()) -> int:
|
||||
if isinstance(date, datetime.date):
|
||||
date = datetime.datetime.combine(date, datetime.time(0, 0, 0))
|
||||
return (date - datetime.datetime(2007, 11, 20, 0, 0, 0)).days
|
||||
|
||||
|
||||
def date_from_eday(eday: int) -> datetime.date:
|
||||
return localize_dt(datetime.date(2007, 11, 20)) + datetime.timedelta(days=eday)
|
||||
|
||||
|
||||
def get_sleep_seconds(time_untill: datetime.datetime) -> int:
|
||||
""" time_until aware datetime object Wrapper for sleeping until """
|
||||
sleep_seconds = int((time_untill - now()).total_seconds())
|
||||
return sleep_seconds if sleep_seconds > 0 else 0
|
||||
|
||||
|
||||
def interactive_sleep(sleep_seconds: int):
|
||||
while sleep_seconds > 0:
|
||||
seconds = sleep_seconds
|
||||
if (seconds - 1) // 1800:
|
||||
seconds = seconds % 1800 if seconds % 1800 else 1800
|
||||
elif (seconds - 1) // 300:
|
||||
seconds = seconds % 300 if seconds % 300 else 300
|
||||
elif (seconds - 1) // 60:
|
||||
seconds = seconds % 60 if seconds % 60 else 60
|
||||
# elif (seconds - 1) // 30:
|
||||
# seconds = seconds % 30 if seconds % 30 else 30
|
||||
else:
|
||||
seconds = 1
|
||||
sys.stdout.write("\rSleeping for {:4} more seconds".format(sleep_seconds))
|
||||
sys.stdout.flush()
|
||||
time.sleep(seconds)
|
||||
sleep_seconds -= seconds
|
||||
sys.stdout.write("\r")
|
||||
|
||||
|
||||
silent_sleep = time.sleep
|
||||
|
||||
|
||||
def _write_log(msg, timestamp: bool = True, should_print: bool = False):
|
||||
erep_time_now = now()
|
||||
txt = "[{}] {}".format(erep_time_now.strftime('%F %T'), msg) if timestamp else msg
|
||||
if not os.path.isdir('log'):
|
||||
os.mkdir('log')
|
||||
with open("log/%s.log" % erep_time_now.strftime('%F'), 'a', encoding="utf-8") as f:
|
||||
f.write("%s\n" % txt)
|
||||
if should_print:
|
||||
print(txt)
|
||||
|
||||
|
||||
def write_interactive_log(*args, **kwargs):
|
||||
_write_log(should_print=True, *args, **kwargs)
|
||||
|
||||
|
||||
def write_silent_log(*args, **kwargs):
|
||||
_write_log(should_print=False, *args, **kwargs)
|
||||
|
||||
|
||||
def get_file(filepath: str) -> str:
|
||||
file = Path(filepath)
|
||||
if file.exists():
|
||||
if file.is_dir():
|
||||
return str(file / "new_file.txt")
|
||||
else:
|
||||
version = 1
|
||||
try:
|
||||
version = int(file.suffix[1:]) + 1
|
||||
basename = file.stem
|
||||
except ValueError:
|
||||
basename = file.name
|
||||
version += 1
|
||||
|
||||
full_name = file.parent / f"{basename}.{version}"
|
||||
while full_name.exists():
|
||||
version += 1
|
||||
full_name = file.parent / f"{basename}.{version}"
|
||||
return str(full_name)
|
||||
else:
|
||||
os.makedirs(file.parent, exist_ok=True)
|
||||
return str(file)
|
||||
|
||||
|
||||
def write_file(filename: str, content: str) -> int:
|
||||
filename = get_file(filename)
|
||||
with open(filename, 'ab') as f:
|
||||
return f.write(content.encode("utf-8"))
|
||||
|
||||
|
||||
def write_request(response: requests.Response, is_error: bool = False):
|
||||
from erepublik_script import Citizen
|
||||
# Remove GET args from url name
|
||||
url = response.url
|
||||
last_index = url.index("?") if "?" in url else len(response.url)
|
||||
|
||||
name = slugify(response.url[len(Citizen.url):last_index])
|
||||
html = response.text
|
||||
|
||||
try:
|
||||
json.loads(html)
|
||||
ext = "json"
|
||||
except json.decoder.JSONDecodeError:
|
||||
ext = "html"
|
||||
|
||||
if not is_error:
|
||||
filename = "debug/requests/{}_{}.{}".format(now().strftime('%F_%H-%M-%S'), name, ext)
|
||||
write_file(filename, html)
|
||||
else:
|
||||
return {"name": "{}_{}.{}".format(now().strftime('%F_%H-%M-%S'), name, ext),
|
||||
"content": html.encode('utf-8'),
|
||||
"mimetype": "application/json" if ext == "json" else "text/html"}
|
||||
|
||||
|
||||
def send_email(name, content: list, player=None, local_vars=dict, promo: bool = False, captcha: bool = False):
|
||||
from erepublik_script import Citizen
|
||||
|
||||
file_content_template = "<html><head><title>{title}</title></head><body>{body}</body></html>"
|
||||
if isinstance(player, Citizen):
|
||||
resp = write_request(player.r, is_error=True)
|
||||
else:
|
||||
resp = {"name": "None.html", "mimetype": "text/html",
|
||||
"content": file_content_template.format(body="<br/>".join(content), title="Error"), }
|
||||
|
||||
if promo:
|
||||
resp = {"name": "%s.html" % name, "mimetype": "text/html",
|
||||
"content": file_content_template.format(title="Promo", body="<br/>".join(content))}
|
||||
subject = "[eBot][{}] Promos: {}".format(now().strftime('%F %T'), name)
|
||||
|
||||
elif captcha:
|
||||
resp = {"name": "%s.html" % name, "mimetype": "text/html",
|
||||
"content": file_content_template.format(title="ReCaptcha", body="<br/>".join(content))}
|
||||
subject = "[eBot][{}] RECAPTCHA: {}".format(now().strftime('%F %T'), name)
|
||||
else:
|
||||
subject = "[eBot][%s] Bug trace: %s" % (now().strftime('%F %T'), name)
|
||||
|
||||
body = "".join(traceback.format_stack()) + \
|
||||
"\n\n" + \
|
||||
"\n".join(content)
|
||||
data = dict(send_mail=True, subject=subject, bugtrace=body)
|
||||
if promo:
|
||||
data.update({'promo': True})
|
||||
elif captcha:
|
||||
data.update({'captcha': True})
|
||||
else:
|
||||
data.update({"bug": True})
|
||||
|
||||
files = [('file', (resp.get("name"), resp.get("content"), resp.get("mimetype"))), ]
|
||||
filename = "log/%s.log" % now().strftime('%F')
|
||||
if os.path.isfile(filename):
|
||||
files.append(('file', (filename[4:], open(filename, 'rb'), "text/plain")))
|
||||
if local_vars:
|
||||
if "state_thread" in local_vars:
|
||||
local_vars.pop('state_thread', None)
|
||||
files.append(('file', ("local_vars.json", json.dumps(local_vars, indent=2,
|
||||
cls=MyJSONEncoder, sort_keys=True), "application/json")))
|
||||
if isinstance(player, Citizen):
|
||||
files.append(('file', ("instance.json", player.to_json(indent=True), "application/json")))
|
||||
requests.post('https://pasts.72.lv', data=data, files=files)
|
||||
|
||||
|
||||
def parse_input(msg: str) -> bool:
|
||||
msg += " (y|n):"
|
||||
data = None
|
||||
while data not in ['', 'y', 'Y', 'n', 'N']:
|
||||
try:
|
||||
data = input(msg)
|
||||
except EOFError:
|
||||
data = 'n'
|
||||
|
||||
return data in ['', 'y', 'Y']
|
||||
|
||||
|
||||
def parse_config(config=None) -> dict:
|
||||
if config is None:
|
||||
config = {}
|
||||
|
||||
if not config.get('email'):
|
||||
config['email'] = input("Player email: ")
|
||||
|
||||
if not config.get('password'):
|
||||
config['password'] = input("Player password: ")
|
||||
|
||||
if 'wt' in config:
|
||||
config['work'] = config['wt']
|
||||
config['train'] = config['wt']
|
||||
|
||||
if 'work' not in config:
|
||||
config['work'] = parse_input('Should I work')
|
||||
|
||||
if 'train' not in config:
|
||||
config['train'] = parse_input('Should I train')
|
||||
|
||||
if 'ot' not in config:
|
||||
config['ot'] = parse_input('Should I work overtime')
|
||||
|
||||
if 'wam' not in config:
|
||||
config['wam'] = parse_input('Should I WAM')
|
||||
|
||||
if 'employ' not in config:
|
||||
config['employ'] = parse_input('Should I employ employees')
|
||||
|
||||
if config['wam'] or config['employ']:
|
||||
if "autosell" in config:
|
||||
config.pop("autosell")
|
||||
if "autosell_raw" in config:
|
||||
config.pop("autosell_raw")
|
||||
if "autosell_final" in config:
|
||||
config.pop("autosell_final")
|
||||
|
||||
if 'auto_sell' not in config or not isinstance(config['auto_sell'], list):
|
||||
if parse_input('Should I auto sell produced products'):
|
||||
config['auto_sell'] = []
|
||||
if parse_input("Should I auto sell Food products"):
|
||||
if parse_input("Should I auto sell Food products"):
|
||||
config['auto_sell'].append("food")
|
||||
if parse_input("Should I auto sell Weapon products"):
|
||||
config['auto_sell'].append("weapon")
|
||||
if parse_input("Should I auto sell House products"):
|
||||
config['auto_sell'].append("house")
|
||||
if parse_input("Should I auto sell Aircraft products"):
|
||||
config['auto_sell'].append("airplane")
|
||||
if parse_input("Should I auto sell raw products"):
|
||||
if parse_input("Should I auto sell Food raw"):
|
||||
config['auto_sell'].append("foodRaw")
|
||||
if parse_input("Should I auto sell Weapon raw"):
|
||||
config['auto_sell'].append("weaponRaw")
|
||||
if parse_input("Should I auto sell House raw"):
|
||||
config['auto_sell'].append("houseRaw")
|
||||
if parse_input("Should I auto sell Airplane raw"):
|
||||
config['auto_sell'].append("airplaneRaw")
|
||||
if config['auto_sell']:
|
||||
if 'auto_sell_all' not in config:
|
||||
print("When selling produced items should I also sell items already in inventory?")
|
||||
config['auto_sell_all'] = parse_input('Y - sell all, N - only just produced')
|
||||
else:
|
||||
config['auto_sell_all'] = False
|
||||
|
||||
if 'auto_buy_raw' not in config:
|
||||
config['auto_buy_raw'] = parse_input('Should I auto buy raw deficit at WAM or employ')
|
||||
else:
|
||||
config['auto_sell'] = []
|
||||
config['auto_sell_all'] = False
|
||||
config['auto_buy_raw'] = False
|
||||
|
||||
if 'fight' not in config:
|
||||
config['fight'] = parse_input('Should I fight')
|
||||
|
||||
if config.get('fight'):
|
||||
if 'air' not in config:
|
||||
config['air'] = parse_input('Should I fight in AIR')
|
||||
|
||||
if 'ground' not in config:
|
||||
config['ground'] = parse_input('Should I fight in GROUND')
|
||||
|
||||
if 'all_in' not in config:
|
||||
print("When full energy should I go all in")
|
||||
config['all_in'] = parse_input('Y - all in, N - 1h worth of energy')
|
||||
|
||||
if 'next_energy' not in config:
|
||||
config['next_energy'] = parse_input('Should I fight when next pp +1 energy available')
|
||||
|
||||
if 'boosters' not in config:
|
||||
config['boosters'] = parse_input('Should I use +50% dmg boosters')
|
||||
|
||||
if 'travel_to_fight' not in config:
|
||||
config['travel_to_fight'] = parse_input('Should I travel to fight')
|
||||
|
||||
if 'epic_hunt' not in config:
|
||||
config['epic_hunt'] = parse_input('Should I check for epic battles')
|
||||
if not config['epic_hunt']:
|
||||
config['epic_hunt_ebs'] = False
|
||||
|
||||
if not config['epic_hunt']:
|
||||
config['epic_hunt_ebs'] = False
|
||||
elif 'epic_hunt_ebs' not in config:
|
||||
config['epic_hunt_ebs'] = parse_input('Should I eat EBs when fighting in epic battle')
|
||||
|
||||
if 'rw_def_side' not in config:
|
||||
config['rw_def_side'] = parse_input('Should I fight on defenders side in RWs')
|
||||
|
||||
if 'continuous_fighting' not in config:
|
||||
config['continuous_fighting'] = parse_input('If already fought in any battle, \n'
|
||||
'should I continue to fight all FF in that battle')
|
||||
else:
|
||||
config['air'] = False
|
||||
config['ground'] = False
|
||||
config['all_in'] = False
|
||||
config['next_energy'] = False
|
||||
config['boosters'] = False
|
||||
config['travel_to_fight'] = False
|
||||
config['epic_hunt'] = False
|
||||
config['epic_hunt_ebs'] = False
|
||||
config['rw_def_side'] = False
|
||||
config['continuous_fighting'] = False
|
||||
|
||||
if 'debug' not in config:
|
||||
config['debug'] = parse_input('Should I generate debug files')
|
||||
|
||||
if 'random_sleep' not in config:
|
||||
config['random_sleep'] = parse_input('Should I add random amount (0-120sec) to sleep time')
|
||||
|
||||
if 'gold_buy' not in config:
|
||||
config['gold_buy'] = parse_input('Should I auto buy 10g every day')
|
||||
|
||||
if 'interactive' not in config:
|
||||
config['interactive'] = parse_input('Should I print output to console?')
|
||||
|
||||
return config
|
||||
|
||||
|
||||
def normalize_html_json(js: str) -> str:
|
||||
js = re.sub(r' \'(.*?)\'', lambda a: '"%s"' % a.group(1), js)
|
||||
js = re.sub(r'(\d\d):(\d\d):(\d\d)', r'\1\2\3', js)
|
||||
js = re.sub(r'([{\s,])(\w+)(:)(?!"})', r'\1"\2"\3', js)
|
||||
js = re.sub(r',\s*}', '}', js)
|
||||
return js
|
||||
|
||||
|
||||
def process_error(log_info: str, name: str, exc_info: tuple, citizen=None, commit_id: str = None,
|
||||
interactive: bool = False):
|
||||
"""
|
||||
Process error logging and email sending to developer
|
||||
:param error:
|
||||
:param interactive: Should print interactively
|
||||
:param log_info: String to be written in output
|
||||
:param name: String Instance name
|
||||
:param exc_info: tuple output from sys.exc_info()
|
||||
:param citizen: Citizen instance
|
||||
:param commit_id: Code's version by commit id
|
||||
"""
|
||||
type_, value_, traceback_ = exc_info
|
||||
bugtrace = [] if not commit_id else ["Commit id: %s" % commit_id, ]
|
||||
bugtrace += [str(value_), str(type_), ''.join(traceback.format_tb(traceback_))]
|
||||
|
||||
if interactive:
|
||||
write_interactive_log(log_info)
|
||||
else:
|
||||
write_silent_log(log_info)
|
||||
send_email(name, bugtrace, citizen, local_vars=inspect.trace()[-1][0].f_locals)
|
||||
|
||||
|
||||
def aviator_support(citizen, send_food=False, free_food=None):
|
||||
forbidden_ids = []
|
||||
if free_food is None:
|
||||
free_food = {} # {"q1": 0, "q2": 1000, ...}
|
||||
context = {'PLAYER_COUNT': 0, 'TABLE': "",
|
||||
'STARTING_ENERGY': sum([amount * FOOD_ENERGY[q] for q, amount in free_food.items()]),
|
||||
'TOTAL_CC': 0, 'TOTAL_ENERGY': 0, 'END_ENERGY': 0}
|
||||
from erepublik_script import Citizen
|
||||
if not isinstance(citizen, Citizen):
|
||||
from .classes import ErepublikException
|
||||
raise ErepublikException("\"citizen\" must be instance of erepublik.Citizen")
|
||||
citizen.config.interactive = True
|
||||
aviators = dict()
|
||||
time_string = "%Y-%m-%d %H:%M:%S"
|
||||
latest_article = requests.get('https://erep.lv/aviator/latest_article/').json()
|
||||
for quality, amount in latest_article.get('free_food', {}).items():
|
||||
free_food[quality] = free_food.get(quality, 0) + amount
|
||||
|
||||
if not latest_article.get('status'):
|
||||
from .classes import ErepublikException
|
||||
raise ErepublikException('Article ID and week problem')
|
||||
context.update(WEEK=latest_article.get('week', 0) + 1)
|
||||
comments = citizen.post_article_comments(citizen.token, latest_article.get('article_id'), 1).json()
|
||||
ranking = citizen.get_leaderboards_kills_aircraft_rankings(71, 1, 0).json()
|
||||
|
||||
if not comments.get("comments", {}):
|
||||
from .classes import ErepublikException
|
||||
raise ErepublikException("No comments found")
|
||||
for comment_data in comments.get("comments", {}).values():
|
||||
if comment_data.get('authorId') == 1954361:
|
||||
start_dt = localize_dt(datetime.datetime.strptime(comment_data.get('createdAt'), time_string))
|
||||
days_ahead = 1 - start_dt.weekday()
|
||||
if days_ahead <= 0:
|
||||
days_ahead += 7
|
||||
end_dt = (good_timedelta(start_dt, datetime.timedelta(days_ahead))).replace(hour=0, minute=0, second=0)
|
||||
if not comment_data.get('replies', {}):
|
||||
from .classes import ErepublikException
|
||||
raise ErepublikException("No replies found")
|
||||
|
||||
for reply_data in comment_data.get('replies').values():
|
||||
if localize_dt(datetime.datetime.strptime(reply_data.get('createdAt'), time_string)) > end_dt:
|
||||
continue
|
||||
if re.search(r'piesakos', reply_data.get('message'), re.I):
|
||||
aviators.update({int(reply_data.get('authorId')): dict(
|
||||
id=reply_data.get('authorId'), name="", kills=0, rank=0, residency=None, health=0, extra=[],
|
||||
factories=0
|
||||
)})
|
||||
|
||||
context['PLAYER_COUNT'] = len(aviators)
|
||||
write_interactive_log("{:^9} | {:<28} | {:4} | {:26} | {:6} | {}".format(
|
||||
"ID", "Vārds", "Kili", "Gaisa rangs", "Energy", "Aktivizētās mājas"
|
||||
))
|
||||
|
||||
for player_top_data in ranking.get('top'):
|
||||
player_id = int(player_top_data.get('id'))
|
||||
if player_id in aviators:
|
||||
aviators[player_id]["kills"] = int(player_top_data['values'])
|
||||
|
||||
for aviator_id, aviator_data in aviators.items():
|
||||
aviator_info = citizen.get_citizen_profile(aviator_id).json()
|
||||
aviator_data.update({
|
||||
'rank': aviator_info['military']['militaryData']['aircraft']['rankNumber'],
|
||||
'name': aviator_info['citizen']['name'],
|
||||
'residency': aviator_info['city']['residenceCityId']
|
||||
})
|
||||
|
||||
if aviator_info.get("isBanned"):
|
||||
aviator_data.update({'health': 0, 'extra': ["BANNED", ]})
|
||||
else:
|
||||
if aviator_data['rank'] < 44:
|
||||
if aviator_data['rank'] < 38:
|
||||
health = aviator_data['kills'] * 30
|
||||
else:
|
||||
health = aviator_data['kills'] * 20
|
||||
has_pp = False
|
||||
if aviator_info.get("activePacks"):
|
||||
has_pp = bool(aviator_info.get("activePacks").get("power_pack"))
|
||||
max_health = 7 * 24 * (500 if has_pp else 300)
|
||||
if health < max_health:
|
||||
aviator_data['health'] = health
|
||||
else:
|
||||
aviator_data['health'] = max_health
|
||||
|
||||
if not aviator_data["residency"]:
|
||||
aviator_data['health'] = 0
|
||||
aviator_data['extra'].append("No residency set")
|
||||
else:
|
||||
residency = citizen.get_city_data_residents(
|
||||
aviator_data["residency"], params={"search": aviator_data['name']}
|
||||
).json()
|
||||
|
||||
for resident in residency.get('widgets', {}).get('residents', {}).get('residents'):
|
||||
if int(resident.get('citizenId')) == aviator_id:
|
||||
if resident['numFactories']:
|
||||
aviator_data['factories'] = resident['numFactories']
|
||||
else:
|
||||
aviator_data['factories'] = 0
|
||||
if not resident.get('activeHouses'):
|
||||
aviator_data['health'] = 0
|
||||
if resident['numHouses']:
|
||||
aviator_data['extra'].append(", ".join(resident['activeHouses']))
|
||||
else:
|
||||
aviator_data['extra'].append("Nav māja")
|
||||
aviator_data['health'] = 0
|
||||
|
||||
else:
|
||||
aviator_data['extra'].append("Rank")
|
||||
|
||||
write_interactive_log("{id:>9} | {name:<28} | {kills:4} | {:26} | {health:6} | {}".format(
|
||||
AIR_RANKS[aviator_data['rank']],
|
||||
", ".join(aviator_data["extra"]),
|
||||
**aviator_data)
|
||||
)
|
||||
|
||||
db_post_data = []
|
||||
for aviator_id, aviator_data in aviators.items():
|
||||
db_post_data.append(dict(id=aviator_id, name=aviator_data['name'],
|
||||
rank=aviator_data['rank'], factory_count=aviator_data['factories']))
|
||||
requests.post('https://erep.lv/aviator/set/', json=db_post_data)
|
||||
|
||||
for aviator_id, new in aviators.items():
|
||||
resp = requests.get('https://erep.lv/aviator/check/{}/'.format(aviator_id))
|
||||
if not resp.json()['status']:
|
||||
aviators[aviator_id]['health'] = 0
|
||||
aviators[aviator_id]['extra'] = ["Nav izmaiņas fabriku skaitā", ]
|
||||
|
||||
for player_id in forbidden_ids:
|
||||
if player_id in aviators:
|
||||
aviators[player_id]['health'] = 0
|
||||
if "BANNED" not in aviators[player_id]['extra']:
|
||||
aviators[player_id]['extra'] = ["Aizliegta pieteikšanās", ]
|
||||
|
||||
sent_data = []
|
||||
if send_food:
|
||||
for aviator_data in sorted(aviators.values(), key=lambda t: (-t["health"], -t['kills'])):
|
||||
remaining = aviator_data['health']
|
||||
if not remaining:
|
||||
sent_data.append({
|
||||
"player_id": aviator_data['id'], "name": aviator_data['name'], "quality": 0,
|
||||
"amount": 0, "energy": 0, "price": 0, "cost": 0,
|
||||
})
|
||||
while remaining > 0:
|
||||
o = []
|
||||
if free_food:
|
||||
# Reversed because need to start with higher qualities so that q1 stays available
|
||||
for quality in reversed(list(free_food.keys())):
|
||||
if free_food[quality]:
|
||||
o.append((quality, {'price': 0., 'amount': free_food[quality]}))
|
||||
else:
|
||||
free_food.pop(quality)
|
||||
if not free_food:
|
||||
offers = citizen.get_market_offers(71, product="food")
|
||||
o += sorted(offers.items(), key=lambda v: (v[1]['price'] / FOOD_ENERGY[v[0]],
|
||||
-v[1]['amount'] * FOOD_ENERGY[v[0]]))
|
||||
|
||||
for _o in o:
|
||||
q, q_data = _o
|
||||
if FOOD_ENERGY[q] <= remaining:
|
||||
break
|
||||
else:
|
||||
write_interactive_log(
|
||||
"{name} needs to receive extra {remaining}hp".format(name=aviator_data['name'],
|
||||
remaining=remaining))
|
||||
break
|
||||
|
||||
if q_data['amount'] * FOOD_ENERGY[q] <= remaining:
|
||||
amount = q_data['amount']
|
||||
else:
|
||||
amount = remaining // FOOD_ENERGY[q]
|
||||
|
||||
if q_data['price']:
|
||||
# print(f"citizen._buy_market_offer(offer={q_data['offer_id']}, amount={amount})")
|
||||
citizen.post_economy_marketplace_actions(citizen.token, amount=amount, buy=True,
|
||||
offer=q_data["offer_id"])
|
||||
else:
|
||||
free_food[q] -= amount
|
||||
|
||||
# print(f"citizen.donate_items(citizen_id={aviator_data['id']},
|
||||
# amount={amount}, industry_id=1, quality={int(q[1])})")
|
||||
citizen.donate_items(citizen_id=aviator_data['id'], amount=amount, industry_id=1, quality=int(q[1]))
|
||||
remaining -= amount * FOOD_ENERGY[q]
|
||||
context['TOTAL_CC'] += q_data['price'] * amount
|
||||
context['TOTAL_ENERGY'] += amount * FOOD_ENERGY[q]
|
||||
sent_data.append(
|
||||
{"player_id": aviator_data['id'], "name": aviator_data['name'], "quality": q, "amount": amount,
|
||||
"energy": amount * FOOD_ENERGY[q], "price": q_data['price'],
|
||||
"cost": q_data['price'] * amount, })
|
||||
|
||||
with open(get_file("{eday}.csv".format(eday=eday_from_date(now()))), 'a') as f:
|
||||
f.write('PlayerID, Quality, Amount, Energy, Price, Cost\n')
|
||||
for player_data in sent_data:
|
||||
f.write('{player_id}, {quality}, {amount}, {energy}, {price}, {cost}\n'.format(**player_data))
|
||||
|
||||
columns = ('[columns][b]Spēlētajs[/b]\n'
|
||||
'{players}[nextcol][b]Kili[/b]\n'
|
||||
'{kills}\n'
|
||||
'[nextcol][right][b]Enerģija[/b]\n'
|
||||
'{health}\n'
|
||||
'[/right][/columns]')
|
||||
player_template = '[b][url=https://www.erepublik.com/en/citizen/profile/{id}]{name}[/url][/b]'
|
||||
players = []
|
||||
kills = []
|
||||
health = []
|
||||
write_interactive_log("\n".join(["{}: {}".format(q, a) for q, a in free_food.items()]))
|
||||
context['TOTAL_CC'] = round(context['TOTAL_CC'], 2)
|
||||
context["END_ENERGY"] = sum([amount * FOOD_ENERGY[q] for q, amount in free_food.items()])
|
||||
data = {}
|
||||
for row in sent_data:
|
||||
pid = int(row['player_id'])
|
||||
if pid not in data:
|
||||
data.update({pid: dict(id=pid, name=row['name'], energy=0, cost=0, kills=aviators[pid]['kills'])})
|
||||
|
||||
data[pid]["energy"] += row['energy']
|
||||
data[pid]["cost"] += row['cost']
|
||||
|
||||
for pid, player_data in sorted(aviators.items(), key=lambda t: (-t[1]["health"], -t[1]['kills'])):
|
||||
players.append(player_template.format(id=pid, name=player_data['name']))
|
||||
kills.append(str(player_data['kills']))
|
||||
health.append(str(player_data['health'] or ", ".join(player_data['extra'])))
|
||||
else:
|
||||
context['TABLE'] = columns.format(
|
||||
players="\n".join(players),
|
||||
kills="\n".join(kills),
|
||||
health="\n".join(health)
|
||||
)
|
||||
|
||||
if os.path.isfile("scripts/KM_piloti.txt"):
|
||||
with open("scripts/KM_piloti.txt") as f:
|
||||
template = f.read()
|
||||
article = template.format(**context)
|
||||
with open(get_file("{eday}.txt".format(eday=eday_from_date(now()))), "w") as f:
|
||||
f.write(article)
|
||||
if send_food:
|
||||
article_data = dict(
|
||||
title="[KM] Gaisa maizītes [d{} {}]".format(citizen.eday, citizen.now.strftime("%H:%M")),
|
||||
content=article,
|
||||
kind=3
|
||||
)
|
||||
from_eday = eday_from_date(good_timedelta(now(), - datetime.timedelta(days=now().weekday() + 6)))
|
||||
till_eday = eday_from_date(good_timedelta(now(), - datetime.timedelta(days=now().weekday())))
|
||||
comment_data = dict(
|
||||
message="★★★★ MAIZE PAR NEDĒĻU [DAY {}-{}] IZDALĪTA ★★★★\n★ Apgādei piesakāmies šī komentāra reply "
|
||||
"komentāros ar saucienu - piesakos! ★".format(from_eday, till_eday))
|
||||
total_cc = int(round(context['TOTAL_CC']))
|
||||
wall_body = ("★★★ [ KONGRESA BALSOJUMS ] ★★★\n\nDotācija pilotiem par d{}-{} {}cc apmērā.\n\n"
|
||||
"Balsot ar Par/Pret\nBalsošanas laiks 24h līdz d{} {}").format(
|
||||
from_eday, till_eday, total_cc, citizen.eday + 1, citizen.now.strftime("%H:%M"))
|
||||
|
||||
citizen.write_log("Publishing info:\n\n### Article ###\n{}\n\n{}\n\n### Wall ###\n{}".format(
|
||||
article_data['title'], comment_data['message'], wall_body
|
||||
))
|
||||
|
||||
KM_account: Citizen = Citizen("kara-ministrija@erep.lv", "KMPar0le")
|
||||
KM_account.set_debug(True)
|
||||
KM_account.update_citizen_info()
|
||||
resp = KM_account.publish_article(**article_data)
|
||||
article_id = resp.history[1].url.split("/")[-3]
|
||||
comment_data.update({"article_id": article_id})
|
||||
KM_account.write_article_comment(**comment_data)
|
||||
citizen.write_on_country_wall(wall_body)
|
||||
requests.post('https://erep.lv/aviator/latest_article/',
|
||||
data=dict(week=context["WEEK"], article_id=article_id))
|
6
requirements.txt
Normal file
6
requirements.txt
Normal file
@ -0,0 +1,6 @@
|
||||
ipython==7.3.0
|
||||
pycryptodome==3.7.3
|
||||
PyInstaller==3.4
|
||||
python-slugify==2.0.1
|
||||
pytz==2018.9
|
||||
requests==2.21.0
|
10
requirements_dev.txt
Normal file
10
requirements_dev.txt
Normal file
@ -0,0 +1,10 @@
|
||||
pip==18.1
|
||||
bumpversion==0.5.3
|
||||
wheel==0.32.1
|
||||
watchdog==0.9.0
|
||||
flake8==3.5.0
|
||||
tox==3.5.2
|
||||
coverage==4.5.1
|
||||
Sphinx==1.8.1
|
||||
twine==1.12.1
|
||||
ipython==7.3.0
|
24
setup.cfg
Normal file
24
setup.cfg
Normal file
@ -0,0 +1,24 @@
|
||||
[bumpversion]
|
||||
current_version = 0.1.0
|
||||
commit = True
|
||||
tag = True
|
||||
|
||||
[bumpversion:file:setup.py]
|
||||
search = version='{current_version}'
|
||||
replace = version='{new_version}'
|
||||
|
||||
[bumpversion:file:erepublik_script/__init__.py]
|
||||
search = __version__ = '{current_version}'
|
||||
replace = __version__ = '{new_version}'
|
||||
|
||||
[bdist_wheel]
|
||||
universal = 1
|
||||
|
||||
[flake8]
|
||||
exclude = docs
|
||||
max-line-length = 120
|
||||
ignore = E722
|
||||
|
||||
[aliases]
|
||||
# Define setup.py command aliases here
|
||||
|
55
setup.py
Normal file
55
setup.py
Normal file
@ -0,0 +1,55 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""The setup script."""
|
||||
|
||||
from setuptools import setup, find_packages
|
||||
|
||||
with open('README.rst') as readme_file:
|
||||
readme = readme_file.read()
|
||||
|
||||
with open('HISTORY.rst') as history_file:
|
||||
history = history_file.read()
|
||||
|
||||
requirements = ['Click>=6.0', ]
|
||||
|
||||
setup_requirements = [ ]
|
||||
|
||||
test_requirements = [ ]
|
||||
|
||||
setup(
|
||||
author="Eriks Karls",
|
||||
author_email='eriks@72.lv',
|
||||
classifiers=[
|
||||
'Development Status :: 2 - Pre-Alpha',
|
||||
'Intended Audience :: Developers',
|
||||
'License :: OSI Approved :: MIT License',
|
||||
'Natural Language :: English',
|
||||
"Programming Language :: Python :: 2",
|
||||
'Programming Language :: Python :: 2.7',
|
||||
'Programming Language :: Python :: 3',
|
||||
'Programming Language :: Python :: 3.4',
|
||||
'Programming Language :: Python :: 3.5',
|
||||
'Programming Language :: Python :: 3.6',
|
||||
'Programming Language :: Python :: 3.7',
|
||||
],
|
||||
description="Python package for eRepublik automated playing",
|
||||
entry_points={
|
||||
'console_scripts': [
|
||||
'erepublik_script=erepublik_script.cli:main',
|
||||
],
|
||||
},
|
||||
install_requires=requirements,
|
||||
license="MIT license",
|
||||
long_description=readme + '\n\n' + history,
|
||||
include_package_data=True,
|
||||
keywords='erepublik_script',
|
||||
name='erepublik_script',
|
||||
packages=find_packages(include=['erepublik_script']),
|
||||
setup_requires=setup_requirements,
|
||||
test_suite='tests',
|
||||
tests_require=test_requirements,
|
||||
url='https://github.com/eeriks/erepublik_script',
|
||||
version='0.1.0',
|
||||
zip_safe=False,
|
||||
)
|
3
tests/__init__.py
Normal file
3
tests/__init__.py
Normal file
@ -0,0 +1,3 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""Unit test package for erepublik_script."""
|
34
tests/test_erepublik_script.py
Normal file
34
tests/test_erepublik_script.py
Normal file
@ -0,0 +1,34 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
"""Tests for `erepublik_script` package."""
|
||||
|
||||
|
||||
import unittest
|
||||
from click.testing import CliRunner
|
||||
|
||||
from erepublik_script import erepublik_script
|
||||
from erepublik_script import cli
|
||||
|
||||
|
||||
class TestErepublik_script(unittest.TestCase):
|
||||
"""Tests for `erepublik_script` package."""
|
||||
|
||||
def setUp(self):
|
||||
"""Set up test fixtures, if any."""
|
||||
|
||||
def tearDown(self):
|
||||
"""Tear down test fixtures, if any."""
|
||||
|
||||
def test_000_something(self):
|
||||
"""Test something."""
|
||||
|
||||
def test_command_line_interface(self):
|
||||
"""Test the CLI."""
|
||||
runner = CliRunner()
|
||||
result = runner.invoke(cli.main)
|
||||
assert result.exit_code == 0
|
||||
assert 'erepublik_script.cli.main' in result.output
|
||||
help_result = runner.invoke(cli.main, ['--help'])
|
||||
assert help_result.exit_code == 0
|
||||
assert '--help Show this message and exit.' in help_result.output
|
21
tox.ini
Normal file
21
tox.ini
Normal file
@ -0,0 +1,21 @@
|
||||
[tox]
|
||||
envlist = py27, py34, py35, py36, flake8
|
||||
|
||||
[travis]
|
||||
python =
|
||||
3.6: py36
|
||||
3.5: py35
|
||||
3.4: py34
|
||||
2.7: py27
|
||||
|
||||
[testenv:flake8]
|
||||
basepython = python
|
||||
deps = flake8
|
||||
commands = flake8 erepublik_script
|
||||
|
||||
[testenv]
|
||||
setenv =
|
||||
PYTHONPATH = {toxinidir}
|
||||
|
||||
commands = python setup.py test
|
||||
|
Reference in New Issue
Block a user