mirror of
https://github.com/arcan1s/ffxivbis.git
synced 2025-10-24 05:39:58 +00:00
Compare commits
1 Commits
0.9.2
...
feature/pa
Author | SHA1 | Date | |
---|---|---|---|
4ff985bf81 |
162
.gitignore
vendored
162
.gitignore
vendored
@ -1,88 +1,96 @@
|
|||||||
#### joe made this: http://goel.io/joe
|
# Byte-compiled / optimized / DLL files
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
||||||
|
|
||||||
#### jetbrains ####
|
# C extensions
|
||||||
|
*.so
|
||||||
|
|
||||||
# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and Webstorm
|
# Distribution / packaging
|
||||||
# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
|
.Python
|
||||||
|
env/
|
||||||
|
build/
|
||||||
|
develop-eggs/
|
||||||
|
dist/
|
||||||
|
downloads/
|
||||||
|
eggs/
|
||||||
|
.eggs/
|
||||||
|
parts/
|
||||||
|
sdist/
|
||||||
|
*.egg-info/
|
||||||
|
.installed.cfg
|
||||||
|
*.egg
|
||||||
|
|
||||||
# User-specific stuff:
|
# PyInstaller
|
||||||
.idea
|
# 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
|
||||||
|
|
||||||
## File-based project format:
|
# Installer logs
|
||||||
*.iws
|
pip-log.txt
|
||||||
|
pip-delete-this-directory.txt
|
||||||
|
|
||||||
## Plugin-specific files:
|
# Unit test / coverage reports
|
||||||
|
htmlcov/
|
||||||
# IntelliJ
|
.tox/
|
||||||
/out/
|
.coverage
|
||||||
|
.coverage.*
|
||||||
# mpeltonen/sbt-idea plugin
|
|
||||||
.idea_modules/
|
|
||||||
|
|
||||||
# JIRA plugin
|
|
||||||
atlassian-ide-plugin.xml
|
|
||||||
|
|
||||||
# Crashlytics plugin (for Android Studio and IntelliJ)
|
|
||||||
com_crashlytics_export_strings.xml
|
|
||||||
crashlytics.properties
|
|
||||||
crashlytics-build.properties
|
|
||||||
fabric.properties
|
|
||||||
|
|
||||||
#### gradle ####
|
|
||||||
|
|
||||||
.gradle
|
|
||||||
/build/
|
|
||||||
|
|
||||||
# Ignore Gradle GUI config
|
|
||||||
gradle-app.setting
|
|
||||||
|
|
||||||
# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
|
|
||||||
!gradle-wrapper.jar
|
|
||||||
|
|
||||||
# Cache of project
|
|
||||||
.gradletasknamecache
|
|
||||||
|
|
||||||
# # Work around https://youtrack.jetbrains.com/issue/IDEA-116898
|
|
||||||
# gradle/wrapper/gradle-wrapper.properties
|
|
||||||
|
|
||||||
#### java ####
|
|
||||||
|
|
||||||
*.class
|
|
||||||
|
|
||||||
# Mobile Tools for Java (J2ME)
|
|
||||||
.mtj.tmp/
|
|
||||||
|
|
||||||
# Package Files #
|
|
||||||
*.jar
|
|
||||||
*.war
|
|
||||||
*.ear
|
|
||||||
|
|
||||||
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
|
|
||||||
hs_err_pid*
|
|
||||||
|
|
||||||
#### scala ####
|
|
||||||
|
|
||||||
*.class
|
|
||||||
*.log
|
|
||||||
|
|
||||||
# sbt specific
|
|
||||||
.cache
|
.cache
|
||||||
.history
|
nosetests.xml
|
||||||
.lib/
|
coverage.xml
|
||||||
dist/*
|
*,cover
|
||||||
|
.hypothesis/
|
||||||
|
|
||||||
|
# Translations
|
||||||
|
*.mo
|
||||||
|
*.pot
|
||||||
|
|
||||||
|
# Django stuff:
|
||||||
|
*.log*
|
||||||
|
local_settings.py
|
||||||
|
|
||||||
|
# Flask stuff:
|
||||||
|
instance/
|
||||||
|
.webassets-cache
|
||||||
|
|
||||||
|
# Scrapy stuff:
|
||||||
|
.scrapy
|
||||||
|
|
||||||
|
# Sphinx documentation
|
||||||
|
docs/_build/
|
||||||
|
|
||||||
|
# PyBuilder
|
||||||
target/
|
target/
|
||||||
lib_managed/
|
|
||||||
src_managed/
|
|
||||||
project/boot/
|
|
||||||
project/plugins/project/
|
|
||||||
|
|
||||||
# Scala-IDE specific
|
# IPython Notebook
|
||||||
.scala_dependencies
|
.ipynb_checkpoints
|
||||||
.worksheet
|
|
||||||
|
|
||||||
# ENSIME specific
|
# pyenv
|
||||||
.ensime_cache/
|
.python-version
|
||||||
.ensime
|
|
||||||
|
# celery beat schedule file
|
||||||
|
celerybeat-schedule
|
||||||
|
|
||||||
|
# dotenv
|
||||||
|
.env
|
||||||
|
|
||||||
|
# virtualenv
|
||||||
|
venv/
|
||||||
|
ENV/
|
||||||
|
|
||||||
|
# Spyder project settings
|
||||||
|
.spyderproject
|
||||||
|
|
||||||
|
# Rope project settings
|
||||||
|
.ropeproject
|
||||||
|
|
||||||
|
*.deb
|
||||||
|
|
||||||
|
.idea/
|
||||||
|
|
||||||
|
.mypy_cache/
|
||||||
|
|
||||||
|
/cache
|
||||||
|
|
||||||
*.db
|
*.db
|
||||||
*.sc
|
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
sudo: required
|
|
||||||
language: generic
|
|
||||||
|
|
||||||
services:
|
|
||||||
- docker
|
|
||||||
|
|
||||||
script:
|
|
||||||
- docker run -it --rm -v "$(pwd):/opt/build" -w /opt/build mozilla/sbt sbt compile
|
|
||||||
- docker run -it --rm -v "$(pwd):/opt/build" -w /opt/build mozilla/sbt sbt test
|
|
86
README.md
86
README.md
@ -4,20 +4,98 @@ Service which allows to manage savage loot distribution easy.
|
|||||||
|
|
||||||
## Installation and usage
|
## Installation and usage
|
||||||
|
|
||||||
|
This service requires python >= 3.7. For other dependencies see `setup.py`.
|
||||||
|
|
||||||
In general installation process looks like:
|
In general installation process looks like:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sbt assembly
|
python setup.py build install
|
||||||
|
python setup.py test # if you want to run tests
|
||||||
```
|
```
|
||||||
|
|
||||||
Or alternatively you can download latest jar from releases page. Service can be run by using command:
|
With virtualenv (make sure that virtualenv package was installed) the process may look like:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
java -cp ./target/scala-2.13/ffxivbis-scala-assembly-0.1.jar me.arcanis.ffxivbis.ffxivbis
|
virtualenv -p python3.7 env
|
||||||
|
source env/bin/activate
|
||||||
|
python setup.py install
|
||||||
|
pip install aiosqlite # setup.py does not handle extras
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Service can be run by using command (if you don't use virtualenv, you have to run it from `src` directory):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python -m ffxivbis.application.application
|
||||||
|
```
|
||||||
|
|
||||||
|
To see all available options type `--help`.
|
||||||
|
|
||||||
## Web service
|
## Web service
|
||||||
|
|
||||||
REST API documentation is available at `http://0.0.0.0:8000/swagger`. HTML representation is available at `http://0.0.0.0:8000`.
|
REST API documentation is available at `http://0.0.0.0:8000/api-docs`. HTML representation is available at `http://0.0.0.0:8000`.
|
||||||
|
|
||||||
*Note*: host and port depend on configuration settings.
|
*Note*: host and port depend on configuration settings.
|
||||||
|
|
||||||
|
### Authorization
|
||||||
|
|
||||||
|
Default admin user is `admin:qwerty`, but it may be changed by generating new hash, e.g.:
|
||||||
|
|
||||||
|
```python
|
||||||
|
from passlib.hash import md5_crypt
|
||||||
|
md5_crypt.hash('newstrongpassword')
|
||||||
|
```
|
||||||
|
|
||||||
|
and add new password to configuration.
|
||||||
|
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
* `settings` section
|
||||||
|
|
||||||
|
General project settings.
|
||||||
|
|
||||||
|
* `include`: path to include configuration directory, string, optional.
|
||||||
|
* `logging`: path to logging configuration, see `logging.ini` for reference, string, optional.
|
||||||
|
* `database`: database provide name, string, required. Allowed values: `sqlite`, `postgres`.
|
||||||
|
* `priority`: methods of `Player` class which will be called to sort players for loot priority, space separated list of strings, required.
|
||||||
|
|
||||||
|
* `ariyala` section
|
||||||
|
|
||||||
|
Settings related to ariyala parser.
|
||||||
|
|
||||||
|
* `ariyala_url`: ariyala base url, string, required.
|
||||||
|
* `xivapi_key`: xivapi developer key, string, optional.
|
||||||
|
* `xivapi_url`: xivapi base url, string, required.
|
||||||
|
|
||||||
|
* `auth` section
|
||||||
|
|
||||||
|
Authentication settings.
|
||||||
|
|
||||||
|
* `enabled`: whether authentication enabled or not, boolean, required.
|
||||||
|
* `root_username`: username of administrator, string, required.
|
||||||
|
* `root_password`: md5 hashed password of administrator, string, required.
|
||||||
|
|
||||||
|
* `postgres` section
|
||||||
|
|
||||||
|
Database settings for `postgres` provider.
|
||||||
|
|
||||||
|
* `database`: database name, string, required.
|
||||||
|
* `host`: database host, string, required.
|
||||||
|
* `password`: database password, string, required.
|
||||||
|
* `port`: database port, int, required.
|
||||||
|
* `username`: database username, string, required.
|
||||||
|
* `migrations_path`: path to database migrations, string, required.
|
||||||
|
|
||||||
|
* `sqlite` section
|
||||||
|
|
||||||
|
Database settings for `sqlite` provider.
|
||||||
|
|
||||||
|
* `database_path`: path to sqlite database, string, required.
|
||||||
|
* `migrations_path`: path to database migrations, string, required.
|
||||||
|
|
||||||
|
* `web` section
|
||||||
|
|
||||||
|
Web server related settings.
|
||||||
|
|
||||||
|
* `host`: address to bind, string, required.
|
||||||
|
* `port`: port to bind, int, required.
|
||||||
|
* `templates`: path to directory with jinja templates, string, required.
|
4
TODO.md
4
TODO.md
@ -1,3 +1,3 @@
|
|||||||
* [x] items improvements
|
* [ ] items improvements
|
||||||
* [x] multiple parties support
|
* [ ] multiple parties support
|
||||||
* [ ] pretty UI
|
* [ ] pretty UI
|
14
build.sbt
14
build.sbt
@ -1,14 +0,0 @@
|
|||||||
name := "ffxivbis"
|
|
||||||
|
|
||||||
scalaVersion := "2.13.1"
|
|
||||||
|
|
||||||
scalacOptions ++= Seq("-deprecation", "-feature")
|
|
||||||
|
|
||||||
assemblyMergeStrategy in assembly := {
|
|
||||||
case PathList("META-INF", xs @ _*) => MergeStrategy.discard
|
|
||||||
case "application.conf" => MergeStrategy.concat
|
|
||||||
case "module-info.class" => MergeStrategy.first
|
|
||||||
case x =>
|
|
||||||
val oldStrategy = (assemblyMergeStrategy in assembly).value
|
|
||||||
oldStrategy(x)
|
|
||||||
}
|
|
@ -1,19 +0,0 @@
|
|||||||
libraryDependencies += "ch.qos.logback" % "logback-classic" % "1.2.3"
|
|
||||||
libraryDependencies += "com.typesafe.scala-logging" %% "scala-logging" % "3.9.2"
|
|
||||||
|
|
||||||
libraryDependencies += "com.typesafe.akka" %% "akka-http" % "10.1.10"
|
|
||||||
libraryDependencies += "com.typesafe.akka" %% "akka-http-spray-json" % "10.1.10"
|
|
||||||
libraryDependencies += "com.typesafe.akka" %% "akka-stream" % "2.5.23"
|
|
||||||
libraryDependencies += "com.github.swagger-akka-http" %% "swagger-akka-http" % "2.0.4"
|
|
||||||
libraryDependencies += "javax.ws.rs" % "javax.ws.rs-api" % "2.1.1"
|
|
||||||
|
|
||||||
libraryDependencies += "io.spray" %% "spray-json" % "1.3.5"
|
|
||||||
libraryDependencies += "com.lihaoyi" %% "scalatags" % "0.7.0"
|
|
||||||
|
|
||||||
libraryDependencies += "com.typesafe.slick" %% "slick" % "3.3.2"
|
|
||||||
libraryDependencies += "com.typesafe.slick" %% "slick-hikaricp" % "3.3.2"
|
|
||||||
libraryDependencies += "org.flywaydb" % "flyway-core" % "6.0.6"
|
|
||||||
libraryDependencies += "org.xerial" % "sqlite-jdbc" % "3.28.0"
|
|
||||||
libraryDependencies += "org.postgresql" % "postgresql" % "9.3-1104-jdbc4"
|
|
||||||
|
|
||||||
libraryDependencies += "org.mindrot" % "jbcrypt" % "0.3m"
|
|
38
migrations/20190830_01_sYYZL-init-tables.py
Normal file
38
migrations/20190830_01_sYYZL-init-tables.py
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
'''
|
||||||
|
init tables
|
||||||
|
'''
|
||||||
|
|
||||||
|
from yoyo import step
|
||||||
|
|
||||||
|
__depends__ = {}
|
||||||
|
|
||||||
|
steps = [
|
||||||
|
step('''create table players (
|
||||||
|
player_id integer primary key,
|
||||||
|
created integer not null,
|
||||||
|
nick text not null,
|
||||||
|
job text not null,
|
||||||
|
bis_link text,
|
||||||
|
priority integer not null default 1
|
||||||
|
)'''),
|
||||||
|
step('''create unique index players_nick_job_idx on players(nick, job)'''),
|
||||||
|
|
||||||
|
step('''create table loot (
|
||||||
|
loot_id integer primary key,
|
||||||
|
player_id integer not null,
|
||||||
|
created integer not null,
|
||||||
|
piece text not null,
|
||||||
|
is_tome integer not null,
|
||||||
|
foreign key (player_id) references players(player_id) on delete cascade
|
||||||
|
)'''),
|
||||||
|
step('''create index loot_owner_idx on loot(player_id)'''),
|
||||||
|
|
||||||
|
step('''create table bis (
|
||||||
|
player_id integer not null,
|
||||||
|
created integer not null,
|
||||||
|
piece text not null,
|
||||||
|
is_tome integer not null,
|
||||||
|
foreign key (player_id) references players(player_id) on delete cascade
|
||||||
|
)'''),
|
||||||
|
step('''create unique index bis_piece_player_id_idx on bis(player_id, piece)''')
|
||||||
|
]
|
17
migrations/20190910_01_tgBmx-users-table.py
Normal file
17
migrations/20190910_01_tgBmx-users-table.py
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
'''
|
||||||
|
users table
|
||||||
|
'''
|
||||||
|
|
||||||
|
from yoyo import step
|
||||||
|
|
||||||
|
__depends__ = {}
|
||||||
|
|
||||||
|
steps = [
|
||||||
|
step('''create table users (
|
||||||
|
user_id integer primary key,
|
||||||
|
username text not null,
|
||||||
|
password text not null,
|
||||||
|
permission text not null
|
||||||
|
)'''),
|
||||||
|
step('''create unique index users_username_idx on users(username)''')
|
||||||
|
]
|
75
migrations/20190916_01_zGTB1-party-id.py
Normal file
75
migrations/20190916_01_zGTB1-party-id.py
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
'''
|
||||||
|
party id
|
||||||
|
'''
|
||||||
|
|
||||||
|
import random
|
||||||
|
import string
|
||||||
|
|
||||||
|
from yoyo import step
|
||||||
|
|
||||||
|
__depends__ = {'20190830_01_sYYZL-init-tables', '20190910_01_tgBmx-users-table'}
|
||||||
|
party_id = ''.join(random.sample(string.ascii_letters, 16))
|
||||||
|
|
||||||
|
steps = [
|
||||||
|
step('''create table players2 (
|
||||||
|
party_id text not null,
|
||||||
|
player_id integer primary key,
|
||||||
|
created integer not null,
|
||||||
|
nick text not null,
|
||||||
|
job text not null,
|
||||||
|
bis_link text,
|
||||||
|
priority integer not null default 1
|
||||||
|
)'''),
|
||||||
|
# not safe for injections, but sqlite and psycopg have different placeholders for parameters
|
||||||
|
step('''insert into players2 select '%s' as party_id, players.* from players''' % (party_id,)),
|
||||||
|
step('''drop index if exists players_nick_job_idx'''),
|
||||||
|
step('''create unique index players_nick_job_idx on players2(party_id, nick, job)'''),
|
||||||
|
|
||||||
|
step('''create table loot2 (
|
||||||
|
loot_id integer primary key,
|
||||||
|
player_id integer not null,
|
||||||
|
created integer not null,
|
||||||
|
piece text not null,
|
||||||
|
is_tome integer not null,
|
||||||
|
foreign key (player_id) references players2(player_id) on delete cascade
|
||||||
|
)'''),
|
||||||
|
step('''insert into loot2 select * from loot'''),
|
||||||
|
step('''drop index if exists loot_owner_idx'''),
|
||||||
|
step('''create index loot_owner_idx on loot(player_id)'''),
|
||||||
|
|
||||||
|
step('''create table bis2 (
|
||||||
|
player_id integer not null,
|
||||||
|
created integer not null,
|
||||||
|
piece text not null,
|
||||||
|
is_tome integer not null,
|
||||||
|
foreign key (player_id) references players2(player_id) on delete cascade
|
||||||
|
)'''),
|
||||||
|
step('''insert into bis2 select * from bis'''),
|
||||||
|
step('''drop index if exists bis_piece_player_id_idx'''),
|
||||||
|
step('''create unique index bis_piece_player_id_idx on bis2(player_id, piece)'''),
|
||||||
|
|
||||||
|
step('''create table users2 (
|
||||||
|
party_id text not null,
|
||||||
|
user_id integer primary key,
|
||||||
|
username text not null,
|
||||||
|
password text not null,
|
||||||
|
permission text not null,
|
||||||
|
foreign key (party_id) references players2(party_id) on delete cascade
|
||||||
|
)'''),
|
||||||
|
# not safe for injections, but sqlite and psycopg have different placeholders for parameters
|
||||||
|
step('''insert into users2 select '%s' as party_id, users.* from users''' % (party_id,)),
|
||||||
|
step('''drop index if exists users_username_idx'''),
|
||||||
|
step('''create unique index users_username_idx on users2(party_id, username)'''),
|
||||||
|
|
||||||
|
step('''drop table users'''),
|
||||||
|
step('''alter table users2 rename to users'''),
|
||||||
|
|
||||||
|
step('''drop table loot'''),
|
||||||
|
step('''alter table loot2 rename to loot'''),
|
||||||
|
|
||||||
|
step('''drop table bis'''),
|
||||||
|
step('''alter table bis2 rename to bis'''),
|
||||||
|
|
||||||
|
step('''drop table players'''),
|
||||||
|
step('''alter table players2 rename to players''')
|
||||||
|
]
|
10
package/ini/ffxivbis.ini
Normal file
10
package/ini/ffxivbis.ini
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
[settings]
|
||||||
|
include = ffxivbis.ini.d
|
||||||
|
logging = ffxivbis.ini.d/logging.ini
|
||||||
|
database = sqlite
|
||||||
|
priority = is_required loot_count_bis loot_priority loot_count loot_count_total
|
||||||
|
|
||||||
|
[web]
|
||||||
|
host = 0.0.0.0
|
||||||
|
port = 8000
|
||||||
|
templates = /home/arcanis/Documents/github/ffxivbis/templates
|
3
package/ini/ffxivbis.ini.d/ariyala.ini
Normal file
3
package/ini/ffxivbis.ini.d/ariyala.ini
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[ariyala]
|
||||||
|
ariyala_url = https://ffxiv.ariyala.com
|
||||||
|
xivapi_url = https://xivapi.com
|
4
package/ini/ffxivbis.ini.d/auth.ini
Normal file
4
package/ini/ffxivbis.ini.d/auth.ini
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
[auth]
|
||||||
|
enabled = yes
|
||||||
|
root_username = admin
|
||||||
|
root_password = $1$R3j4sym6$HtvrKOJ66f7w3.9Zc3U6h1
|
44
package/ini/ffxivbis.ini.d/logging.ini
Normal file
44
package/ini/ffxivbis.ini.d/logging.ini
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
[loggers]
|
||||||
|
keys = root,application,database,http
|
||||||
|
|
||||||
|
[handlers]
|
||||||
|
keys = file_handler
|
||||||
|
|
||||||
|
[formatters]
|
||||||
|
keys = generic_format
|
||||||
|
|
||||||
|
[handler_console_handler]
|
||||||
|
class = StreamHandler
|
||||||
|
level = INFO
|
||||||
|
formatter = generic_format
|
||||||
|
args = (sys.stdout,)
|
||||||
|
|
||||||
|
[handler_file_handler]
|
||||||
|
class = logging.handlers.RotatingFileHandler
|
||||||
|
level = INFO
|
||||||
|
formatter = generic_format
|
||||||
|
args = ('ffxivbis.log', 'a', 20971520, 20)
|
||||||
|
|
||||||
|
[formatter_generic_format]
|
||||||
|
format = [%(levelname)s] [%(asctime)s] [%(threadName)s] [%(name)s] [%(funcName)s]: %(message)s
|
||||||
|
datefmt =
|
||||||
|
|
||||||
|
[logger_root]
|
||||||
|
level = INFO
|
||||||
|
handlers = file_handler
|
||||||
|
qualname = root
|
||||||
|
|
||||||
|
[logger_application]
|
||||||
|
level = INFO
|
||||||
|
handlers = file_handler
|
||||||
|
qualname = application
|
||||||
|
|
||||||
|
[logger_database]
|
||||||
|
level = INFO
|
||||||
|
handlers = file_handler
|
||||||
|
qualname = database
|
||||||
|
|
||||||
|
[logger_http]
|
||||||
|
level = INFO
|
||||||
|
handlers = file_handler
|
||||||
|
qualname = http
|
3
package/ini/ffxivbis.ini.d/sqlite.ini
Normal file
3
package/ini/ffxivbis.ini.d/sqlite.ini
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[sqlite]
|
||||||
|
database_path = /home/arcanis/Documents/github/ffxivbis/ffxivbis.db
|
||||||
|
migrations_path = /home/arcanis/Documents/github/ffxivbis/migrations
|
@ -1 +0,0 @@
|
|||||||
addSbtPlugin("com.eed3si9n" % "sbt-assembly" % "0.14.6")
|
|
@ -1 +0,0 @@
|
|||||||
sbt.version = 1.3.3
|
|
@ -1 +0,0 @@
|
|||||||
addSbtPlugin("net.virtual-void" % "sbt-dependency-graph" % "0.10.0-RC1")
|
|
5
setup.cfg
Normal file
5
setup.cfg
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
[aliases]
|
||||||
|
test=pytest
|
||||||
|
|
||||||
|
[tool:pytest]
|
||||||
|
addopts = --verbose --pyargs .
|
52
setup.py
Normal file
52
setup.py
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
from distutils.util import convert_path
|
||||||
|
from setuptools import setup, find_packages
|
||||||
|
from os import path
|
||||||
|
|
||||||
|
|
||||||
|
here = path.abspath(path.dirname(__file__))
|
||||||
|
metadata = dict()
|
||||||
|
with open(convert_path('src/ffxivbis/core/version.py')) as metadata_file:
|
||||||
|
exec(metadata_file.read(), metadata)
|
||||||
|
|
||||||
|
|
||||||
|
setup(
|
||||||
|
name='ffxivbis',
|
||||||
|
|
||||||
|
version=metadata['__version__'],
|
||||||
|
zip_safe=False,
|
||||||
|
|
||||||
|
description='Helper to handle loot drop',
|
||||||
|
|
||||||
|
author='Evgeniy Alekseev',
|
||||||
|
author_email='i@arcanis.me',
|
||||||
|
|
||||||
|
license='BSD',
|
||||||
|
|
||||||
|
package_dir={'': 'src'},
|
||||||
|
packages=find_packages(where='src', exclude=['contrib', 'docs', 'test']),
|
||||||
|
|
||||||
|
install_requires=[
|
||||||
|
'aiohttp==3.6.0',
|
||||||
|
'aiohttp_jinja2',
|
||||||
|
'aiohttp_security',
|
||||||
|
'apispec',
|
||||||
|
'iniherit',
|
||||||
|
'Jinja2',
|
||||||
|
'passlib',
|
||||||
|
'yoyo_migrations'
|
||||||
|
],
|
||||||
|
setup_requires=[
|
||||||
|
'pytest-runner'
|
||||||
|
],
|
||||||
|
tests_require=[
|
||||||
|
'pytest', 'pytest-aiohttp', 'pytest-asyncio'
|
||||||
|
],
|
||||||
|
|
||||||
|
include_package_data=True,
|
||||||
|
|
||||||
|
extras_require={
|
||||||
|
'Postgresql': ['asyncpg'],
|
||||||
|
'SQLite': ['aiosqlite'],
|
||||||
|
'test': ['coverage', 'pytest'],
|
||||||
|
},
|
||||||
|
)
|
0
src/ffxivbis/__init__.py
Normal file
0
src/ffxivbis/__init__.py
Normal file
0
src/ffxivbis/api/__init__.py
Normal file
0
src/ffxivbis/api/__init__.py
Normal file
60
src/ffxivbis/api/auth.py
Normal file
60
src/ffxivbis/api/auth.py
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import middleware, Request, Response
|
||||||
|
from aiohttp_security import AbstractAuthorizationPolicy, check_permission
|
||||||
|
from typing import Callable, Optional, Tuple
|
||||||
|
|
||||||
|
from ffxivbis.core.database import Database
|
||||||
|
|
||||||
|
|
||||||
|
class AuthorizationPolicy(AbstractAuthorizationPolicy):
|
||||||
|
|
||||||
|
def __init__(self, database: Database) -> None:
|
||||||
|
self.database = database
|
||||||
|
|
||||||
|
def split_identity(self, identity: str) -> Tuple[str, str]:
|
||||||
|
# identity is party_id + username
|
||||||
|
party_id, username = identity.split('+')
|
||||||
|
return party_id, username
|
||||||
|
|
||||||
|
async def authorized_userid(self, identity: str) -> Optional[str]:
|
||||||
|
party_id, username = self.split_identity(identity)
|
||||||
|
user = await self.database.get_user(party_id, username)
|
||||||
|
return username if user is not None else None
|
||||||
|
|
||||||
|
async def permits(self, identity: str, permission: str, context: str = None) -> bool:
|
||||||
|
party_id, username = self.split_identity(identity)
|
||||||
|
user = await self.database.get_user(party_id, username)
|
||||||
|
if user is None:
|
||||||
|
return False
|
||||||
|
if user.username != identity:
|
||||||
|
return False
|
||||||
|
if user.permission == 'admin':
|
||||||
|
return True
|
||||||
|
return permission == 'get' or user.permission == permission
|
||||||
|
|
||||||
|
|
||||||
|
def authorize_factory() -> Callable:
|
||||||
|
allowed_paths = {'/', '/favicon.ico', '/api/v1/login', '/api/v1/logout'}
|
||||||
|
allowed_paths_groups = {'/api-docs', '/static'}
|
||||||
|
|
||||||
|
@middleware
|
||||||
|
async def authorize(request: Request, handler: Callable) -> Response:
|
||||||
|
if request.path.startswith('/admin'):
|
||||||
|
permission = 'admin'
|
||||||
|
else:
|
||||||
|
permission = 'get' if request.method in ('GET', 'HEAD') else 'post'
|
||||||
|
if request.path not in allowed_paths \
|
||||||
|
and not any(request.path.startswith(path) for path in allowed_paths_groups):
|
||||||
|
await check_permission(request, permission)
|
||||||
|
|
||||||
|
return await handler(request)
|
||||||
|
|
||||||
|
return authorize
|
||||||
|
|
34
src/ffxivbis/api/json.py
Normal file
34
src/ffxivbis/api/json.py
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from enum import Enum
|
||||||
|
from json import JSONEncoder
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
|
||||||
|
class HttpEncoder(JSONEncoder):
|
||||||
|
def default(self, obj: Any) -> Any:
|
||||||
|
if isinstance(obj, dict):
|
||||||
|
data = {}
|
||||||
|
for key, value in obj.items():
|
||||||
|
data[key] = self.default(value)
|
||||||
|
return data
|
||||||
|
elif isinstance(obj, Enum):
|
||||||
|
return obj.name
|
||||||
|
elif hasattr(obj, '_ast'):
|
||||||
|
return self.default(obj._ast())
|
||||||
|
elif hasattr(obj, '__iter__') and not isinstance(obj, str):
|
||||||
|
return [self.default(value) for value in obj]
|
||||||
|
elif hasattr(obj, '__dict__'):
|
||||||
|
data = {
|
||||||
|
key: self.default(value)
|
||||||
|
for key, value in obj.__dict__.items()
|
||||||
|
if not callable(value) and not key.startswith('_')}
|
||||||
|
return data
|
||||||
|
else:
|
||||||
|
return obj
|
66
src/ffxivbis/api/routes.py
Normal file
66
src/ffxivbis/api/routes.py
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Application
|
||||||
|
|
||||||
|
from .views.api.bis import BiSView
|
||||||
|
from .views.api.login import LoginView
|
||||||
|
from .views.api.logout import LogoutView
|
||||||
|
from .views.api.loot import LootView
|
||||||
|
from .views.api.player import PlayerView
|
||||||
|
from .views.html.api import ApiDocVIew, ApiHtmlView
|
||||||
|
from .views.html.bis import BiSHtmlView
|
||||||
|
from .views.html.index import IndexHtmlView
|
||||||
|
from .views.html.loot import LootHtmlView
|
||||||
|
from .views.html.loot_suggest import LootSuggestHtmlView
|
||||||
|
from .views.html.player import PlayerHtmlView
|
||||||
|
from .views.html.static import StaticHtmlView
|
||||||
|
from .views.html.users import UsersHtmlView
|
||||||
|
|
||||||
|
|
||||||
|
def setup_routes(app: Application) -> None:
|
||||||
|
# api routes
|
||||||
|
app.router.add_delete('/admin/api/v1/{party_id}/login/{username}', LoginView)
|
||||||
|
app.router.add_post('/api/v1/{party_id}/login', LoginView)
|
||||||
|
app.router.add_post('/api/v1/{party_id}/logout', LogoutView)
|
||||||
|
app.router.add_put('/admin/api/v1/{party_id}/login', LoginView)
|
||||||
|
|
||||||
|
app.router.add_get('/api/v1/party/{party_id}', PlayerView)
|
||||||
|
app.router.add_post('/api/v1/party/{party_id}', PlayerView)
|
||||||
|
|
||||||
|
app.router.add_get('/api/v1/party/{party_id}/bis', BiSView)
|
||||||
|
app.router.add_post('/api/v1/party/{party_id}/bis', BiSView)
|
||||||
|
app.router.add_put('/api/v1/party/{party_id}/bis', BiSView)
|
||||||
|
|
||||||
|
app.router.add_get('/api/v1/party/{party_id}/loot', LootView)
|
||||||
|
app.router.add_post('/api/v1/party/{party_id}/loot', LootView)
|
||||||
|
app.router.add_put('/api/v1/party/{party_id}/loot', LootView)
|
||||||
|
|
||||||
|
# html routes
|
||||||
|
app.router.add_get('/', IndexHtmlView)
|
||||||
|
app.router.add_get('/static/{resource_id}', StaticHtmlView)
|
||||||
|
|
||||||
|
app.router.add_get('/api-docs', ApiHtmlView)
|
||||||
|
app.router.add_get('/api-docs/swagger.json', ApiDocVIew)
|
||||||
|
|
||||||
|
app.router.add_get('/party/{party_id}', PlayerHtmlView)
|
||||||
|
app.router.add_post('/party/{party_id}', PlayerHtmlView)
|
||||||
|
|
||||||
|
app.router.add_get('/bis/{party_id}', BiSHtmlView)
|
||||||
|
app.router.add_post('/bis/{party_id}', BiSHtmlView)
|
||||||
|
|
||||||
|
app.router.add_get('/loot/{party_id}', LootHtmlView)
|
||||||
|
app.router.add_post('/loot/{party_id}', LootHtmlView)
|
||||||
|
|
||||||
|
app.router.add_get('/suggest/{party_id}', LootSuggestHtmlView)
|
||||||
|
app.router.add_post('/suggest/{party_id}', LootSuggestHtmlView)
|
||||||
|
|
||||||
|
app.router.add_get('/admin/users/{party_id}', UsersHtmlView)
|
||||||
|
app.router.add_post('/admin/users/{party_id}', UsersHtmlView)
|
||||||
|
|
||||||
|
|
78
src/ffxivbis/api/spec.py
Normal file
78
src/ffxivbis/api/spec.py
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Application
|
||||||
|
from apispec import APISpec
|
||||||
|
|
||||||
|
from ffxivbis.core.version import __version__
|
||||||
|
from ffxivbis.models.action import Action
|
||||||
|
from ffxivbis.models.bis import BiS, BiSLink
|
||||||
|
from ffxivbis.models.error import Error
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.loot import Loot
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId, PlayerIdWithCounters
|
||||||
|
from ffxivbis.models.player_edit import PlayerEdit
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
|
||||||
|
def get_spec(app: Application) -> APISpec:
|
||||||
|
spec = APISpec(
|
||||||
|
title='FFXIV loot helper',
|
||||||
|
version=__version__,
|
||||||
|
openapi_version='3.0.2',
|
||||||
|
info=dict(description='Loot manager for FFXIV statics'),
|
||||||
|
)
|
||||||
|
|
||||||
|
# routes
|
||||||
|
for route in app.router.routes():
|
||||||
|
path = route.get_info().get('path') or route.get_info().get('formatter')
|
||||||
|
method = route.method.lower()
|
||||||
|
|
||||||
|
spec_method = f'endpoint_{method}_spec'
|
||||||
|
if not hasattr(route.handler, spec_method):
|
||||||
|
continue
|
||||||
|
operations = getattr(route.handler, spec_method)()
|
||||||
|
if not operations:
|
||||||
|
continue
|
||||||
|
|
||||||
|
spec.path(path, operations={method: operations})
|
||||||
|
|
||||||
|
# components
|
||||||
|
spec.components.schema(Action.model_name(), Action.model_spec())
|
||||||
|
spec.components.schema(BiS.model_name(), BiS.model_spec())
|
||||||
|
spec.components.schema(BiSLink.model_name(), BiSLink.model_spec())
|
||||||
|
spec.components.schema(Error.model_name(), Error.model_spec())
|
||||||
|
spec.components.schema(Job.model_name(), Job.model_spec())
|
||||||
|
spec.components.schema(Loot.model_name(), Loot.model_spec())
|
||||||
|
spec.components.schema(Piece.model_name(), Piece.model_spec())
|
||||||
|
spec.components.schema(Player.model_name(), Player.model_spec())
|
||||||
|
spec.components.schema(PlayerEdit.model_name(), PlayerEdit.model_spec())
|
||||||
|
spec.components.schema(PlayerId.model_name(), PlayerId.model_spec())
|
||||||
|
spec.components.schema(PlayerIdWithCounters.model_name(), PlayerIdWithCounters.model_spec())
|
||||||
|
spec.components.schema(Upgrade.model_name(), Upgrade.model_spec())
|
||||||
|
spec.components.schema(User.model_name(), User.model_spec())
|
||||||
|
|
||||||
|
# default responses
|
||||||
|
spec.components.response('BadRequest', dict(
|
||||||
|
description='Bad parameters applied or bad request was formed',
|
||||||
|
content={'application/json': {'schema': {'$ref': Error.model_ref('Error')}}}
|
||||||
|
))
|
||||||
|
spec.components.response('Forbidden', dict(
|
||||||
|
description='User permissions do not allow this action'
|
||||||
|
))
|
||||||
|
spec.components.response('ServerError', dict(
|
||||||
|
description='Server was unable to process request',
|
||||||
|
content={'application/json': {'schema': {'$ref': Error.model_ref('Error')}}}
|
||||||
|
))
|
||||||
|
spec.components.response('Unauthorized', dict(
|
||||||
|
description='User was not authorized'
|
||||||
|
))
|
||||||
|
|
||||||
|
return spec
|
42
src/ffxivbis/api/utils.py
Normal file
42
src/ffxivbis/api/utils.py
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import json
|
||||||
|
|
||||||
|
from aiohttp.web import HTTPException, Response
|
||||||
|
from typing import Any, Mapping, List
|
||||||
|
|
||||||
|
from .json import HttpEncoder
|
||||||
|
|
||||||
|
|
||||||
|
def make_json(response: Any) -> str:
|
||||||
|
return json.dumps(response, cls=HttpEncoder, sort_keys=True)
|
||||||
|
|
||||||
|
|
||||||
|
def wrap_exception(exception: Exception, args: Mapping[str, Any], code: int = 500) -> Response:
|
||||||
|
if isinstance(exception, HTTPException):
|
||||||
|
raise exception # reraise return
|
||||||
|
return wrap_json({
|
||||||
|
'message': repr(exception),
|
||||||
|
'arguments': dict(args)
|
||||||
|
}, code)
|
||||||
|
|
||||||
|
|
||||||
|
def wrap_invalid_param(params: List[str], args: Mapping[str, Any], code: int = 400) -> Response:
|
||||||
|
return wrap_json({
|
||||||
|
'message': f'invalid or missing parameters: `{params}`',
|
||||||
|
'arguments': dict(args)
|
||||||
|
}, code)
|
||||||
|
|
||||||
|
|
||||||
|
def wrap_json(response: Any, code: int = 200) -> Response:
|
||||||
|
return Response(
|
||||||
|
text=make_json(response),
|
||||||
|
status=code,
|
||||||
|
content_type='application/json'
|
||||||
|
)
|
0
src/ffxivbis/api/views/__init__.py
Normal file
0
src/ffxivbis/api/views/__init__.py
Normal file
0
src/ffxivbis/api/views/api/__init__.py
Normal file
0
src/ffxivbis/api/views/api/__init__.py
Normal file
159
src/ffxivbis/api/views/api/bis.py
Normal file
159
src/ffxivbis/api/views/api/bis.py
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import PlayerId
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param, wrap_json
|
||||||
|
from ffxivbis.api.views.common.bis_base import BiSBaseView
|
||||||
|
|
||||||
|
from .openapi import OpenApi
|
||||||
|
|
||||||
|
|
||||||
|
class BiSView(BiSBaseView, OpenApi):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Get party players BiS items'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
'name': 'nick',
|
||||||
|
'in': 'query',
|
||||||
|
'description': 'player nick name to filter',
|
||||||
|
'required': False,
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': { 'schema': {
|
||||||
|
'type': 'array',
|
||||||
|
'items': {
|
||||||
|
'allOf': [{'$ref': cls.model_ref('Piece')}]
|
||||||
|
}}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'get party BiS items'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['BiS']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Add new item to player BiS or remove existing'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['Piece', 'PlayerEdit']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {'$ref': cls.model_ref('Loot')}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'edit BiS'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['BiS']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_consumes(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['application/json']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Generate new BiS set'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['BiSLink']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {'$ref': cls.model_ref('BiS')}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'update BiS'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['BiS']
|
||||||
|
|
||||||
|
async def get(self) -> Response:
|
||||||
|
try:
|
||||||
|
loot = self.bis_get(self.request.query.getone('nick', None))
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get bis')
|
||||||
|
return wrap_exception(e, self.request.query)
|
||||||
|
|
||||||
|
return wrap_json(loot)
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['action', 'is_tome', 'job', 'name', 'nick']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
action = data.get('action')
|
||||||
|
if action not in ('add', 'remove'):
|
||||||
|
return wrap_invalid_param(['action'], data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
player_id = PlayerId(Job[data['job']], data['nick'])
|
||||||
|
piece: Piece = Piece.get(data) # type: ignore
|
||||||
|
await self.bis_post(action, player_id, piece)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not add bis')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json({'piece': piece, 'player_id': player_id})
|
||||||
|
|
||||||
|
async def put(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['job', 'link', 'nick']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
player_id = PlayerId(Job[data['job']], data['nick'])
|
||||||
|
bis = await self.bis_put(player_id, data['link'])
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not parse bis')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json(bis)
|
140
src/ffxivbis/api/views/api/login.py
Normal file
140
src/ffxivbis/api/views/api/login.py
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param, wrap_json
|
||||||
|
from ffxivbis.api.views.common.login_base import LoginBaseView
|
||||||
|
|
||||||
|
from .openapi import OpenApi
|
||||||
|
|
||||||
|
|
||||||
|
class LoginView(LoginBaseView, OpenApi):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Delete registered user'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
'name': 'username',
|
||||||
|
'in': 'path',
|
||||||
|
'description': 'username to remove',
|
||||||
|
'required': True,
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'type': 'object'}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'delete user'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['users']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Login as user'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['User']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'type': 'object'}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'login'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['users']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Create new user'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['User']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'type': 'object'}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'create user'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['users']
|
||||||
|
|
||||||
|
async def delete(self) -> Response:
|
||||||
|
username = self.request.match_info['username']
|
||||||
|
|
||||||
|
try:
|
||||||
|
await self.remove_user(username)
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('cannot remove user')
|
||||||
|
return wrap_exception(e, {'username': username})
|
||||||
|
|
||||||
|
return wrap_json({})
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['username', 'password']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
await self.login(data['username'], data['password'])
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('cannot login user')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json({})
|
||||||
|
|
||||||
|
async def put(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['username', 'password', 'party_id']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
await self.create_user(data['party_id'], data['username'],
|
||||||
|
data['password'], data.get('permission', 'get'))
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('cannot create user')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json({})
|
46
src/ffxivbis/api/views/api/logout.py
Normal file
46
src/ffxivbis/api/views/api/logout.py
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_json
|
||||||
|
from ffxivbis.api.views.common.login_base import LoginBaseView
|
||||||
|
|
||||||
|
from .openapi import OpenApi
|
||||||
|
|
||||||
|
|
||||||
|
class LogoutView(LoginBaseView, OpenApi):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Logout'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'type': 'object'}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'logout'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['users']
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
try:
|
||||||
|
await self.logout()
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('cannot logout user')
|
||||||
|
return wrap_exception(e, {})
|
||||||
|
|
||||||
|
return wrap_json({})
|
159
src/ffxivbis/api/views/api/loot.py
Normal file
159
src/ffxivbis/api/views/api/loot.py
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import PlayerId
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param, wrap_json
|
||||||
|
from ffxivbis.api.views.common.loot_base import LootBaseView
|
||||||
|
|
||||||
|
from .openapi import OpenApi
|
||||||
|
|
||||||
|
|
||||||
|
class LootView(LootBaseView, OpenApi):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Get party players loot'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
'name': 'nick',
|
||||||
|
'in': 'query',
|
||||||
|
'description': 'player nick name to filter',
|
||||||
|
'required': False,
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {
|
||||||
|
'type': 'array',
|
||||||
|
'items': {
|
||||||
|
'allOf': [{'$ref': cls.model_ref('Piece')}]
|
||||||
|
}}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'get party loot'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['loot']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Add new loot item or remove existing'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['Piece', 'PlayerEdit']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {'$ref': cls.model_ref('Loot')}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'edit loot'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['loot']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Suggest loot to party member'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['Piece']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {
|
||||||
|
'type': 'array',
|
||||||
|
'items': {
|
||||||
|
'allOf': [{'$ref': cls.model_ref('PlayerIdWithCounters')}]
|
||||||
|
}}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'suggest loot'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['loot']
|
||||||
|
|
||||||
|
async def get(self) -> Response:
|
||||||
|
try:
|
||||||
|
loot = self.loot_get(self.request.query.getone('nick', None))
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get loot')
|
||||||
|
return wrap_exception(e, self.request.query)
|
||||||
|
|
||||||
|
return wrap_json(loot)
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['action', 'is_tome', 'job', 'name', 'nick']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
action = data.get('action')
|
||||||
|
if action not in ('add', 'remove'):
|
||||||
|
return wrap_invalid_param(['action'], data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
player_id = PlayerId(Job[data['job']], data['nick'])
|
||||||
|
piece: Piece = Piece.get(data) # type: ignore
|
||||||
|
await self.loot_post(action, player_id, piece)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not add loot')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json({'piece': piece, 'player_id': player_id})
|
||||||
|
|
||||||
|
async def put(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['is_tome', 'name']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
piece: Piece = Piece.get(data) # type: ignore
|
||||||
|
players = self.loot_put(piece)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not suggest loot')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json(players)
|
195
src/ffxivbis/api/views/api/openapi.py
Normal file
195
src/ffxivbis/api/views/api/openapi.py
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.models.serializable import Serializable
|
||||||
|
|
||||||
|
|
||||||
|
class OpenApi(Serializable):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_delete_spec(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
description = cls.endpoint_delete_description()
|
||||||
|
if description is None:
|
||||||
|
return {}
|
||||||
|
return {
|
||||||
|
'description': description,
|
||||||
|
'parameters': cls.endpoint_delete_parameters(),
|
||||||
|
'responses': cls.endpoint_with_default_responses(cls.endpoint_delete_responses()),
|
||||||
|
'summary': cls.endpoint_delete_summary(),
|
||||||
|
'tags': cls.endpoint_delete_tags()
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_spec(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
description = cls.endpoint_get_description()
|
||||||
|
if description is None:
|
||||||
|
return {}
|
||||||
|
return {
|
||||||
|
'description': description,
|
||||||
|
'parameters': cls.endpoint_get_parameters(),
|
||||||
|
'responses': cls.endpoint_with_default_responses(cls.endpoint_get_responses()),
|
||||||
|
'summary': cls.endpoint_get_summary(),
|
||||||
|
'tags': cls.endpoint_get_tags()
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_consumes(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['application/json']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_spec(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
description = cls.endpoint_post_description()
|
||||||
|
if description is None:
|
||||||
|
return {}
|
||||||
|
return {
|
||||||
|
'consumes': cls.endpoint_post_consumes(),
|
||||||
|
'description': description,
|
||||||
|
'requestBody': {
|
||||||
|
'content': {
|
||||||
|
content_type: {
|
||||||
|
'schema': {'allOf': [
|
||||||
|
{'$ref': cls.model_ref(ref)}
|
||||||
|
for ref in cls.endpoint_post_request_body(content_type)
|
||||||
|
]}
|
||||||
|
}
|
||||||
|
for content_type in cls.endpoint_post_consumes()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'responses': cls.endpoint_with_default_responses(cls.endpoint_post_responses()),
|
||||||
|
'summary': cls.endpoint_post_summary(),
|
||||||
|
'tags': cls.endpoint_post_tags()
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_consumes(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['application/json']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_put_spec(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
description = cls.endpoint_put_description()
|
||||||
|
if description is None:
|
||||||
|
return {}
|
||||||
|
return {
|
||||||
|
'consumes': cls.endpoint_put_consumes(),
|
||||||
|
'description': description,
|
||||||
|
'requestBody': {
|
||||||
|
'content': {
|
||||||
|
content_type: {
|
||||||
|
'schema': {'allOf': [
|
||||||
|
{'$ref': cls.model_ref(ref)}
|
||||||
|
for ref in cls.endpoint_put_request_body(content_type)
|
||||||
|
]}
|
||||||
|
}
|
||||||
|
for content_type in cls.endpoint_put_consumes()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'responses': cls.endpoint_with_default_responses(cls.endpoint_put_responses()),
|
||||||
|
'summary': cls.endpoint_put_summary(),
|
||||||
|
'tags': cls.endpoint_put_tags()
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_spec(cls: Type[OpenApi], operations: List[str]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
operation.lower(): getattr(cls, f'endpoint_{operation.lower()}_spec')
|
||||||
|
for operation in operations
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_with_default_responses(cls: Type[OpenApi], responses: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
|
responses.update({
|
||||||
|
'400': {'$ref': cls.model_ref('BadRequest', 'responses')},
|
||||||
|
'401': {'$ref': cls.model_ref('Unauthorized', 'responses')},
|
||||||
|
'403': {'$ref': cls.model_ref('Forbidden', 'responses')},
|
||||||
|
'500': {'$ref': cls.model_ref('ServerError', 'responses')}
|
||||||
|
})
|
||||||
|
return responses
|
107
src/ffxivbis/api/views/api/player.py
Normal file
107
src/ffxivbis/api/views/api/player.py
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from typing import Any, Dict, List, Optional, Type
|
||||||
|
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param, wrap_json
|
||||||
|
from ffxivbis.api.views.common.player_base import PlayerBaseView
|
||||||
|
|
||||||
|
from .openapi import OpenApi
|
||||||
|
|
||||||
|
|
||||||
|
class PlayerView(PlayerBaseView, OpenApi):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Get party players with optional nick filter'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_parameters(cls: Type[OpenApi]) -> List[Dict[str, Any]]:
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
'name': 'nick',
|
||||||
|
'in': 'query',
|
||||||
|
'description': 'player nick name to filter',
|
||||||
|
'required': False,
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {'$ref': cls.model_ref('Player')}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'get party players'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_get_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['party']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_description(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'Create new party player or remove existing'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_request_body(cls: Type[OpenApi], content_type: str) -> List[str]:
|
||||||
|
return ['PlayerEdit']
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_responses(cls: Type[OpenApi]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'200': {'content': {'application/json': {'schema': {'$ref': cls.model_ref('PlayerId')}}}}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_summary(cls: Type[OpenApi]) -> Optional[str]:
|
||||||
|
return 'add or remove player'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def endpoint_post_tags(cls: Type[OpenApi]) -> List[str]:
|
||||||
|
return ['party']
|
||||||
|
|
||||||
|
async def get(self) -> Response:
|
||||||
|
try:
|
||||||
|
party = self.player_get(self.request.query.getone('nick', None))
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get party')
|
||||||
|
return wrap_exception(e, self.request.query)
|
||||||
|
|
||||||
|
return wrap_json(party)
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
try:
|
||||||
|
data = await self.request.json()
|
||||||
|
except Exception:
|
||||||
|
data = dict(await self.request.post())
|
||||||
|
|
||||||
|
required = ['action', 'job', 'nick']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
priority = data.get('priority', 0)
|
||||||
|
link = data.get('link', None)
|
||||||
|
|
||||||
|
action = data.get('action')
|
||||||
|
if action not in ('add', 'remove'):
|
||||||
|
return wrap_invalid_param(['action'], data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
player_id = await self.player_post(action, Job[data['job']], data['nick'], link, priority)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not add loot')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return wrap_json(player_id)
|
0
src/ffxivbis/api/views/common/__init__.py
Normal file
0
src/ffxivbis/api/views/common/__init__.py
Normal file
49
src/ffxivbis/api/views/common/bis_base.py
Normal file
49
src/ffxivbis/api/views/common/bis_base.py
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import View
|
||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from ffxivbis.core.ariyala_parser import AriyalaParser
|
||||||
|
from ffxivbis.models.bis import BiS
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import PlayerId
|
||||||
|
|
||||||
|
|
||||||
|
class BiSBaseView(View):
|
||||||
|
|
||||||
|
async def bis_add(self, player_id: PlayerId, piece: Piece) -> Piece:
|
||||||
|
await self.request.app['party'].set_item_bis(player_id, piece)
|
||||||
|
return piece
|
||||||
|
|
||||||
|
def bis_get(self, nick: Optional[str]) -> List[Piece]:
|
||||||
|
party = [
|
||||||
|
player
|
||||||
|
for player in self.request.app['party'].party
|
||||||
|
if nick is None or player.nick == nick
|
||||||
|
]
|
||||||
|
return list(sum([player.bis.pieces for player in party], []))
|
||||||
|
|
||||||
|
async def bis_post(self, action: str, player_id: PlayerId, piece: Piece) -> Optional[Piece]:
|
||||||
|
if action == 'add':
|
||||||
|
return await self.bis_add(player_id, piece)
|
||||||
|
elif action == 'remove':
|
||||||
|
return await self.bis_remove(player_id, piece)
|
||||||
|
return None
|
||||||
|
|
||||||
|
async def bis_put(self, player_id: PlayerId, link: str) -> BiS:
|
||||||
|
parser = AriyalaParser(self.request.app['config'])
|
||||||
|
items = await parser.get(link, player_id.job.name)
|
||||||
|
for piece in items:
|
||||||
|
await self.request.app['party'].set_item_bis(player_id, piece)
|
||||||
|
await self.request.app['party'].set_bis_link(player_id, link)
|
||||||
|
return self.request.app['party'].players[player_id].bis
|
||||||
|
|
||||||
|
async def bis_remove(self, player_id: PlayerId, piece: Piece) -> Piece:
|
||||||
|
await self.request.app['party'].remove_item_bis(player_id, piece)
|
||||||
|
return piece
|
43
src/ffxivbis/api/views/common/login_base.py
Normal file
43
src/ffxivbis/api/views/common/login_base.py
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import HTTPFound, HTTPUnauthorized, View
|
||||||
|
from aiohttp_security import check_authorized, forget, remember
|
||||||
|
from passlib.hash import md5_crypt
|
||||||
|
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
|
||||||
|
class LoginBaseView(View):
|
||||||
|
|
||||||
|
async def check_credentials(self, username: str, password: str) -> bool:
|
||||||
|
user = await self.request.app['database'].get_user(username)
|
||||||
|
if user is None:
|
||||||
|
return False
|
||||||
|
return md5_crypt.verify(password, user.password)
|
||||||
|
|
||||||
|
async def create_user(self, party_id: str, username: str, password: str, permission: str) -> None:
|
||||||
|
await self.request.app['database'].insert_user(party_id, User(username, password, permission), False)
|
||||||
|
|
||||||
|
async def login(self, username: str, password: str) -> None:
|
||||||
|
if await self.check_credentials(username, password):
|
||||||
|
response = HTTPFound('/')
|
||||||
|
await remember(self.request, response, username)
|
||||||
|
raise response
|
||||||
|
|
||||||
|
raise HTTPUnauthorized()
|
||||||
|
|
||||||
|
async def logout(self) -> None:
|
||||||
|
await check_authorized(self.request)
|
||||||
|
response = HTTPFound('/')
|
||||||
|
await forget(self.request, response)
|
||||||
|
|
||||||
|
raise response
|
||||||
|
|
||||||
|
async def remove_user(self, username: str) -> None:
|
||||||
|
await self.request.app['database'].delete_user(username)
|
43
src/ffxivbis/api/views/common/loot_base.py
Normal file
43
src/ffxivbis/api/views/common/loot_base.py
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import View
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import PlayerId, PlayerIdWithCounters
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
|
||||||
|
|
||||||
|
class LootBaseView(View):
|
||||||
|
|
||||||
|
async def loot_add(self, player_id: PlayerId, piece: Piece) -> Piece:
|
||||||
|
await self.request.app['party'].set_item(player_id, piece)
|
||||||
|
return piece
|
||||||
|
|
||||||
|
def loot_get(self, nick: Optional[str]) -> List[Piece]:
|
||||||
|
party = [
|
||||||
|
player
|
||||||
|
for player in self.request.app['party'].party
|
||||||
|
if nick is None or player.nick == nick
|
||||||
|
]
|
||||||
|
return list(sum([player.loot for player in party], []))
|
||||||
|
|
||||||
|
async def loot_post(self, action: str, player_id: PlayerId, piece: Piece) -> Optional[Piece]:
|
||||||
|
if action == 'add':
|
||||||
|
return await self.loot_add(player_id, piece)
|
||||||
|
elif action == 'remove':
|
||||||
|
return await self.loot_remove(player_id, piece)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def loot_put(self, piece: Union[Piece, Upgrade]) -> List[PlayerIdWithCounters]:
|
||||||
|
return self.request.app['loot'].suggest(piece)
|
||||||
|
|
||||||
|
async def loot_remove(self, player_id: PlayerId, piece: Piece) -> Piece:
|
||||||
|
await self.request.app['party'].remove_item(player_id, piece)
|
||||||
|
return piece
|
50
src/ffxivbis/api/views/common/player_base.py
Normal file
50
src/ffxivbis/api/views/common/player_base.py
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import View
|
||||||
|
from typing import List, Optional
|
||||||
|
|
||||||
|
from ffxivbis.core.ariyala_parser import AriyalaParser
|
||||||
|
from ffxivbis.models.bis import BiS
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
|
||||||
|
|
||||||
|
class PlayerBaseView(View):
|
||||||
|
|
||||||
|
async def player_add(self, job: Job, nick: str, link: Optional[str], priority: int) -> PlayerId:
|
||||||
|
player = Player(job, nick, BiS(), [], link, int(priority))
|
||||||
|
player_id = player.player_id
|
||||||
|
await self.request.app['party'].set_player(player)
|
||||||
|
|
||||||
|
if link:
|
||||||
|
parser = AriyalaParser(self.request.app['config'])
|
||||||
|
items = await parser.get(link, job.name)
|
||||||
|
for piece in items:
|
||||||
|
await self.request.app['party'].set_item_bis(player_id, piece)
|
||||||
|
|
||||||
|
return player_id
|
||||||
|
|
||||||
|
def player_get(self, nick: Optional[str]) -> List[Player]:
|
||||||
|
return [
|
||||||
|
player
|
||||||
|
for player in self.request.app['party'].party
|
||||||
|
if nick is None or player.nick == nick
|
||||||
|
]
|
||||||
|
|
||||||
|
async def player_post(self, action: str, job: Job, nick: str, link: Optional[str], priority: int) -> Optional[PlayerId]:
|
||||||
|
if action == 'add':
|
||||||
|
return await self.player_add(job, nick, link, priority)
|
||||||
|
elif action == 'remove':
|
||||||
|
return await self.player_remove(job, nick)
|
||||||
|
return None
|
||||||
|
|
||||||
|
async def player_remove(self, job: Job, nick: str) -> PlayerId:
|
||||||
|
player_id = PlayerId(job, nick)
|
||||||
|
await self.request.app['party'].remove_player(player_id)
|
||||||
|
return player_id
|
0
src/ffxivbis/api/views/html/__init__.py
Normal file
0
src/ffxivbis/api/views/html/__init__.py
Normal file
29
src/ffxivbis/api/views/html/api.py
Normal file
29
src/ffxivbis/api/views/html/api.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import json
|
||||||
|
|
||||||
|
from aiohttp.web import Response, View
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
|
||||||
|
class ApiDocVIew(View):
|
||||||
|
|
||||||
|
async def get(self) -> Response:
|
||||||
|
return Response(
|
||||||
|
text=json.dumps(self.request.app['spec'].to_dict()),
|
||||||
|
status=200,
|
||||||
|
content_type='application/json'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiHtmlView(View):
|
||||||
|
|
||||||
|
@template('api.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
return {}
|
82
src/ffxivbis/api/views/html/bis.py
Normal file
82
src/ffxivbis/api/views/html/bis.py
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import HTTPFound, Response
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict, List
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param
|
||||||
|
from ffxivbis.api.views.common.bis_base import BiSBaseView
|
||||||
|
from ffxivbis.api.views.common.player_base import PlayerBaseView
|
||||||
|
|
||||||
|
|
||||||
|
class BiSHtmlView(BiSBaseView, PlayerBaseView):
|
||||||
|
|
||||||
|
@template('bis.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
error = None
|
||||||
|
items: List[Dict[str, str]] = []
|
||||||
|
players: List[Player] = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
players = self.player_get(None)
|
||||||
|
items = [
|
||||||
|
{
|
||||||
|
'player': player.player_id.pretty_name,
|
||||||
|
'piece': piece.name,
|
||||||
|
'is_tome': 'yes' if piece.is_tome else 'no'
|
||||||
|
}
|
||||||
|
for player in players
|
||||||
|
for piece in player.bis.pieces
|
||||||
|
]
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get bis')
|
||||||
|
error = repr(e)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'items': items,
|
||||||
|
'pieces': Piece.available(),
|
||||||
|
'players': [player.player_id.pretty_name for player in players],
|
||||||
|
'request_error': error
|
||||||
|
}
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
data = await self.request.post()
|
||||||
|
|
||||||
|
required = ['method', 'player']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
method = data.getone('method')
|
||||||
|
player_id = PlayerId.from_pretty_name(data.getone('player')) # type: ignore
|
||||||
|
|
||||||
|
if method == 'post':
|
||||||
|
required = ['action', 'piece']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
is_tome = (data.getone('is_tome', None) == 'on')
|
||||||
|
await self.bis_post(data.getone('action'), player_id, # type: ignore
|
||||||
|
Piece.get({'piece': data.getone('piece'), 'is_tome': is_tome})) # type: ignore
|
||||||
|
|
||||||
|
elif method == 'put':
|
||||||
|
required = ['bis']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
await self.bis_put(player_id, data.getone('bis')) # type: ignore
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not manage bis')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return HTTPFound(self.request.url)
|
23
src/ffxivbis/api/views/html/index.py
Normal file
23
src/ffxivbis/api/views/html/index.py
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import View
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from aiohttp_security import authorized_userid
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
|
||||||
|
class IndexHtmlView(View):
|
||||||
|
|
||||||
|
@template('index.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
username = await authorized_userid(self.request)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'logged': username
|
||||||
|
}
|
70
src/ffxivbis/api/views/html/loot.py
Normal file
70
src/ffxivbis/api/views/html/loot.py
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import HTTPFound, Response
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict, List
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param
|
||||||
|
from ffxivbis.api.views.common.loot_base import LootBaseView
|
||||||
|
from ffxivbis.api.views.common.player_base import PlayerBaseView
|
||||||
|
|
||||||
|
|
||||||
|
class LootHtmlView(LootBaseView, PlayerBaseView):
|
||||||
|
|
||||||
|
@template('loot.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
error = None
|
||||||
|
items: List[Dict[str, str]] = []
|
||||||
|
players: List[Player] = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
players = self.player_get(None)
|
||||||
|
items = [
|
||||||
|
{
|
||||||
|
'player': player.player_id.pretty_name,
|
||||||
|
'piece': piece.name,
|
||||||
|
'is_tome': 'yes' if getattr(piece, 'is_tome', True) else 'no'
|
||||||
|
}
|
||||||
|
for player in players
|
||||||
|
for piece in player.loot
|
||||||
|
]
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get loot')
|
||||||
|
error = repr(e)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'items': items,
|
||||||
|
'pieces': Piece.available() + [upgrade.name for upgrade in Upgrade],
|
||||||
|
'players': [player.player_id.pretty_name for player in players],
|
||||||
|
'request_error': error
|
||||||
|
}
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
data = await self.request.post()
|
||||||
|
|
||||||
|
required = ['action', 'piece', 'player']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
player_id = PlayerId.from_pretty_name(data.getone('player')) # type: ignore
|
||||||
|
is_tome = (data.getone('is_tome', None) == 'on')
|
||||||
|
await self.loot_post(data.getone('action'), player_id, # type: ignore
|
||||||
|
Piece.get({'piece': data.getone('piece'), 'is_tome': is_tome})) # type: ignore
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not manage loot')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return HTTPFound(self.request.url)
|
64
src/ffxivbis/api/views/html/loot_suggest.py
Normal file
64
src/ffxivbis/api/views/html/loot_suggest.py
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import Response
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict, List, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import PlayerIdWithCounters
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_invalid_param
|
||||||
|
from ffxivbis.api.views.common.loot_base import LootBaseView
|
||||||
|
from ffxivbis.api.views.common.player_base import PlayerBaseView
|
||||||
|
|
||||||
|
|
||||||
|
class LootSuggestHtmlView(LootBaseView, PlayerBaseView):
|
||||||
|
|
||||||
|
@template('loot_suggest.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'pieces': Piece.available() + [upgrade.name for upgrade in Upgrade]
|
||||||
|
}
|
||||||
|
|
||||||
|
@template('loot_suggest.jinja2')
|
||||||
|
async def post(self) -> Union[Dict[str, Any], Response]:
|
||||||
|
data = await self.request.post()
|
||||||
|
error = None
|
||||||
|
item_values: Dict[str, Any] = {}
|
||||||
|
players: List[PlayerIdWithCounters] = []
|
||||||
|
|
||||||
|
required = ['piece']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
piece = Piece.get({'piece': data.getone('piece'), 'is_tome': data.getone('is_tome', False)})
|
||||||
|
players = self.loot_put(piece)
|
||||||
|
item_values = {'piece': piece.name, 'is_tome': getattr(piece, 'is_tome', True)}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not manage loot')
|
||||||
|
error = repr(e)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'item': item_values,
|
||||||
|
'pieces': Piece.available() + [upgrade.name for upgrade in Upgrade],
|
||||||
|
'request_error': error,
|
||||||
|
'suggest': [
|
||||||
|
{
|
||||||
|
'player': player.pretty_name,
|
||||||
|
'is_required': 'yes' if player.is_required else 'no',
|
||||||
|
'loot_count': player.loot_count,
|
||||||
|
'loot_count_bis': player.loot_count_bis,
|
||||||
|
'loot_count_total': player.loot_count_total
|
||||||
|
}
|
||||||
|
for player in players
|
||||||
|
]
|
||||||
|
}
|
67
src/ffxivbis/api/views/html/player.py
Normal file
67
src/ffxivbis/api/views/html/player.py
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import HTTPFound, Response
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict, List
|
||||||
|
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.player import PlayerIdWithCounters
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param
|
||||||
|
from ffxivbis.api.views.common.player_base import PlayerBaseView
|
||||||
|
|
||||||
|
|
||||||
|
class PlayerHtmlView(PlayerBaseView):
|
||||||
|
|
||||||
|
@template('party.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
counters: List[PlayerIdWithCounters] = []
|
||||||
|
error = None
|
||||||
|
|
||||||
|
try:
|
||||||
|
party = self.player_get(None)
|
||||||
|
counters = [player.player_id_with_counters(None) for player in party]
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get party')
|
||||||
|
error = repr(e)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'jobs': [job.name for job in Job],
|
||||||
|
'players': [
|
||||||
|
{
|
||||||
|
'job': player.job.name,
|
||||||
|
'nick': player.nick,
|
||||||
|
'loot_count_bis': player.loot_count_bis,
|
||||||
|
'loot_count_total': player.loot_count_total,
|
||||||
|
'priority': player.priority
|
||||||
|
}
|
||||||
|
for player in counters
|
||||||
|
],
|
||||||
|
'request_error': error
|
||||||
|
}
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
data = await self.request.post()
|
||||||
|
|
||||||
|
required = ['action', 'job', 'nick']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
action = data.getone('action')
|
||||||
|
priority = data.getone('priority', 0)
|
||||||
|
link = data.getone('bis', None)
|
||||||
|
await self.player_post(action, Job[data['job'].upper()], data['nick'], link, priority) # type: ignore
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not manage players')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return HTTPFound(self.request.url)
|
31
src/ffxivbis/api/views/html/static.py
Normal file
31
src/ffxivbis/api/views/html/static.py
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import os
|
||||||
|
|
||||||
|
from aiohttp.web import HTTPNotFound, Response, View
|
||||||
|
|
||||||
|
|
||||||
|
class StaticHtmlView(View):
|
||||||
|
|
||||||
|
def __get_content_type(self, filename: str) -> str:
|
||||||
|
_, ext = os.path.splitext(filename)
|
||||||
|
if ext == '.css':
|
||||||
|
return 'text/css'
|
||||||
|
elif ext == '.js':
|
||||||
|
return 'text/javascript'
|
||||||
|
return 'text/plain'
|
||||||
|
|
||||||
|
async def get(self) -> Response:
|
||||||
|
resource_name = self.request.match_info['resource_id']
|
||||||
|
resource_path = os.path.join(self.request.app['templates_root'], 'static', resource_name)
|
||||||
|
if not os.path.exists(resource_path) or os.path.isdir(resource_path):
|
||||||
|
return HTTPNotFound()
|
||||||
|
content_type = self.__get_content_type(resource_name)
|
||||||
|
|
||||||
|
with open(resource_path) as resource_file:
|
||||||
|
return Response(text=resource_file.read(), content_type=content_type)
|
62
src/ffxivbis/api/views/html/users.py
Normal file
62
src/ffxivbis/api/views/html/users.py
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from aiohttp.web import HTTPFound, Response
|
||||||
|
from aiohttp_jinja2 import template
|
||||||
|
from typing import Any, Dict, List
|
||||||
|
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
from ffxivbis.api.utils import wrap_exception, wrap_invalid_param
|
||||||
|
from ffxivbis.api.views.common.login_base import LoginBaseView
|
||||||
|
|
||||||
|
|
||||||
|
class UsersHtmlView(LoginBaseView):
|
||||||
|
|
||||||
|
@template('users.jinja2')
|
||||||
|
async def get(self) -> Dict[str, Any]:
|
||||||
|
error = None
|
||||||
|
users: List[User] = []
|
||||||
|
|
||||||
|
try:
|
||||||
|
users = await self.request.app['database'].get_users()
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not get users')
|
||||||
|
error = repr(e)
|
||||||
|
|
||||||
|
return {
|
||||||
|
'request_error': error,
|
||||||
|
'users': users
|
||||||
|
}
|
||||||
|
|
||||||
|
async def post(self) -> Response:
|
||||||
|
data = await self.request.post()
|
||||||
|
|
||||||
|
required = ['action', 'username']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
action = data.getone('action')
|
||||||
|
username = str(data.getone('username'))
|
||||||
|
|
||||||
|
if action == 'add':
|
||||||
|
required = ['password', 'permission']
|
||||||
|
if any(param not in data for param in required):
|
||||||
|
return wrap_invalid_param(required, data)
|
||||||
|
await self.create_user(username, data.getone('password'), data.getone('permission')) # type: ignore
|
||||||
|
elif action == 'remove':
|
||||||
|
await self.remove_user(username)
|
||||||
|
else:
|
||||||
|
return wrap_invalid_param(['action'], data)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.request.app.logger.exception('could not manage users')
|
||||||
|
return wrap_exception(e, data)
|
||||||
|
|
||||||
|
return HTTPFound(self.request.url)
|
67
src/ffxivbis/api/web.py
Normal file
67
src/ffxivbis/api/web.py
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import aiohttp_jinja2
|
||||||
|
import jinja2
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from aiohttp import web
|
||||||
|
from aiohttp_security import setup as setup_security
|
||||||
|
from aiohttp_security import CookiesIdentityPolicy
|
||||||
|
|
||||||
|
from ffxivbis.core.config import Configuration
|
||||||
|
from ffxivbis.core.database import Database
|
||||||
|
from ffxivbis.core.party_aggregator import PartyAggregator
|
||||||
|
|
||||||
|
from .auth import AuthorizationPolicy, authorize_factory
|
||||||
|
from .routes import setup_routes
|
||||||
|
from .spec import get_spec
|
||||||
|
|
||||||
|
|
||||||
|
async def on_shutdown(app: web.Application) -> None:
|
||||||
|
app.logger.warning('server terminated')
|
||||||
|
|
||||||
|
|
||||||
|
def run_server(app: web.Application) -> None:
|
||||||
|
app.logger.info('start server')
|
||||||
|
web.run_app(app,
|
||||||
|
host=app['config'].get('web', 'host'),
|
||||||
|
port=app['config'].getint('web', 'port'),
|
||||||
|
handle_signals=False)
|
||||||
|
|
||||||
|
def setup_service(config: Configuration, database: Database, aggregator: PartyAggregator) -> web.Application:
|
||||||
|
app = web.Application(logger=logging.getLogger('http'))
|
||||||
|
app.on_shutdown.append(on_shutdown)
|
||||||
|
|
||||||
|
app.middlewares.append(web.normalize_path_middleware(append_slash=False, remove_slash=True))
|
||||||
|
|
||||||
|
# auth related
|
||||||
|
auth_required = config.getboolean('auth', 'enabled')
|
||||||
|
if auth_required:
|
||||||
|
setup_security(app, CookiesIdentityPolicy(), AuthorizationPolicy(database))
|
||||||
|
app.middlewares.append(authorize_factory())
|
||||||
|
|
||||||
|
# routes
|
||||||
|
app.logger.info('setup routes')
|
||||||
|
setup_routes(app)
|
||||||
|
if config.has_option('web', 'templates'):
|
||||||
|
templates_root = app['templates_root'] = config.get('web', 'templates')
|
||||||
|
app['static_root_url'] = '/static'
|
||||||
|
aiohttp_jinja2.setup(app, loader=jinja2.FileSystemLoader(templates_root))
|
||||||
|
app['spec'] = get_spec(app)
|
||||||
|
|
||||||
|
app.logger.info('setup configuration')
|
||||||
|
app['config'] = config
|
||||||
|
|
||||||
|
app.logger.info('setup database')
|
||||||
|
app['database'] = database
|
||||||
|
|
||||||
|
app.logger.info('setup aggregator')
|
||||||
|
app['aggregator'] = aggregator
|
||||||
|
|
||||||
|
return app
|
0
src/ffxivbis/application/__init__.py
Normal file
0
src/ffxivbis/application/__init__.py
Normal file
31
src/ffxivbis/application/application.py
Normal file
31
src/ffxivbis/application/application.py
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from ffxivbis.core.config import Configuration
|
||||||
|
|
||||||
|
from .core import Application
|
||||||
|
|
||||||
|
|
||||||
|
def get_config(config_path: str) -> Configuration:
|
||||||
|
config = Configuration()
|
||||||
|
config.load(config_path, {})
|
||||||
|
config.load_logging()
|
||||||
|
|
||||||
|
return config
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(description='Simple loot recorder for FFXIV')
|
||||||
|
parser.add_argument('-c', '--config', help='configuration path', default='ffxivbis.ini')
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
config = get_config(args.config)
|
||||||
|
app = Application(config)
|
||||||
|
app.run()
|
33
src/ffxivbis/application/core.py
Normal file
33
src/ffxivbis/application/core.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import asyncio
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from ffxivbis.api.web import run_server, setup_service
|
||||||
|
from ffxivbis.core.config import Configuration
|
||||||
|
from ffxivbis.core.database import Database
|
||||||
|
from ffxivbis.core.party_aggregator import PartyAggregator
|
||||||
|
|
||||||
|
|
||||||
|
class Application:
|
||||||
|
|
||||||
|
def __init__(self, config: Configuration) -> None:
|
||||||
|
self.config = config
|
||||||
|
self.logger = logging.getLogger('application')
|
||||||
|
|
||||||
|
def run(self) -> None:
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
|
||||||
|
database = loop.run_until_complete(Database.get(self.config))
|
||||||
|
database.migration()
|
||||||
|
|
||||||
|
aggregator = PartyAggregator(self.config, database)
|
||||||
|
|
||||||
|
web = setup_service(self.config, database, aggregator)
|
||||||
|
run_server(web)
|
0
src/ffxivbis/core/__init__.py
Normal file
0
src/ffxivbis/core/__init__.py
Normal file
83
src/ffxivbis/core/ariyala_parser.py
Normal file
83
src/ffxivbis/core/ariyala_parser.py
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import os
|
||||||
|
import socket
|
||||||
|
|
||||||
|
from aiohttp import ClientSession
|
||||||
|
from typing import Dict, List, Optional
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
|
||||||
|
from .config import Configuration
|
||||||
|
|
||||||
|
|
||||||
|
class AriyalaParser:
|
||||||
|
|
||||||
|
def __init__(self, config: Configuration) -> None:
|
||||||
|
self.ariyala_url = config.get('ariyala', 'ariyala_url')
|
||||||
|
self.xivapi_key = config.get('ariyala', 'xivapi_key', fallback=None)
|
||||||
|
self.xivapi_url = config.get('ariyala', 'xivapi_url')
|
||||||
|
|
||||||
|
def __remap_key(self, key: str) -> Optional[str]:
|
||||||
|
if key == 'mainhand':
|
||||||
|
return 'weapon'
|
||||||
|
elif key == 'chest':
|
||||||
|
return 'body'
|
||||||
|
elif key == 'ringLeft':
|
||||||
|
return 'left_ring'
|
||||||
|
elif key == 'ringRight':
|
||||||
|
return 'right_ring'
|
||||||
|
elif key in ('head', 'hands', 'waist', 'legs', 'feet', 'ears', 'neck', 'wrist'):
|
||||||
|
return key
|
||||||
|
return None
|
||||||
|
|
||||||
|
async def get(self, url: str, job: str) -> List[Piece]:
|
||||||
|
items = await self.get_ids(url, job)
|
||||||
|
return [
|
||||||
|
Piece.get({'piece': slot, 'is_tome': await self.get_is_tome(item_id)}) # type: ignore
|
||||||
|
for slot, item_id in items.items()
|
||||||
|
]
|
||||||
|
|
||||||
|
async def get_ids(self, url: str, job: str) -> Dict[str, int]:
|
||||||
|
norm_path = os.path.normpath(url)
|
||||||
|
set_id = os.path.basename(norm_path)
|
||||||
|
async with ClientSession() as session:
|
||||||
|
async with session.get(f'{self.ariyala_url}/store.app', params={'identifier': set_id}) as response:
|
||||||
|
response.raise_for_status()
|
||||||
|
data = await response.json(content_type='text/html')
|
||||||
|
|
||||||
|
# it has job in response but for some reasons job name differs sometimes from one in dictionary,
|
||||||
|
# e.g. http://ffxiv.ariyala.com/store.app?identifier=1AJB8
|
||||||
|
api_job = data['content']
|
||||||
|
try:
|
||||||
|
bis = data['datasets'][api_job]['normal']['items']
|
||||||
|
except KeyError:
|
||||||
|
bis = data['datasets'][job]['normal']['items']
|
||||||
|
|
||||||
|
result: Dict[str, int] = {}
|
||||||
|
for original_key, value in bis.items():
|
||||||
|
key = self.__remap_key(original_key)
|
||||||
|
if key is None:
|
||||||
|
continue
|
||||||
|
result[key] = value
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def get_is_tome(self, item_id: int) -> bool:
|
||||||
|
params = {'columns': 'IsEquippable'}
|
||||||
|
if self.xivapi_key is not None:
|
||||||
|
params['private_key'] = self.xivapi_key
|
||||||
|
|
||||||
|
async with ClientSession() as session:
|
||||||
|
# for some reasons ipv6 does not work for me
|
||||||
|
session.connector._family = socket.AF_INET # type: ignore
|
||||||
|
async with session.get(f'{self.xivapi_url}/item/{item_id}', params=params) as response:
|
||||||
|
response.raise_for_status()
|
||||||
|
data = await response.json()
|
||||||
|
|
||||||
|
return data['IsEquippable'] == 0 # don't ask
|
65
src/ffxivbis/core/config.py
Normal file
65
src/ffxivbis/core/config.py
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import configparser
|
||||||
|
import os
|
||||||
|
|
||||||
|
from logging.config import fileConfig
|
||||||
|
from typing import Any, Dict, Mapping, Optional
|
||||||
|
|
||||||
|
from .exceptions import MissingConfiguration
|
||||||
|
|
||||||
|
|
||||||
|
class Configuration(configparser.RawConfigParser):
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
configparser.RawConfigParser.__init__(self, allow_no_value=True)
|
||||||
|
self.path: Optional[str] = None
|
||||||
|
self.root_path: Optional[str] = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def include(self) -> str:
|
||||||
|
return self.__with_root_path(self.get('settings', 'include'))
|
||||||
|
|
||||||
|
def __load_section(self, conf: str) -> None:
|
||||||
|
self.read(os.path.join(self.include, conf))
|
||||||
|
|
||||||
|
def __with_root_path(self, path: str) -> str:
|
||||||
|
if self.root_path is None:
|
||||||
|
return path
|
||||||
|
return os.path.join(self.root_path, path)
|
||||||
|
|
||||||
|
def get_section(self, section: str) -> Dict[str, str]:
|
||||||
|
if not self.has_section(section):
|
||||||
|
raise MissingConfiguration(section)
|
||||||
|
return dict(self[section])
|
||||||
|
|
||||||
|
def load(self, path: str, values: Mapping[str, Mapping[str, Any]]) -> None:
|
||||||
|
self.path = path
|
||||||
|
self.root_path = os.path.dirname(self.path)
|
||||||
|
|
||||||
|
self.read(self.path)
|
||||||
|
self.load_includes()
|
||||||
|
|
||||||
|
# don't use direct ConfigParser.update here, it overrides whole section
|
||||||
|
for section, options in values.items():
|
||||||
|
if section not in self:
|
||||||
|
self.add_section(section)
|
||||||
|
for key, value in options.items():
|
||||||
|
self.set(section, key, value)
|
||||||
|
|
||||||
|
def load_includes(self) -> None:
|
||||||
|
try:
|
||||||
|
include_dir = self.include
|
||||||
|
for conf in filter(lambda p: p.endswith('.ini'), sorted(os.listdir(include_dir))):
|
||||||
|
self.__load_section(conf)
|
||||||
|
except (FileNotFoundError, configparser.NoOptionError, configparser.NoSectionError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def load_logging(self) -> None:
|
||||||
|
fileConfig(self.__with_root_path(self.get('settings', 'logging')))
|
113
src/ffxivbis/core/database.py
Normal file
113
src/ffxivbis/core/database.py
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import datetime
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from yoyo import get_backend, read_migrations
|
||||||
|
from typing import List, Mapping, Optional, Type, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.loot import Loot
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
from .config import Configuration
|
||||||
|
from .exceptions import InvalidDatabase
|
||||||
|
|
||||||
|
|
||||||
|
class Database:
|
||||||
|
|
||||||
|
def __init__(self, migrations_path: str) -> None:
|
||||||
|
self.migrations_path = migrations_path
|
||||||
|
self.logger = logging.getLogger('database')
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def now() -> int:
|
||||||
|
return int(datetime.datetime.now().timestamp())
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
async def get(cls: Type[Database], config: Configuration) -> Database:
|
||||||
|
database_type = config.get('settings', 'database')
|
||||||
|
database_settings = config.get_section(database_type)
|
||||||
|
|
||||||
|
if database_type == 'sqlite':
|
||||||
|
from .sqlite import SQLiteDatabase
|
||||||
|
obj: Type[Database] = SQLiteDatabase
|
||||||
|
elif database_type == 'postgres':
|
||||||
|
from .postgres import PostgresDatabase
|
||||||
|
obj = PostgresDatabase
|
||||||
|
else:
|
||||||
|
raise InvalidDatabase(database_type)
|
||||||
|
|
||||||
|
database = obj(**database_settings)
|
||||||
|
await database.init()
|
||||||
|
return database
|
||||||
|
|
||||||
|
@property
|
||||||
|
def connection(self) -> str:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def init(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
async def delete_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def delete_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def delete_player(self, party_id: str, player_id: PlayerId) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def delete_user(self, party_id: str, username: str) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def get_party(self, party_id: str) -> List[Player]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def get_player(self, party_id: str, player_id: PlayerId) -> Optional[int]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def get_players(self, party_id: str) -> List[int]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def get_user(self, party_id: str, username: str) -> Optional[User]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def get_users(self, party_id: str) -> List[User]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def insert_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def insert_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def insert_player(self, party_id: str, player: Player) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
async def insert_user(self, party_id: str, user: User, hashed_password: bool) -> None:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def migration(self) -> None:
|
||||||
|
self.logger.info('perform migrations')
|
||||||
|
backend = get_backend(self.connection)
|
||||||
|
migrations = read_migrations(self.migrations_path)
|
||||||
|
with backend.lock():
|
||||||
|
backend.apply_migrations(backend.to_apply(migrations))
|
||||||
|
|
||||||
|
def set_loot(self, party: Mapping[int, Player], bis: List[Loot], loot: List[Loot]) -> List[Player]:
|
||||||
|
for piece in bis:
|
||||||
|
party[piece.player_id].bis.set_item(piece.piece)
|
||||||
|
for piece in loot:
|
||||||
|
party[piece.player_id].loot.append(piece.piece)
|
||||||
|
return list(party.values())
|
27
src/ffxivbis/core/exceptions.py
Normal file
27
src/ffxivbis/core/exceptions.py
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from typing import Any, Mapping
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidDatabase(Exception):
|
||||||
|
|
||||||
|
def __init__(self, database_type: str) -> None:
|
||||||
|
Exception.__init__(self, f'Unsupported database {database_type}')
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidDataRow(Exception):
|
||||||
|
|
||||||
|
def __init__(self, data: Mapping[str, Any]) -> None:
|
||||||
|
Exception.__init__(self, f'Invalid data row `{data}`')
|
||||||
|
|
||||||
|
|
||||||
|
class MissingConfiguration(Exception):
|
||||||
|
|
||||||
|
def __init__(self, section: str) -> None:
|
||||||
|
Exception.__init__(self, f'Missing configuration section {section}')
|
32
src/ffxivbis/core/loot_selector.py
Normal file
32
src/ffxivbis/core/loot_selector.py
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from typing import Iterable, List, Tuple, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.player import Player, PlayerIdWithCounters
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
|
||||||
|
from .party import Party
|
||||||
|
|
||||||
|
|
||||||
|
class LootSelector:
|
||||||
|
|
||||||
|
def __init__(self, party: Party, order_by: List[str] = None) -> None:
|
||||||
|
self.party = party
|
||||||
|
self.order_by = order_by or ['is_required', 'loot_count_bis', 'loot_count_total', 'loot_count', 'loot_priority']
|
||||||
|
|
||||||
|
def __order_by(self, player: Player, piece: Union[Piece, Upgrade]) -> Tuple:
|
||||||
|
return tuple(map(lambda method: getattr(player, method)(piece), self.order_by))
|
||||||
|
|
||||||
|
def __sorted_by(self, piece: Union[Piece, Upgrade]) -> Iterable[Player]:
|
||||||
|
# pycharm is lying, don't trust it
|
||||||
|
return sorted(self.party.players.values(), key=lambda player: self.__order_by(player, piece), reverse=True)
|
||||||
|
|
||||||
|
def suggest(self, piece: Union[Piece, Upgrade]) -> List[PlayerIdWithCounters]:
|
||||||
|
return [player.player_id_with_counters(piece) for player in self.__sorted_by(piece)]
|
82
src/ffxivbis/core/party.py
Normal file
82
src/ffxivbis/core/party.py
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from threading import Lock
|
||||||
|
from typing import Dict, List, Optional, Type, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
|
||||||
|
from .database import Database
|
||||||
|
|
||||||
|
|
||||||
|
class Party:
|
||||||
|
|
||||||
|
def __init__(self, party_id: str, database: Database) -> None:
|
||||||
|
self.lock = Lock()
|
||||||
|
self.party_id = party_id
|
||||||
|
self.players: Dict[PlayerId, Player] = {}
|
||||||
|
self.database = database
|
||||||
|
|
||||||
|
@property
|
||||||
|
def party(self) -> List[Player]:
|
||||||
|
with self.lock:
|
||||||
|
return list(self.players.values())
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
async def get(cls: Type[Party], party_id: str, database: Database) -> Party:
|
||||||
|
obj = cls(party_id, database)
|
||||||
|
players = await database.get_party(party_id)
|
||||||
|
for player in players:
|
||||||
|
obj.players[player.player_id] = player
|
||||||
|
return obj
|
||||||
|
|
||||||
|
async def set_bis_link(self, player_id: PlayerId, link: str) -> None:
|
||||||
|
with self.lock:
|
||||||
|
player = self.players[player_id]
|
||||||
|
player.link = link
|
||||||
|
await self.database.insert_player(self.party_id, player)
|
||||||
|
|
||||||
|
async def remove_player(self, player_id: PlayerId) -> Optional[Player]:
|
||||||
|
await self.database.delete_player(self.party_id, player_id)
|
||||||
|
with self.lock:
|
||||||
|
player = self.players.pop(player_id, None)
|
||||||
|
return player
|
||||||
|
|
||||||
|
async def set_player(self, player: Player) -> PlayerId:
|
||||||
|
player_id = player.player_id
|
||||||
|
await self.database.insert_player(self.party_id, player)
|
||||||
|
with self.lock:
|
||||||
|
self.players[player_id] = player
|
||||||
|
return player_id
|
||||||
|
|
||||||
|
async def set_item(self, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
await self.database.insert_piece(self.party_id, player_id, piece)
|
||||||
|
with self.lock:
|
||||||
|
self.players[player_id].loot.append(piece)
|
||||||
|
|
||||||
|
async def remove_item(self, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
await self.database.delete_piece(self.party_id, player_id, piece)
|
||||||
|
with self.lock:
|
||||||
|
try:
|
||||||
|
self.players[player_id].loot.remove(piece)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
async def set_item_bis(self, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
await self.database.insert_piece_bis(self.party_id, player_id, piece)
|
||||||
|
with self.lock:
|
||||||
|
self.players[player_id].bis.set_item(piece)
|
||||||
|
|
||||||
|
async def remove_item_bis(self, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
await self.database.delete_piece_bis(self.party_id, player_id, piece)
|
||||||
|
with self.lock:
|
||||||
|
self.players[player_id].bis.remove_item(piece)
|
26
src/ffxivbis/core/party_aggregator.py
Normal file
26
src/ffxivbis/core/party_aggregator.py
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from .config import Configuration
|
||||||
|
from .database import Database
|
||||||
|
from .loot_selector import LootSelector
|
||||||
|
from .party import Party
|
||||||
|
|
||||||
|
|
||||||
|
class PartyAggregator:
|
||||||
|
|
||||||
|
def __init__(self, config: Configuration, database: Database) -> None:
|
||||||
|
self.config = config
|
||||||
|
self.database = database
|
||||||
|
|
||||||
|
async def get_party(self, party_id: str) -> Party:
|
||||||
|
return await Party.get(party_id, self.database)
|
||||||
|
|
||||||
|
async def get_loot_selector(self, party: Party) -> LootSelector:
|
||||||
|
priority = self.config.get('settings', 'priority').split()
|
||||||
|
return LootSelector(party, priority)
|
175
src/ffxivbis/core/postgres.py
Normal file
175
src/ffxivbis/core/postgres.py
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import asyncpg
|
||||||
|
|
||||||
|
from passlib.hash import md5_crypt
|
||||||
|
from psycopg2.extras import DictCursor
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.bis import BiS
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.loot import Loot
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
from .database import Database
|
||||||
|
|
||||||
|
|
||||||
|
class PostgresDatabase(Database):
|
||||||
|
|
||||||
|
def __init__(self, host: str, port: int, username: str, password: str, database: str, migrations_path: str) -> None:
|
||||||
|
Database.__init__(self, migrations_path)
|
||||||
|
self.host = host
|
||||||
|
self.port = int(port)
|
||||||
|
self.username = username
|
||||||
|
self.password = password
|
||||||
|
self.database = database
|
||||||
|
self.pool: asyncpg.pool.Pool = None # type: ignore
|
||||||
|
|
||||||
|
@property
|
||||||
|
def connection(self) -> str:
|
||||||
|
return f'postgresql://{self.username}:{self.password}@{self.host}:{self.port}/{self.database}'
|
||||||
|
|
||||||
|
async def init(self) -> None:
|
||||||
|
self.pool = await asyncpg.create_pool(host=self.host, port=self.port, username=self.username,
|
||||||
|
password=self.password, database=self.database)
|
||||||
|
|
||||||
|
async def delete_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''delete from loot
|
||||||
|
where loot_id in (
|
||||||
|
select loot_id from loot
|
||||||
|
where player_id = $1 and piece = $2 and is_tome = $3 order by created desc limit 1
|
||||||
|
)''',
|
||||||
|
player, piece.name, getattr(piece, 'is_tome', True)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def delete_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''delete from bis where player_id = $1 and piece = $2''',
|
||||||
|
player, piece.name)
|
||||||
|
|
||||||
|
async def delete_player(self, party_id: str, player_id: PlayerId) -> None:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute('''delete from players where nick = $1 and job = $2 and party_id = $3''',
|
||||||
|
player_id.nick, player_id.job.name, party_id)
|
||||||
|
|
||||||
|
async def delete_user(self, party_id: str, username: str) -> None:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute('''delete from users where username = $1 and party_id = $2''',
|
||||||
|
(username, party_id))
|
||||||
|
|
||||||
|
async def get_party(self, party_id: str) -> List[Player]:
|
||||||
|
players = await self.get_players(party_id)
|
||||||
|
if not players:
|
||||||
|
return []
|
||||||
|
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
rows = await conn.fetch('''select * from bis where player_id in $1''', players)
|
||||||
|
bis_pieces = [Loot(row['player_id'], Piece.get(row)) for row in rows]
|
||||||
|
|
||||||
|
rows = await conn.fetch('''select * from loot where player_id in $1''', players)
|
||||||
|
loot_pieces = [Loot(row['player_id'], Piece.get(row)) for row in rows]
|
||||||
|
|
||||||
|
rows = await conn.fetch('''select * from players where party_id = $1''', party_id)
|
||||||
|
party = {
|
||||||
|
row['player_id']: Player(Job[row['job']], row['nick'], BiS(), [], row['bis_link'], row['priority'])
|
||||||
|
for row in rows
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.set_loot(party, bis_pieces, loot_pieces)
|
||||||
|
|
||||||
|
async def get_player(self, party_id: str, player_id: PlayerId) -> Optional[int]:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
player = await conn.fetchrow('''select player_id from players where nick = $1 and job = $2 and party_id = $3''',
|
||||||
|
player_id.nick, player_id.job.name, party_id)
|
||||||
|
return player['player_id'] if player is not None else None
|
||||||
|
|
||||||
|
async def get_players(self, party_id: str) -> List[int]:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
players = await conn.fetch('''select player_id from players where party_id = $1''', (party_id,))
|
||||||
|
return [player['player_id'] for player in players]
|
||||||
|
|
||||||
|
async def get_user(self, party_id: str, username: str) -> Optional[User]:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
user = await conn.fetchrow('''select * from users where username = $1 and party_id = $2''',
|
||||||
|
username, party_id)
|
||||||
|
return User(user['username'], user['password'], user['permission']) if user is not None else None
|
||||||
|
|
||||||
|
async def get_users(self, party_id: str) -> List[User]:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
users = await conn.fetch('''select * from users where party_id = $1''', party_id)
|
||||||
|
return [User(user['username'], user['password'], user['permission']) for user in users]
|
||||||
|
|
||||||
|
async def insert_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''insert into loot
|
||||||
|
(created, piece, is_tome, player_id)
|
||||||
|
values
|
||||||
|
($1, $2, $3, $4)''',
|
||||||
|
Database.now(), piece.name, getattr(piece, 'is_tome', True), player
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''insert into bis
|
||||||
|
(created, piece, is_tome, player_id)
|
||||||
|
values
|
||||||
|
($1, $2, $3, $4)
|
||||||
|
on conflict on constraint bis_piece_player_id_idx do update set
|
||||||
|
created = $1, is_tome = $3''',
|
||||||
|
Database.now(), piece.name, piece.is_tome, player
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_player(self, party_id: str, player: Player) -> None:
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''insert into players
|
||||||
|
(party_id, created, nick, job, bis_link, priority)
|
||||||
|
values
|
||||||
|
($1, $2, $3, $4, $5, $6)
|
||||||
|
on conflict on constraint players_nick_job_idx do update set
|
||||||
|
created = $1, bis_link = $4, priority = $5''',
|
||||||
|
Database.now(), player.nick, player.job.name, player.link, player.priority, party_id
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_user(self, party_id: str, user: User, hashed_password: bool) -> None:
|
||||||
|
password = user.password if hashed_password else md5_crypt.hash(user.password)
|
||||||
|
async with self.pool.acquire() as conn:
|
||||||
|
await conn.execute(
|
||||||
|
'''insert into users
|
||||||
|
(party_id, username, password, permission)
|
||||||
|
values
|
||||||
|
($1, $2, $3, $4)
|
||||||
|
on conflict on constraint users_username_idx do update set
|
||||||
|
password = $2, permission = $3''',
|
||||||
|
party_id, user.username, password, user.permission
|
||||||
|
)
|
165
src/ffxivbis/core/sqlite.py
Normal file
165
src/ffxivbis/core/sqlite.py
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from passlib.hash import md5_crypt
|
||||||
|
from typing import List, Optional, Union
|
||||||
|
|
||||||
|
from ffxivbis.models.bis import BiS
|
||||||
|
from ffxivbis.models.job import Job
|
||||||
|
from ffxivbis.models.loot import Loot
|
||||||
|
from ffxivbis.models.piece import Piece
|
||||||
|
from ffxivbis.models.player import Player, PlayerId
|
||||||
|
from ffxivbis.models.upgrade import Upgrade
|
||||||
|
from ffxivbis.models.user import User
|
||||||
|
|
||||||
|
from .database import Database
|
||||||
|
from .sqlite_helper import SQLiteHelper
|
||||||
|
|
||||||
|
|
||||||
|
class SQLiteDatabase(Database):
|
||||||
|
|
||||||
|
def __init__(self, database_path: str, migrations_path: str) -> None:
|
||||||
|
Database.__init__(self, migrations_path)
|
||||||
|
self.database_path = database_path
|
||||||
|
|
||||||
|
@property
|
||||||
|
def connection(self) -> str:
|
||||||
|
return f'sqlite:///{self.database_path}'
|
||||||
|
|
||||||
|
async def delete_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''delete from loot
|
||||||
|
where loot_id in (
|
||||||
|
select loot_id from loot
|
||||||
|
where player_id = ? and piece = ? and is_tome = ? order by created desc limit 1
|
||||||
|
)''',
|
||||||
|
(player, piece.name, getattr(piece, 'is_tome', True)))
|
||||||
|
|
||||||
|
async def delete_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''delete from bis where player_id = ? and piece = ?''',
|
||||||
|
(player, piece.name))
|
||||||
|
|
||||||
|
async def delete_player(self, party_id: str, player_id: PlayerId) -> None:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''delete from players where nick = ? and job = ? and party_id = ?''',
|
||||||
|
(player_id.nick, player_id.job.name, party_id))
|
||||||
|
|
||||||
|
async def delete_user(self, party_id: str, username: str) -> None:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''delete from users where username = ? and party_id = ?''',
|
||||||
|
(username, party_id))
|
||||||
|
|
||||||
|
async def get_party(self, party_id: str) -> List[Player]:
|
||||||
|
players = await self.get_players(party_id)
|
||||||
|
if not players:
|
||||||
|
return []
|
||||||
|
placeholder = ', '.join(['?'] * len(players))
|
||||||
|
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''select * from bis where player_id in ({})'''.format(placeholder), players)
|
||||||
|
rows = await cursor.fetchall()
|
||||||
|
bis_pieces = [Loot(row['player_id'], Piece.get(row)) for row in rows]
|
||||||
|
|
||||||
|
await cursor.execute('''select * from loot where player_id in ({})'''.format(placeholder), players)
|
||||||
|
rows = await cursor.fetchall()
|
||||||
|
loot_pieces = [Loot(row['player_id'], Piece.get(row)) for row in rows]
|
||||||
|
|
||||||
|
await cursor.execute('''select * from players where party_id = ?''', (party_id,))
|
||||||
|
rows = await cursor.fetchall()
|
||||||
|
party = {
|
||||||
|
row['player_id']: Player(Job[row['job']], row['nick'], BiS(), [], row['bis_link'], row['priority'])
|
||||||
|
for row in rows
|
||||||
|
}
|
||||||
|
|
||||||
|
return self.set_loot(party, bis_pieces, loot_pieces)
|
||||||
|
|
||||||
|
async def get_player(self, party_id: str, player_id: PlayerId) -> Optional[int]:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''select player_id from players where nick = ? and job = ? and party_id = ?''',
|
||||||
|
(player_id.nick, player_id.job.name, party_id))
|
||||||
|
player = await cursor.fetchone()
|
||||||
|
return player['player_id'] if player is not None else None
|
||||||
|
|
||||||
|
async def get_players(self, party_id: str) -> List[int]:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''select player_id from players where party_id = ?''', (party_id,))
|
||||||
|
players = await cursor.fetchall()
|
||||||
|
return [player['player_id'] for player in players]
|
||||||
|
|
||||||
|
async def get_user(self, party_id: str, username: str) -> Optional[User]:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''select * from users where username = ? and party_id = ?''',
|
||||||
|
(username, party_id))
|
||||||
|
user = await cursor.fetchone()
|
||||||
|
return User(user['username'], user['password'], user['permission']) if user is not None else None
|
||||||
|
|
||||||
|
async def get_users(self, party_id: str) -> List[User]:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute('''select * from users where party_id = ?''', (party_id,))
|
||||||
|
users = await cursor.fetchall()
|
||||||
|
return [User(user['username'], user['password'], user['permission']) for user in users]
|
||||||
|
|
||||||
|
async def insert_piece(self, party_id: str, player_id: PlayerId, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''insert into loot
|
||||||
|
(created, piece, is_tome, player_id)
|
||||||
|
values
|
||||||
|
(?, ?, ?, ?)''',
|
||||||
|
(Database.now(), piece.name, getattr(piece, 'is_tome', True), player)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_piece_bis(self, party_id: str, player_id: PlayerId, piece: Piece) -> None:
|
||||||
|
player = await self.get_player(party_id, player_id)
|
||||||
|
if player is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''replace into bis
|
||||||
|
(created, piece, is_tome, player_id)
|
||||||
|
values
|
||||||
|
(?, ?, ?, ?)''',
|
||||||
|
(Database.now(), piece.name, piece.is_tome, player)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_player(self, party_id: str, player: Player) -> None:
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''replace into players
|
||||||
|
(party_id, created, nick, job, bis_link, priority)
|
||||||
|
values
|
||||||
|
(?, ?, ?, ?, ?, ?)''',
|
||||||
|
(party_id, Database.now(), player.nick, player.job.name, player.link, player.priority)
|
||||||
|
)
|
||||||
|
|
||||||
|
async def insert_user(self, party_id: str, user: User, hashed_password: bool) -> None:
|
||||||
|
password = user.password if hashed_password else md5_crypt.hash(user.password)
|
||||||
|
async with SQLiteHelper(self.database_path) as cursor:
|
||||||
|
await cursor.execute(
|
||||||
|
'''replace into users
|
||||||
|
(party_id, username, password, permission)
|
||||||
|
values
|
||||||
|
(?, ?, ?, ?)''',
|
||||||
|
(party_id, user.username, password, user.permission)
|
||||||
|
)
|
36
src/ffxivbis/core/sqlite_helper.py
Normal file
36
src/ffxivbis/core/sqlite_helper.py
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
# because sqlite3 does not support context management
|
||||||
|
import aiosqlite
|
||||||
|
|
||||||
|
from types import TracebackType
|
||||||
|
from typing import Any, Dict, Optional, Type
|
||||||
|
|
||||||
|
|
||||||
|
def dict_factory(cursor: aiosqlite.Cursor, row: aiosqlite.Row) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
key: value
|
||||||
|
for key, value in zip([column[0] for column in cursor.description], row)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class SQLiteHelper():
|
||||||
|
def __init__(self, database_path: str) -> None:
|
||||||
|
self.database_path = database_path
|
||||||
|
|
||||||
|
async def __aenter__(self) -> aiosqlite.Cursor:
|
||||||
|
self.conn = await aiosqlite.connect(self.database_path)
|
||||||
|
self.conn.row_factory = dict_factory
|
||||||
|
await self.conn.execute('''pragma foreign_keys = on''')
|
||||||
|
return await self.conn.cursor()
|
||||||
|
|
||||||
|
async def __aexit__(self, exc_type: Optional[Type[BaseException]], exc_value: Optional[BaseException],
|
||||||
|
traceback: Optional[TracebackType]) -> None:
|
||||||
|
await self.conn.commit()
|
||||||
|
await self.conn.close()
|
9
src/ffxivbis/core/version.py
Normal file
9
src/ffxivbis/core/version.py
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
__version__ = '0.1.1'
|
0
src/ffxivbis/models/__init__.py
Normal file
0
src/ffxivbis/models/__init__.py
Normal file
16
src/ffxivbis/models/action.py
Normal file
16
src/ffxivbis/models/action.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from enum import auto
|
||||||
|
|
||||||
|
from .serializable import SerializableEnum
|
||||||
|
|
||||||
|
|
||||||
|
class Action(SerializableEnum):
|
||||||
|
add = auto()
|
||||||
|
remove = auto()
|
140
src/ffxivbis/models/bis.py
Normal file
140
src/ffxivbis/models/bis.py
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
import itertools
|
||||||
|
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Optional, Type, Union
|
||||||
|
|
||||||
|
from .job import Job
|
||||||
|
from .piece import Piece
|
||||||
|
from .serializable import Serializable
|
||||||
|
from .upgrade import Upgrade
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class BiSLink(Serializable):
|
||||||
|
nick: str
|
||||||
|
job: Job
|
||||||
|
link: str
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'job': {
|
||||||
|
'description': 'player job name',
|
||||||
|
'$ref': cls.model_ref('Job')
|
||||||
|
},
|
||||||
|
'link': {
|
||||||
|
'description': 'link to BiS set',
|
||||||
|
'example': 'https://ffxiv.ariyala.com/19V5R',
|
||||||
|
'type': 'string'
|
||||||
|
},
|
||||||
|
'nick': {
|
||||||
|
'description': 'player nick name',
|
||||||
|
'example': 'Siuan Sanche',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['job', 'link', 'nick']
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class BiS(Serializable):
|
||||||
|
weapon: Optional[Piece] = None
|
||||||
|
head: Optional[Piece] = None
|
||||||
|
body: Optional[Piece] = None
|
||||||
|
hands: Optional[Piece] = None
|
||||||
|
waist: Optional[Piece] = None
|
||||||
|
legs: Optional[Piece] = None
|
||||||
|
feet: Optional[Piece] = None
|
||||||
|
ears: Optional[Piece] = None
|
||||||
|
neck: Optional[Piece] = None
|
||||||
|
wrist: Optional[Piece] = None
|
||||||
|
left_ring: Optional[Piece] = None
|
||||||
|
right_ring: Optional[Piece] = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def pieces(self) -> List[Piece]:
|
||||||
|
return [piece for piece in self.__dict__.values() if isinstance(piece, Piece)]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def upgrades_required(self) -> Dict[Upgrade, int]:
|
||||||
|
return {
|
||||||
|
upgrade: len(list(pieces))
|
||||||
|
for upgrade, pieces in itertools.groupby(self.pieces, lambda piece: piece.upgrade)
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'weapon': {
|
||||||
|
'description': 'weapon part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'head': {
|
||||||
|
'description': 'head part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'body': {
|
||||||
|
'description': 'body part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'hands': {
|
||||||
|
'description': 'hands part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'waist': {
|
||||||
|
'description': 'waist part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'legs': {
|
||||||
|
'description': 'legs part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'feet': {
|
||||||
|
'description': 'feet part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'ears': {
|
||||||
|
'description': 'ears part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'neck': {
|
||||||
|
'description': 'neck part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'wrist': {
|
||||||
|
'description': 'wrist part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'left_ring': {
|
||||||
|
'description': 'left_ring part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'right_ring': {
|
||||||
|
'description': 'right_ring part of BiS',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def has_piece(self, piece: Union[Piece, Upgrade]) -> bool:
|
||||||
|
if isinstance(piece, Piece):
|
||||||
|
return piece in self.pieces
|
||||||
|
elif isinstance(piece, Upgrade):
|
||||||
|
return self.upgrades_required.get(piece) is not None
|
||||||
|
return False
|
||||||
|
|
||||||
|
def set_item(self, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
setattr(self, piece.name, piece)
|
||||||
|
|
||||||
|
def remove_item(self, piece: Union[Piece, Upgrade]) -> None:
|
||||||
|
setattr(self, piece.name, None)
|
36
src/ffxivbis/models/error.py
Normal file
36
src/ffxivbis/models/error.py
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Type
|
||||||
|
|
||||||
|
from .serializable import Serializable
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Error(Serializable):
|
||||||
|
message: str
|
||||||
|
arguments: Dict[str, Any]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'arguments': {
|
||||||
|
'description': 'arguments passed to request',
|
||||||
|
'type': 'object',
|
||||||
|
'additionalProperties': True
|
||||||
|
},
|
||||||
|
'message': {
|
||||||
|
'description': 'error message',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['arguments', 'message']
|
87
src/ffxivbis/models/job.py
Normal file
87
src/ffxivbis/models/job.py
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from enum import auto
|
||||||
|
from typing import Tuple
|
||||||
|
|
||||||
|
from .piece import Piece, PieceAccessory, Weapon
|
||||||
|
from .serializable import SerializableEnum
|
||||||
|
|
||||||
|
|
||||||
|
class Job(SerializableEnum):
|
||||||
|
PLD = auto()
|
||||||
|
WAR = auto()
|
||||||
|
DRK = auto()
|
||||||
|
GNB = auto()
|
||||||
|
WHM = auto()
|
||||||
|
SCH = auto()
|
||||||
|
AST = auto()
|
||||||
|
MNK = auto()
|
||||||
|
DRG = auto()
|
||||||
|
NIN = auto()
|
||||||
|
SAM = auto()
|
||||||
|
BRD = auto()
|
||||||
|
MCH = auto()
|
||||||
|
DNC = auto()
|
||||||
|
BLM = auto()
|
||||||
|
SMN = auto()
|
||||||
|
RDM = auto()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_accs_dex() -> Tuple:
|
||||||
|
return Job.group_ranges() + (Job.NIN,)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_accs_str() -> Tuple:
|
||||||
|
return Job.group_mnk() + (Job.DRG,)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_casters() -> Tuple:
|
||||||
|
return (Job.BLM, Job.SMN, Job.RDM)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_healers() -> Tuple:
|
||||||
|
return (Job.WHM, Job.SCH, Job.AST)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_mnk() -> Tuple:
|
||||||
|
return (Job.MNK, Job.SAM)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_ranges() -> Tuple:
|
||||||
|
return (Job.BRD, Job.MCH, Job.DNC)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def group_tanks() -> Tuple:
|
||||||
|
return (Job.PLD, Job.WAR, Job.DRK, Job.GNB)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def has_same_loot(left: Job, right: Job, piece: Piece) -> bool:
|
||||||
|
# same jobs, alright
|
||||||
|
if left == right:
|
||||||
|
return True
|
||||||
|
|
||||||
|
# weapons are unique per class always
|
||||||
|
if isinstance(piece, Weapon):
|
||||||
|
return False
|
||||||
|
|
||||||
|
# group comparison
|
||||||
|
for group in (Job.group_casters(), Job.group_healers(), Job.group_mnk(), Job.group_ranges(), Job.group_tanks()):
|
||||||
|
if left in group and right in group:
|
||||||
|
return True
|
||||||
|
|
||||||
|
# accessories group comparison
|
||||||
|
if isinstance(Piece, PieceAccessory):
|
||||||
|
for group in (Job.group_accs_dex(), Job.group_accs_str()):
|
||||||
|
if left in group and right in group:
|
||||||
|
return True
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
37
src/ffxivbis/models/loot.py
Normal file
37
src/ffxivbis/models/loot.py
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Type, Union
|
||||||
|
|
||||||
|
from .piece import Piece
|
||||||
|
from .serializable import Serializable
|
||||||
|
from .upgrade import Upgrade
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Loot(Serializable):
|
||||||
|
player_id: int
|
||||||
|
piece: Union[Piece, Upgrade]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'piece': {
|
||||||
|
'description': 'player piece',
|
||||||
|
'$ref': cls.model_ref('Piece')
|
||||||
|
},
|
||||||
|
'player_id': {
|
||||||
|
'description': 'player identifier',
|
||||||
|
'$ref': cls.model_ref('PlayerId')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['piece', 'player_id']
|
168
src/ffxivbis/models/piece.py
Normal file
168
src/ffxivbis/models/piece.py
Normal file
@ -0,0 +1,168 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Mapping, Type, Union
|
||||||
|
|
||||||
|
from ffxivbis.core.exceptions import InvalidDataRow
|
||||||
|
|
||||||
|
from .serializable import Serializable
|
||||||
|
from .upgrade import Upgrade
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Piece(Serializable):
|
||||||
|
is_tome: bool
|
||||||
|
name: str
|
||||||
|
|
||||||
|
@property
|
||||||
|
def upgrade(self) -> Upgrade:
|
||||||
|
if not self.is_tome:
|
||||||
|
return Upgrade.NoUpgrade
|
||||||
|
elif isinstance(self, Waist) or isinstance(self, PieceAccessory):
|
||||||
|
return Upgrade.AccessoryUpgrade
|
||||||
|
elif isinstance(self, Weapon):
|
||||||
|
return Upgrade.WeaponUpgrade
|
||||||
|
elif isinstance(self, PieceGear):
|
||||||
|
return Upgrade.GearUpgrade
|
||||||
|
return Upgrade.NoUpgrade
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def available() -> List[str]:
|
||||||
|
return [
|
||||||
|
'weapon',
|
||||||
|
'head', 'body', 'hands', 'waist', 'legs', 'feet',
|
||||||
|
'ears', 'neck', 'wrist', 'left_ring', 'right_ring'
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get(cls: Type[Piece], data: Mapping[str, Any]) -> Union[Piece, Upgrade]:
|
||||||
|
try:
|
||||||
|
piece_type = data.get('piece') or data.get('name')
|
||||||
|
if piece_type is None:
|
||||||
|
raise KeyError
|
||||||
|
is_tome = data['is_tome'] in ('yes', 'on', '1', 1, True)
|
||||||
|
except KeyError:
|
||||||
|
raise InvalidDataRow(data)
|
||||||
|
if piece_type.lower() == 'weapon':
|
||||||
|
return Weapon(is_tome)
|
||||||
|
elif piece_type.lower() == 'head':
|
||||||
|
return Head(is_tome)
|
||||||
|
elif piece_type.lower() == 'body':
|
||||||
|
return Body(is_tome)
|
||||||
|
elif piece_type.lower() == 'hands':
|
||||||
|
return Hands(is_tome)
|
||||||
|
elif piece_type.lower() == 'waist':
|
||||||
|
return Waist(is_tome)
|
||||||
|
elif piece_type.lower() == 'legs':
|
||||||
|
return Legs(is_tome)
|
||||||
|
elif piece_type.lower() == 'feet':
|
||||||
|
return Feet(is_tome)
|
||||||
|
elif piece_type.lower() == 'ears':
|
||||||
|
return Ears(is_tome)
|
||||||
|
elif piece_type.lower() == 'neck':
|
||||||
|
return Neck(is_tome)
|
||||||
|
elif piece_type.lower() == 'wrist':
|
||||||
|
return Wrist(is_tome)
|
||||||
|
elif piece_type.lower() in ('left_ring', 'right_ring', 'ring'):
|
||||||
|
return Ring(is_tome, piece_type.lower())
|
||||||
|
elif piece_type.lower() in Upgrade.dict_types():
|
||||||
|
return Upgrade[piece_type]
|
||||||
|
else:
|
||||||
|
raise InvalidDataRow(data)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'is_tome': {
|
||||||
|
'description': 'is this piece tome gear or not',
|
||||||
|
'type': 'boolean'
|
||||||
|
},
|
||||||
|
'name': {
|
||||||
|
'description': 'piece name',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['is_tome', 'name']
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class PieceAccessory(Piece):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class PieceGear(Piece):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Weapon(Piece):
|
||||||
|
name: str = 'weapon'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Head(PieceGear):
|
||||||
|
name: str = 'head'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Body(PieceGear):
|
||||||
|
name: str = 'body'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Hands(PieceGear):
|
||||||
|
name: str = 'hands'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Waist(PieceGear):
|
||||||
|
name: str = 'waist'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Legs(PieceGear):
|
||||||
|
name: str = 'legs'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Feet(PieceGear):
|
||||||
|
name: str = 'feet'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Ears(PieceAccessory):
|
||||||
|
name: str = 'ears'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Neck(PieceAccessory):
|
||||||
|
name: str = 'neck'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Wrist(PieceAccessory):
|
||||||
|
name: str = 'wrist'
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Ring(PieceAccessory):
|
||||||
|
name: str = 'ring'
|
||||||
|
|
||||||
|
# override __eq__method to be able to compare left/right rings
|
||||||
|
def __eq__(self, other: Any) -> bool:
|
||||||
|
if not isinstance(other, Ring):
|
||||||
|
return False
|
||||||
|
return self.is_tome == other.is_tome
|
201
src/ffxivbis/models/player.py
Normal file
201
src/ffxivbis/models/player.py
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import re
|
||||||
|
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Optional, Type, Union
|
||||||
|
|
||||||
|
from .bis import BiS
|
||||||
|
from .job import Job
|
||||||
|
from .piece import Piece
|
||||||
|
from .serializable import Serializable
|
||||||
|
from .upgrade import Upgrade
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class PlayerId(Serializable):
|
||||||
|
job: Job
|
||||||
|
nick: str
|
||||||
|
|
||||||
|
@property
|
||||||
|
def pretty_name(self) -> str:
|
||||||
|
return f'{self.nick} ({self.job.name})'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_pretty_name(cls: Type[PlayerId], value: str) -> Optional[PlayerId]:
|
||||||
|
matches = re.search('^(?P<nick>.*) \((?P<job>[A-Z]+)\)$', value)
|
||||||
|
if matches is None:
|
||||||
|
return None
|
||||||
|
return PlayerId(Job[matches.group('job')], matches.group('nick'))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'job': {
|
||||||
|
'description': 'player job name',
|
||||||
|
'$ref': cls.model_ref('Job')
|
||||||
|
},
|
||||||
|
'nick': {
|
||||||
|
'description': 'player nick name',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['job', 'nick']
|
||||||
|
|
||||||
|
def __hash__(self) -> int:
|
||||||
|
return hash(str(self))
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class PlayerIdWithCounters(PlayerId):
|
||||||
|
is_required: bool
|
||||||
|
priority: int
|
||||||
|
loot_count: int
|
||||||
|
loot_count_bis: int
|
||||||
|
loot_count_total: int
|
||||||
|
bis_count_total: int
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'bis_count_total': {
|
||||||
|
'description': 'total savage pieces in BiS',
|
||||||
|
'type': 'integer'
|
||||||
|
},
|
||||||
|
'is_required': {
|
||||||
|
'description': 'is item required by BiS or not',
|
||||||
|
'type': 'boolean'
|
||||||
|
},
|
||||||
|
'job': {
|
||||||
|
'description': 'player job name',
|
||||||
|
'$ref': cls.model_ref('Job')
|
||||||
|
},
|
||||||
|
'loot_count': {
|
||||||
|
'description': 'count of this item which was already looted',
|
||||||
|
'type': 'integer'
|
||||||
|
},
|
||||||
|
'loot_count_bis': {
|
||||||
|
'description': 'count of BiS items which were already looted',
|
||||||
|
'type': 'integer'
|
||||||
|
},
|
||||||
|
'loot_count_total': {
|
||||||
|
'description': 'total count of items which were looted',
|
||||||
|
'type': 'integer'
|
||||||
|
},
|
||||||
|
'nick': {
|
||||||
|
'description': 'player nick name',
|
||||||
|
'type': 'string'
|
||||||
|
},
|
||||||
|
'priority': {
|
||||||
|
'description': 'player loot priority',
|
||||||
|
'type': 'integer'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['bis_count_total', 'is_required', 'job', 'loot_count',
|
||||||
|
'loot_count_bis', 'loot_count_total', 'nick', 'priority']
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Player(Serializable):
|
||||||
|
job: Job
|
||||||
|
nick: str
|
||||||
|
bis: BiS
|
||||||
|
loot: List[Union[Piece, Upgrade]]
|
||||||
|
link: Optional[str] = None
|
||||||
|
priority: int = 0
|
||||||
|
|
||||||
|
@property
|
||||||
|
def player_id(self) -> PlayerId:
|
||||||
|
return PlayerId(self.job, self.nick)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'bis': {
|
||||||
|
'description': 'player BiS',
|
||||||
|
'$ref': cls.model_ref('BiS')
|
||||||
|
},
|
||||||
|
'job': {
|
||||||
|
'description': 'player job name',
|
||||||
|
'$ref': cls.model_ref('Job')
|
||||||
|
},
|
||||||
|
'link': {
|
||||||
|
'description': 'link to player BiS',
|
||||||
|
'type': 'string'
|
||||||
|
},
|
||||||
|
'loot': {
|
||||||
|
'description': 'player looted items',
|
||||||
|
'type': 'array',
|
||||||
|
'items': {
|
||||||
|
'anyOf': [
|
||||||
|
{'$ref': cls.model_ref('Piece')},
|
||||||
|
{'$ref': cls.model_ref('Upgrade')}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'nick': {
|
||||||
|
'description': 'player nick name',
|
||||||
|
'type': 'string'
|
||||||
|
},
|
||||||
|
'priority': {
|
||||||
|
'description': 'player loot priority',
|
||||||
|
'type': 'integer'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['bis', 'job', 'loot', 'nick', 'priority']
|
||||||
|
|
||||||
|
def player_id_with_counters(self, piece: Union[Piece, Upgrade, None]) -> PlayerIdWithCounters:
|
||||||
|
return PlayerIdWithCounters(self.job, self.nick, self.is_required(piece), self.priority,
|
||||||
|
abs(self.loot_count(piece)), abs(self.loot_count_bis(piece)),
|
||||||
|
abs(self.loot_count_total(piece)), abs(self.bis_count_total(piece)))
|
||||||
|
|
||||||
|
# ordering methods
|
||||||
|
def is_required(self, piece: Union[Piece, Upgrade, None]) -> bool:
|
||||||
|
if piece is None:
|
||||||
|
return False
|
||||||
|
|
||||||
|
# lets check if it is even in bis
|
||||||
|
if not self.bis.has_piece(piece):
|
||||||
|
return False
|
||||||
|
|
||||||
|
if isinstance(piece, Piece):
|
||||||
|
# alright it is in is, lets check if he even got it
|
||||||
|
return self.loot_count(piece) == 0
|
||||||
|
elif isinstance(piece, Upgrade):
|
||||||
|
# alright it lets check how much upgrades does they need
|
||||||
|
return self.bis.upgrades_required[piece] > self.loot_count(piece)
|
||||||
|
return False
|
||||||
|
|
||||||
|
def loot_count(self, piece: Union[Piece, Upgrade, None]) -> int:
|
||||||
|
if piece is None:
|
||||||
|
return -self.loot_count_total(piece)
|
||||||
|
return -self.loot.count(piece)
|
||||||
|
|
||||||
|
def loot_count_bis(self, _: Union[Piece, Upgrade, None]) -> int:
|
||||||
|
return -len([piece for piece in self.loot if self.bis.has_piece(piece)])
|
||||||
|
|
||||||
|
def loot_count_total(self, _: Union[Piece, Upgrade, None]) -> int:
|
||||||
|
return -len(self.loot)
|
||||||
|
|
||||||
|
def bis_count_total(self, _: Union[Piece, Upgrade, None]) -> int:
|
||||||
|
return len([piece for piece in self.bis.pieces if not piece.is_tome])
|
||||||
|
|
||||||
|
def loot_priority(self, _: Union[Piece, Upgrade, None]) -> int:
|
||||||
|
return self.priority
|
35
src/ffxivbis/models/player_edit.py
Normal file
35
src/ffxivbis/models/player_edit.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from typing import Any, Dict, List, Type
|
||||||
|
|
||||||
|
from .serializable import Serializable
|
||||||
|
|
||||||
|
|
||||||
|
class PlayerEdit(Serializable):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'action': {
|
||||||
|
'description': 'action to perform',
|
||||||
|
'$ref': cls.model_ref('Action')
|
||||||
|
},
|
||||||
|
'job': {
|
||||||
|
'description': 'player job name to edit',
|
||||||
|
'$ref': cls.model_ref('Job')
|
||||||
|
},
|
||||||
|
'nick': {
|
||||||
|
'description': 'player nick name to edit',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['action', 'nick', 'job']
|
57
src/ffxivbis/models/serializable.py
Normal file
57
src/ffxivbis/models/serializable.py
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from enum import Enum
|
||||||
|
from typing import Any, Dict, List, Type
|
||||||
|
|
||||||
|
|
||||||
|
class Serializable:
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_name(cls: Type[Serializable]) -> str:
|
||||||
|
return cls.__name__
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def model_ref(model_name: str, model_group: str = 'schemas') -> str:
|
||||||
|
return f'#/components/{model_group}/{model_name}'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_spec(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'type': cls.model_type(),
|
||||||
|
'properties': cls.model_properties(),
|
||||||
|
'required': cls.model_required()
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_type(cls: Type[Serializable]) -> str:
|
||||||
|
return 'object'
|
||||||
|
|
||||||
|
|
||||||
|
class SerializableEnum(Serializable, Enum):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_spec(cls: Type[SerializableEnum]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'type': cls.model_type(),
|
||||||
|
'enum': [item.name for item in cls]
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_type(cls: Type[Serializable]) -> str:
|
||||||
|
return 'string'
|
23
src/ffxivbis/models/upgrade.py
Normal file
23
src/ffxivbis/models/upgrade.py
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from enum import auto
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
from .serializable import SerializableEnum
|
||||||
|
|
||||||
|
|
||||||
|
class Upgrade(SerializableEnum):
|
||||||
|
NoUpgrade = auto()
|
||||||
|
AccessoryUpgrade = auto()
|
||||||
|
GearUpgrade = auto()
|
||||||
|
WeaponUpgrade = auto()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def dict_types() -> List[str]:
|
||||||
|
return list(map(lambda t: t.name.lower(), Upgrade))
|
42
src/ffxivbis/models/user.py
Normal file
42
src/ffxivbis/models/user.py
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2019 Evgeniy Alekseev.
|
||||||
|
#
|
||||||
|
# This file is part of ffxivbis
|
||||||
|
# (see https://github.com/arcan1s/ffxivbis).
|
||||||
|
#
|
||||||
|
# License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
||||||
|
#
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any, Dict, List, Type
|
||||||
|
|
||||||
|
from .serializable import Serializable
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class User(Serializable):
|
||||||
|
username: str
|
||||||
|
password: str
|
||||||
|
permission: str
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_properties(cls: Type[Serializable]) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
'password': {
|
||||||
|
'description': 'user password',
|
||||||
|
'type': 'string'
|
||||||
|
},
|
||||||
|
'permission': {
|
||||||
|
'default': 'get',
|
||||||
|
'description': 'user action permissions',
|
||||||
|
'type': 'string',
|
||||||
|
'enum': ['admin', 'get', 'post']
|
||||||
|
},
|
||||||
|
'username': {
|
||||||
|
'description': 'user name',
|
||||||
|
'type': 'string'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def model_required(cls: Type[Serializable]) -> List[str]:
|
||||||
|
return ['password', 'username']
|
@ -1,36 +0,0 @@
|
|||||||
create table players (
|
|
||||||
party_id text not null,
|
|
||||||
player_id bigserial unique,
|
|
||||||
created bigint not null,
|
|
||||||
nick text not null,
|
|
||||||
job text not null,
|
|
||||||
bis_link text,
|
|
||||||
priority integer not null default 1);
|
|
||||||
create unique index players_nick_job_idx on players(party_id, nick, job);
|
|
||||||
|
|
||||||
create table loot (
|
|
||||||
loot_id bigserial unique,
|
|
||||||
player_id bigint not null,
|
|
||||||
created bigint not null,
|
|
||||||
piece text not null,
|
|
||||||
is_tome integer not null,
|
|
||||||
job text not null,
|
|
||||||
foreign key (player_id) references players(player_id) on delete cascade);
|
|
||||||
create index loot_owner_idx on loot(player_id);
|
|
||||||
|
|
||||||
create table bis (
|
|
||||||
player_id bigint not null,
|
|
||||||
created bigint not null,
|
|
||||||
piece text not null,
|
|
||||||
is_tome integer not null,
|
|
||||||
job text not null,
|
|
||||||
foreign key (player_id) references players(player_id) on delete cascade);
|
|
||||||
create unique index bis_piece_player_id_idx on bis(player_id, piece);
|
|
||||||
|
|
||||||
create table users (
|
|
||||||
party_id text not null,
|
|
||||||
user_id bigserial unique,
|
|
||||||
username text not null,
|
|
||||||
password text not null,
|
|
||||||
permission text not null);
|
|
||||||
create unique index users_username_idx on users(party_id, username);
|
|
@ -1,36 +0,0 @@
|
|||||||
create table players (
|
|
||||||
party_id text not null,
|
|
||||||
player_id integer primary key autoincrement,
|
|
||||||
created integer not null,
|
|
||||||
nick text not null,
|
|
||||||
job text not null,
|
|
||||||
bis_link text,
|
|
||||||
priority integer not null default 1);
|
|
||||||
create unique index players_nick_job_idx on players(party_id, nick, job);
|
|
||||||
|
|
||||||
create table loot (
|
|
||||||
loot_id integer primary key autoincrement,
|
|
||||||
player_id integer not null,
|
|
||||||
created integer not null,
|
|
||||||
piece text not null,
|
|
||||||
is_tome integer not null,
|
|
||||||
job text not null,
|
|
||||||
foreign key (player_id) references players(player_id) on delete cascade);
|
|
||||||
create index loot_owner_idx on loot(player_id);
|
|
||||||
|
|
||||||
create table bis (
|
|
||||||
player_id integer not null,
|
|
||||||
created integer not null,
|
|
||||||
piece text not null,
|
|
||||||
is_tome integer not null,
|
|
||||||
job text not null,
|
|
||||||
foreign key (player_id) references players(player_id) on delete cascade);
|
|
||||||
create unique index bis_piece_player_id_idx on bis(player_id, piece);
|
|
||||||
|
|
||||||
create table users (
|
|
||||||
party_id text not null,
|
|
||||||
user_id integer primary key autoincrement,
|
|
||||||
username text not null,
|
|
||||||
password text not null,
|
|
||||||
permission text not null);
|
|
||||||
create unique index users_username_idx on users(party_id, username);
|
|
@ -1,27 +0,0 @@
|
|||||||
<included>
|
|
||||||
|
|
||||||
<appender name="application-base" class="ch.qos.logback.core.rolling.RollingFileAppender">
|
|
||||||
<encoder>
|
|
||||||
<pattern>[%-5level %d{yyyy-MM-dd HH:mm:ss.SSS}] [%thread] [%logger{50}]: %msg%n</pattern>
|
|
||||||
</encoder>
|
|
||||||
|
|
||||||
<file>application.log</file>
|
|
||||||
|
|
||||||
<rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy">
|
|
||||||
<minIndex>1</minIndex>
|
|
||||||
<maxIndex>20</maxIndex>
|
|
||||||
<fileNamePattern>application.log.%i.gz</fileNamePattern>
|
|
||||||
</rollingPolicy>
|
|
||||||
|
|
||||||
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
|
|
||||||
<maxFileSize>100MB</maxFileSize>
|
|
||||||
</triggeringPolicy>
|
|
||||||
</appender>
|
|
||||||
|
|
||||||
<appender name="application" class="ch.qos.logback.classic.AsyncAppender">
|
|
||||||
<appender-ref ref="application-base"/>
|
|
||||||
<queueSize>50000</queueSize>
|
|
||||||
<neverBlock>true</neverBlock>
|
|
||||||
</appender>
|
|
||||||
|
|
||||||
</included>
|
|
@ -1,27 +0,0 @@
|
|||||||
<included>
|
|
||||||
|
|
||||||
<appender name="http-base" class="ch.qos.logback.core.rolling.RollingFileAppender">
|
|
||||||
<encoder>
|
|
||||||
<pattern>%msg%n</pattern>
|
|
||||||
</encoder>
|
|
||||||
|
|
||||||
<file>http.log</file>
|
|
||||||
|
|
||||||
<rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy">
|
|
||||||
<minIndex>1</minIndex>
|
|
||||||
<maxIndex>20</maxIndex>
|
|
||||||
<fileNamePattern>http.log.%i.gz</fileNamePattern>
|
|
||||||
</rollingPolicy>
|
|
||||||
|
|
||||||
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
|
|
||||||
<maxFileSize>100MB</maxFileSize>
|
|
||||||
</triggeringPolicy>
|
|
||||||
</appender>
|
|
||||||
|
|
||||||
<appender name="http" class="ch.qos.logback.classic.AsyncAppender">
|
|
||||||
<appender-ref ref="http-base"/>
|
|
||||||
<queueSize>50000</queueSize>
|
|
||||||
<neverBlock>true</neverBlock>
|
|
||||||
</appender>
|
|
||||||
|
|
||||||
</included>
|
|
@ -1,16 +0,0 @@
|
|||||||
<configuration>
|
|
||||||
|
|
||||||
<include resource="logback-application.xml" />
|
|
||||||
<include resource="logback-http.xml" />
|
|
||||||
|
|
||||||
<root level="debug">
|
|
||||||
<appender-ref ref="application" />
|
|
||||||
</root>
|
|
||||||
|
|
||||||
<logger name="me.arcanis.ffxivbis" level="DEBUG" />
|
|
||||||
<logger name="http" level="DEBUG">
|
|
||||||
<appender-ref ref="http" />
|
|
||||||
</logger>
|
|
||||||
<logger name="slick" level="INFO" />
|
|
||||||
|
|
||||||
</configuration>
|
|
@ -1,55 +0,0 @@
|
|||||||
me.arcanis.ffxivbis {
|
|
||||||
ariyala {
|
|
||||||
# ariyala base url, string, required
|
|
||||||
ariyala-url = "https://ffxiv.ariyala.com"
|
|
||||||
# xivapi base url, string, required
|
|
||||||
xivapi-url = "https://xivapi.com"
|
|
||||||
# xivapi developer key, string, optional
|
|
||||||
# xivapi-key = "abcdef"
|
|
||||||
}
|
|
||||||
|
|
||||||
database {
|
|
||||||
# database section. Section must be declared inside
|
|
||||||
# for more detailed section descriptions refer to slick documentation
|
|
||||||
mode = "sqlite"
|
|
||||||
|
|
||||||
sqlite {
|
|
||||||
profile = "slick.jdbc.SQLiteProfile$"
|
|
||||||
db {
|
|
||||||
url = "jdbc:sqlite:ffxivbis.db"
|
|
||||||
user = "user"
|
|
||||||
password = "password"
|
|
||||||
}
|
|
||||||
numThreads = 10
|
|
||||||
}
|
|
||||||
|
|
||||||
postgresql {
|
|
||||||
profile = "slick.jdbc.PostgresProfile$"
|
|
||||||
db {
|
|
||||||
url = "jdbc:postgresql://localhost/ffxivbis"
|
|
||||||
user = "user"
|
|
||||||
password = "password"
|
|
||||||
}
|
|
||||||
numThreads = 10
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
settings {
|
|
||||||
# counters of Player class which will be called to sort players for loot priority
|
|
||||||
# list of strings, required
|
|
||||||
priority = [
|
|
||||||
"isRequired", "lootCountBiS", "priority", "lootCount", "lootCountTotal"
|
|
||||||
]
|
|
||||||
# general request timeout, duratin, required
|
|
||||||
request-timeout = 10s
|
|
||||||
# party in-memory storage lifetime
|
|
||||||
cache-timeout = 1m
|
|
||||||
}
|
|
||||||
|
|
||||||
web {
|
|
||||||
# address to bind, string, required
|
|
||||||
host = "0.0.0.0"
|
|
||||||
# port to bind, int, required
|
|
||||||
port = 8000
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,277 +0,0 @@
|
|||||||
/* in-text images */
|
|
||||||
figure.img {
|
|
||||||
float: right;
|
|
||||||
border: 0px solid #333;
|
|
||||||
padding: 0px;
|
|
||||||
margin: 5px 0px 5px 10px;
|
|
||||||
}
|
|
||||||
figure.img img {
|
|
||||||
max-width: 100%;
|
|
||||||
height: auto;
|
|
||||||
}
|
|
||||||
figure.img figcaption {
|
|
||||||
margin: 0px;
|
|
||||||
font-size: 90%;
|
|
||||||
font-style: italic;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1 .octicon-link, h2 .octicon-link, h3 .octicon-link, h4 .octicon-link, h5 .octicon-link, h6 .octicon-link {
|
|
||||||
display: none;
|
|
||||||
color: #222222;
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1:hover .anchor, h2:hover .anchor, h3:hover .anchor, h4:hover .anchor, h5:hover .anchor, h6:hover .anchor{
|
|
||||||
padding-left: 8px;
|
|
||||||
margin-left: -24px;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1:hover .anchor .octicon-link, h2:hover .anchor .octicon-link, h3:hover .anchor .octicon-link, h4:hover .anchor .octicon-link, h5:hover .anchor .octicon-link, h6:hover .anchor .octicon-link {
|
|
||||||
display: inline-block;
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
|
||||||
padding: 50px;
|
|
||||||
font: 14px/1.5 "Liberation Sans", Helvetica, Arial, sans-serif;
|
|
||||||
color: #555555;
|
|
||||||
background: #eaeaea
|
|
||||||
}
|
|
||||||
|
|
||||||
h1, h2, h3, h4, h5, h6 {
|
|
||||||
color: #222222;
|
|
||||||
margin: 0 0 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
p, ul, ol, table, pre, dl {
|
|
||||||
margin: 0 0 20px;
|
|
||||||
text-align: justify;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1, h2, h3 {
|
|
||||||
line-height: 1.1;
|
|
||||||
}
|
|
||||||
|
|
||||||
h1 {
|
|
||||||
font-size: 28px;
|
|
||||||
}
|
|
||||||
|
|
||||||
h2 {
|
|
||||||
color: #393939;
|
|
||||||
}
|
|
||||||
|
|
||||||
h3, h4, h5, h6 {
|
|
||||||
color: #494949;
|
|
||||||
}
|
|
||||||
|
|
||||||
a {
|
|
||||||
color: #3399cc;
|
|
||||||
font-weight: 350;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
a small {
|
|
||||||
font-size: 11px;
|
|
||||||
color: #777777;
|
|
||||||
margin-top: -0.6em;
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.wrapper {
|
|
||||||
width: 80%;
|
|
||||||
margin: 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
blockquote {
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0 0 0 20px;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
|
|
||||||
code, pre {
|
|
||||||
font-family: "Liberation Mono", Monaco, Bitstream Vera Sans Mono, Lucida Console, Terminal;
|
|
||||||
color: #222222;
|
|
||||||
font-size: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
pre {
|
|
||||||
padding: 8px 15px;
|
|
||||||
border-radius: 5px;
|
|
||||||
border: 1px solid #e5e5e5;
|
|
||||||
overflow-x: auto;
|
|
||||||
overflow-y: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
input, select{
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
}
|
|
||||||
|
|
||||||
th, td {
|
|
||||||
padding: 5px 10px;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
td {
|
|
||||||
text-align: justify;
|
|
||||||
}
|
|
||||||
|
|
||||||
dt {
|
|
||||||
color: #444444;
|
|
||||||
font-weight: 700;
|
|
||||||
}
|
|
||||||
|
|
||||||
th {
|
|
||||||
text-align: left;
|
|
||||||
color: #444444;
|
|
||||||
}
|
|
||||||
|
|
||||||
img {
|
|
||||||
max-width: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
header {
|
|
||||||
width: 20%;
|
|
||||||
float: left;
|
|
||||||
position: fixed;
|
|
||||||
}
|
|
||||||
|
|
||||||
header ul {
|
|
||||||
list-style: none;
|
|
||||||
height: 40px;
|
|
||||||
padding: 0;
|
|
||||||
background: #eeeeee;
|
|
||||||
border-radius: 5px;
|
|
||||||
border: 1px solid #d2d2d2;
|
|
||||||
box-shadow: inset #fff 0 1px 0, inset rgba(0,0,0,0.03) 0 -1px 0;
|
|
||||||
width: 15%;
|
|
||||||
}
|
|
||||||
|
|
||||||
header li {
|
|
||||||
width: 8%;
|
|
||||||
float: left;
|
|
||||||
border-right: 1px solid #d2d2d2;
|
|
||||||
height: 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
header ul a {
|
|
||||||
line-height: 1;
|
|
||||||
font-size: 11px;
|
|
||||||
color: #999999;
|
|
||||||
display: block;
|
|
||||||
text-align: center;
|
|
||||||
padding-top: 6px;
|
|
||||||
height: 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
strong {
|
|
||||||
color: #222222;
|
|
||||||
font-weight: 700;
|
|
||||||
}
|
|
||||||
|
|
||||||
header ul li + li {
|
|
||||||
width: 8%;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
header ul li + li + li {
|
|
||||||
width: 8%;
|
|
||||||
border-right: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
header ul a strong {
|
|
||||||
font-size: 14px;
|
|
||||||
display: block;
|
|
||||||
color: #222222;
|
|
||||||
}
|
|
||||||
|
|
||||||
section {
|
|
||||||
width: 70%;
|
|
||||||
float: right;
|
|
||||||
padding-bottom: 50px;
|
|
||||||
}
|
|
||||||
|
|
||||||
small {
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
hr {
|
|
||||||
border: 0;
|
|
||||||
background: #ffffff;
|
|
||||||
height: 1px;
|
|
||||||
margin: 0 0 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
footer {
|
|
||||||
width: 20%;
|
|
||||||
float: left;
|
|
||||||
position: fixed;
|
|
||||||
bottom: 50px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media print, screen and (max-width: 960px) {
|
|
||||||
div.wrapper {
|
|
||||||
width: auto;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
header, section, footer {
|
|
||||||
float: none;
|
|
||||||
position: static;
|
|
||||||
width: auto;
|
|
||||||
}
|
|
||||||
header {
|
|
||||||
padding-right: 320px;
|
|
||||||
}
|
|
||||||
section {
|
|
||||||
border: 1px solid #e5e5e5;
|
|
||||||
border-width: 1px 0;
|
|
||||||
padding: 20px 0;
|
|
||||||
margin: 0 0 20px;
|
|
||||||
}
|
|
||||||
header a small {
|
|
||||||
display: inline;
|
|
||||||
}
|
|
||||||
header ul {
|
|
||||||
position: absolute;
|
|
||||||
right: 50px;
|
|
||||||
top: 52px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media print, screen and (max-width: 720px) {
|
|
||||||
body {
|
|
||||||
word-wrap: break-word;
|
|
||||||
}
|
|
||||||
header {
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
header ul, header p.view {
|
|
||||||
position: static;
|
|
||||||
}
|
|
||||||
pre, code {
|
|
||||||
word-wrap: normal;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media print, screen and (max-width: 480px) {
|
|
||||||
body {
|
|
||||||
padding: 15px;
|
|
||||||
}
|
|
||||||
header ul {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media print {
|
|
||||||
body {
|
|
||||||
padding: 0.4in;
|
|
||||||
font-size: 12pt;
|
|
||||||
color: #444444;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis
|
|
||||||
|
|
||||||
import akka.actor.{Actor, Props}
|
|
||||||
import akka.http.scaladsl.Http
|
|
||||||
import akka.stream.ActorMaterializer
|
|
||||||
import com.typesafe.scalalogging.StrictLogging
|
|
||||||
import me.arcanis.ffxivbis.http.RootEndpoint
|
|
||||||
import me.arcanis.ffxivbis.service.{Ariyala, PartyService}
|
|
||||||
import me.arcanis.ffxivbis.service.impl.DatabaseImpl
|
|
||||||
import me.arcanis.ffxivbis.storage.Migration
|
|
||||||
|
|
||||||
import scala.concurrent.{Await, ExecutionContext}
|
|
||||||
import scala.concurrent.duration.Duration
|
|
||||||
import scala.util.{Failure, Success}
|
|
||||||
|
|
||||||
class Application extends Actor with StrictLogging {
|
|
||||||
implicit private val executionContext: ExecutionContext = context.system.dispatcher
|
|
||||||
implicit private val materializer: ActorMaterializer = ActorMaterializer()
|
|
||||||
|
|
||||||
private val config = context.system.settings.config
|
|
||||||
private val host = config.getString("me.arcanis.ffxivbis.web.host")
|
|
||||||
private val port = config.getInt("me.arcanis.ffxivbis.web.port")
|
|
||||||
|
|
||||||
override def receive: Receive = Actor.emptyBehavior
|
|
||||||
|
|
||||||
Migration(config).onComplete {
|
|
||||||
case Success(_) =>
|
|
||||||
val ariyala = context.system.actorOf(Ariyala.props, "ariyala")
|
|
||||||
val storage = context.system.actorOf(DatabaseImpl.props, "storage")
|
|
||||||
val party = context.system.actorOf(PartyService.props(storage), "party")
|
|
||||||
val http = new RootEndpoint(context.system, party, ariyala)
|
|
||||||
|
|
||||||
logger.info(s"start server at $host:$port")
|
|
||||||
val bind = Http()(context.system).bindAndHandle(http.route, host, port)
|
|
||||||
Await.result(context.system.whenTerminated, Duration.Inf)
|
|
||||||
bind.foreach(_.unbind())
|
|
||||||
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
object Application {
|
|
||||||
def props: Props = Props(new Application)
|
|
||||||
}
|
|
@ -1,20 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis
|
|
||||||
|
|
||||||
import akka.actor.ActorSystem
|
|
||||||
import com.typesafe.config.ConfigFactory
|
|
||||||
|
|
||||||
object ffxivbis {
|
|
||||||
def main(args: Array[String]): Unit = {
|
|
||||||
val config = ConfigFactory.load()
|
|
||||||
val actorSystem = ActorSystem("ffxivbis", config)
|
|
||||||
actorSystem.actorOf(Application.props, "ffxivbis")
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.models.{BiS, Job, Piece}
|
|
||||||
import me.arcanis.ffxivbis.service.Ariyala
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
class AriyalaHelper(ariyala: ActorRef) {
|
|
||||||
|
|
||||||
def downloadBiS(link: String, job: Job.Job)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[BiS] =
|
|
||||||
(ariyala ? Ariyala.GetBiS(link, job)).mapTo[BiS]
|
|
||||||
}
|
|
@ -1,61 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.model.headers._
|
|
||||||
import akka.http.scaladsl.server.AuthenticationFailedRejection._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.models.{Permission, User}
|
|
||||||
import me.arcanis.ffxivbis.service.impl.DatabaseUserHandler
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
// idea comes from https://synkre.com/bcrypt-for-akka-http-password-encryption/
|
|
||||||
trait Authorization {
|
|
||||||
|
|
||||||
def storage: ActorRef
|
|
||||||
|
|
||||||
def authenticateBasicBCrypt[T](realm: String,
|
|
||||||
authenticate: (String, String) => Future[Option[T]]): Directive1[T] = {
|
|
||||||
def challenge = HttpChallenges.basic(realm)
|
|
||||||
|
|
||||||
extractCredentials.flatMap {
|
|
||||||
case Some(BasicHttpCredentials(username, password)) =>
|
|
||||||
onSuccess(authenticate(username, password)).flatMap {
|
|
||||||
case Some(client) => provide(client)
|
|
||||||
case None => reject(AuthenticationFailedRejection(CredentialsRejected, challenge))
|
|
||||||
}
|
|
||||||
case _ => reject(AuthenticationFailedRejection(CredentialsMissing, challenge))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def authenticator(scope: Permission.Value)(partyId: String)
|
|
||||||
(username: String, password: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Option[String]] =
|
|
||||||
(storage ? DatabaseUserHandler.GetUser(partyId, username)).mapTo[Option[User]].map {
|
|
||||||
case Some(user) if user.verify(password) && user.verityScope(scope) => Some(username)
|
|
||||||
case _ => None
|
|
||||||
}
|
|
||||||
|
|
||||||
def authAdmin(partyId: String)(username: String, password: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Option[String]] =
|
|
||||||
authenticator(Permission.admin)(partyId)(username, password)
|
|
||||||
|
|
||||||
def authGet(partyId: String)(username: String, password: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Option[String]] =
|
|
||||||
authenticator(Permission.get)(partyId)(username, password)
|
|
||||||
|
|
||||||
def authPost(partyId: String)(username: String, password: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Option[String]] =
|
|
||||||
authenticator(Permission.post)(partyId)(username, password)
|
|
||||||
}
|
|
@ -1,47 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json.ApiAction
|
|
||||||
import me.arcanis.ffxivbis.models.{Piece, Player, PlayerId}
|
|
||||||
import me.arcanis.ffxivbis.service.impl.DatabaseBiSHandler
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
class BiSHelper(storage: ActorRef, ariyala: ActorRef) extends AriyalaHelper(ariyala) {
|
|
||||||
|
|
||||||
def addPieceBiS(playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseBiSHandler.AddPieceToBis(playerId, piece.withJob(playerId.job))).mapTo[Int]
|
|
||||||
|
|
||||||
def bis(partyId: String, playerId: Option[PlayerId])
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Seq[Player]] =
|
|
||||||
(storage ? DatabaseBiSHandler.GetBiS(partyId, playerId)).mapTo[Seq[Player]]
|
|
||||||
|
|
||||||
def doModifyBiS(action: ApiAction.Value, playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
action match {
|
|
||||||
case ApiAction.add => addPieceBiS(playerId, piece)
|
|
||||||
case ApiAction.remove => removePieceBiS(playerId, piece)
|
|
||||||
}
|
|
||||||
|
|
||||||
def putBiS(playerId: PlayerId, link: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Unit] =
|
|
||||||
downloadBiS(link, playerId.job).flatMap { bis =>
|
|
||||||
Future.traverse(bis.pieces)(addPieceBiS(playerId, _))
|
|
||||||
}.map(_ => ())
|
|
||||||
|
|
||||||
def removePieceBiS(playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseBiSHandler.RemovePieceFromBiS(playerId, piece)).mapTo[Int]
|
|
||||||
|
|
||||||
}
|
|
@ -1,45 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json.ApiAction
|
|
||||||
import me.arcanis.ffxivbis.models.{Piece, Player, PlayerId, PlayerIdWithCounters}
|
|
||||||
import me.arcanis.ffxivbis.service.LootSelector.LootSelectorResult
|
|
||||||
import me.arcanis.ffxivbis.service.impl.DatabaseLootHandler
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
class LootHelper(storage: ActorRef) {
|
|
||||||
|
|
||||||
def addPieceLoot(playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseLootHandler.AddPieceTo(playerId, piece)).mapTo[Int]
|
|
||||||
|
|
||||||
def doModifyLoot(action: ApiAction.Value, playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
action match {
|
|
||||||
case ApiAction.add => addPieceLoot(playerId, piece)
|
|
||||||
case ApiAction.remove => removePieceLoot(playerId, piece)
|
|
||||||
}
|
|
||||||
|
|
||||||
def loot(partyId: String, playerId: Option[PlayerId])
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Seq[Player]] =
|
|
||||||
(storage ? DatabaseLootHandler.GetLoot(partyId, playerId)).mapTo[Seq[Player]]
|
|
||||||
|
|
||||||
def removePieceLoot(playerId: PlayerId, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseLootHandler.RemovePieceFrom(playerId, piece)).mapTo[Int]
|
|
||||||
|
|
||||||
def suggestPiece(partyId: String, piece: Piece)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Seq[PlayerIdWithCounters]] =
|
|
||||||
(storage ? DatabaseLootHandler.SuggestLoot(partyId, piece)).mapTo[LootSelectorResult].map(_.result)
|
|
||||||
}
|
|
@ -1,53 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json.ApiAction
|
|
||||||
import me.arcanis.ffxivbis.models.{Party, Player, PlayerId}
|
|
||||||
import me.arcanis.ffxivbis.service.impl.{DatabaseBiSHandler, DatabasePartyHandler}
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
class PlayerHelper(storage: ActorRef, ariyala: ActorRef) extends AriyalaHelper(ariyala) {
|
|
||||||
|
|
||||||
def addPlayer(player: Player)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabasePartyHandler.AddPlayer(player)).mapTo[Int].map { res =>
|
|
||||||
player.link match {
|
|
||||||
case Some(link) =>
|
|
||||||
downloadBiS(link, player.job).map { bis =>
|
|
||||||
bis.pieces.map(storage ? DatabaseBiSHandler.AddPieceToBis(player.playerId, _))
|
|
||||||
}.map(_ => res)
|
|
||||||
case None => Future.successful(res)
|
|
||||||
}
|
|
||||||
}.flatten
|
|
||||||
|
|
||||||
def doModifyPlayer(action: ApiAction.Value, player: Player)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
action match {
|
|
||||||
case ApiAction.add => addPlayer(player)
|
|
||||||
case ApiAction.remove => removePlayer(player.playerId)
|
|
||||||
}
|
|
||||||
|
|
||||||
def getPlayers(partyId: String, maybePlayerId: Option[PlayerId])
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Seq[Player]] =
|
|
||||||
maybePlayerId match {
|
|
||||||
case Some(playerId) =>
|
|
||||||
(storage ? DatabasePartyHandler.GetPlayer(playerId)).mapTo[Option[Player]].map(_.toSeq)
|
|
||||||
case None =>
|
|
||||||
(storage ? DatabasePartyHandler.GetParty(partyId)).mapTo[Party].map(_.players.values.toSeq)
|
|
||||||
}
|
|
||||||
|
|
||||||
def removePlayer(playerId: PlayerId)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabasePartyHandler.RemovePlayer(playerId)).mapTo[Int]
|
|
||||||
}
|
|
@ -1,70 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import java.time.Instant
|
|
||||||
|
|
||||||
import akka.actor.{ActorRef, ActorSystem}
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.util.Timeout
|
|
||||||
import com.typesafe.scalalogging.{Logger, StrictLogging}
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.RootApiV1Endpoint
|
|
||||||
import me.arcanis.ffxivbis.http.view.RootView
|
|
||||||
|
|
||||||
class RootEndpoint(system: ActorSystem, storage: ActorRef, ariyala: ActorRef)
|
|
||||||
extends StrictLogging {
|
|
||||||
import me.arcanis.ffxivbis.utils.Implicits._
|
|
||||||
|
|
||||||
private val config = system.settings.config
|
|
||||||
|
|
||||||
implicit val timeout: Timeout =
|
|
||||||
config.getDuration("me.arcanis.ffxivbis.settings.request-timeout")
|
|
||||||
|
|
||||||
private val rootApiV1Endpoint: RootApiV1Endpoint = new RootApiV1Endpoint(storage, ariyala)
|
|
||||||
private val rootView: RootView = new RootView(storage, ariyala)
|
|
||||||
private val httpLogger = Logger("http")
|
|
||||||
|
|
||||||
private val withHttpLog: Directive0 =
|
|
||||||
extractRequestContext.flatMap { context =>
|
|
||||||
val start = Instant.now.toEpochMilli
|
|
||||||
mapResponse { response =>
|
|
||||||
val time = (Instant.now.toEpochMilli - start) / 1000.0
|
|
||||||
httpLogger.debug(s"""- - [${Instant.now}] "${context.request.method.name()} ${context.request.uri.path}" ${response.status.intValue()} ${response.entity.getContentLengthOption.getAsLong} $time""")
|
|
||||||
response
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def route: Route =
|
|
||||||
withHttpLog {
|
|
||||||
apiRoute ~ htmlRoute ~ Swagger.routes ~ swaggerUIRoute
|
|
||||||
}
|
|
||||||
|
|
||||||
private def apiRoute: Route =
|
|
||||||
ignoreTrailingSlash {
|
|
||||||
pathPrefix("api") {
|
|
||||||
pathPrefix(Segment) {
|
|
||||||
case "v1" => rootApiV1Endpoint.route
|
|
||||||
case _ => reject
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private def htmlRoute: Route =
|
|
||||||
ignoreTrailingSlash {
|
|
||||||
pathPrefix("static") {
|
|
||||||
getFromResourceDirectory("static")
|
|
||||||
} ~ rootView.route
|
|
||||||
}
|
|
||||||
|
|
||||||
private def swaggerUIRoute: Route =
|
|
||||||
path("swagger") {
|
|
||||||
getFromResource("swagger/index.html")
|
|
||||||
} ~ getFromResourceDirectory("swagger")
|
|
||||||
}
|
|
@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import com.github.swagger.akka.SwaggerHttpService
|
|
||||||
import com.github.swagger.akka.model.Info
|
|
||||||
import io.swagger.v3.oas.models.security.SecurityScheme
|
|
||||||
|
|
||||||
object Swagger extends SwaggerHttpService {
|
|
||||||
override val apiClasses: Set[Class[_]] = Set(
|
|
||||||
classOf[api.v1.BiSEndpoint], classOf[api.v1.LootEndpoint],
|
|
||||||
classOf[api.v1.PlayerEndpoint], classOf[api.v1.UserEndpoint]
|
|
||||||
)
|
|
||||||
|
|
||||||
override val info: Info = Info()
|
|
||||||
|
|
||||||
private val basicAuth = new SecurityScheme()
|
|
||||||
.description("basic http auth")
|
|
||||||
.`type`(SecurityScheme.Type.HTTP)
|
|
||||||
.in(SecurityScheme.In.HEADER)
|
|
||||||
.scheme("bearer")
|
|
||||||
override def securitySchemes: Map[String, SecurityScheme] = Map("basic auth" -> basicAuth)
|
|
||||||
|
|
||||||
override val unwantedDefinitions: Seq[String] =
|
|
||||||
Seq("Function1", "Function1RequestContextFutureRouteResult")
|
|
||||||
}
|
|
@ -1,40 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.pattern.ask
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.models.User
|
|
||||||
import me.arcanis.ffxivbis.service.PartyService
|
|
||||||
import me.arcanis.ffxivbis.service.impl.DatabaseUserHandler
|
|
||||||
|
|
||||||
import scala.concurrent.{ExecutionContext, Future}
|
|
||||||
|
|
||||||
class UserHelper(storage: ActorRef) {
|
|
||||||
|
|
||||||
def addUser(user: User, isHashedPassword: Boolean)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseUserHandler.AddUser(user, isHashedPassword)).mapTo[Int]
|
|
||||||
|
|
||||||
def newPartyId(implicit executionContext: ExecutionContext, timeout: Timeout): Future[String] =
|
|
||||||
(storage ? PartyService.GetNewPartyId).mapTo[String]
|
|
||||||
|
|
||||||
def user(partyId: String, username: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Option[User]] =
|
|
||||||
(storage ? DatabaseUserHandler.GetUser(partyId, username)).mapTo[Option[User]]
|
|
||||||
|
|
||||||
def users(partyId: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Seq[User]] =
|
|
||||||
(storage ? DatabaseUserHandler.GetUsers(partyId)).mapTo[Seq[User]]
|
|
||||||
|
|
||||||
def removeUser(partyId: String, username: String)
|
|
||||||
(implicit executionContext: ExecutionContext, timeout: Timeout): Future[Int] =
|
|
||||||
(storage ? DatabaseUserHandler.DeleteUser(partyId, username)).mapTo[Int]
|
|
||||||
}
|
|
@ -1,147 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.model.{HttpEntity, StatusCodes}
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.util.Timeout
|
|
||||||
import com.typesafe.scalalogging.StrictLogging
|
|
||||||
import io.swagger.v3.oas.annotations.enums.ParameterIn
|
|
||||||
import io.swagger.v3.oas.annotations.media.{ArraySchema, Content, Schema}
|
|
||||||
import io.swagger.v3.oas.annotations.responses.ApiResponse
|
|
||||||
import io.swagger.v3.oas.annotations.{Operation, Parameter}
|
|
||||||
import io.swagger.v3.oas.annotations.parameters.RequestBody
|
|
||||||
import io.swagger.v3.oas.annotations.security.SecurityRequirement
|
|
||||||
import javax.ws.rs.{Consumes, GET, POST, PUT, Path, Produces}
|
|
||||||
import me.arcanis.ffxivbis.http.{Authorization, BiSHelper}
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json._
|
|
||||||
import me.arcanis.ffxivbis.models.PlayerId
|
|
||||||
|
|
||||||
import scala.util.{Failure, Success}
|
|
||||||
|
|
||||||
@Path("api/v1")
|
|
||||||
class BiSEndpoint(override val storage: ActorRef, ariyala: ActorRef)(implicit timeout: Timeout)
|
|
||||||
extends BiSHelper(storage, ariyala) with Authorization with JsonSupport {
|
|
||||||
|
|
||||||
def route: Route = createBiS ~ getBiS ~ modifyBiS
|
|
||||||
|
|
||||||
@PUT
|
|
||||||
@Path("party/{partyId}/bis")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Operation(summary = "create best in slot", description = "Create the best in slot set",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "player best in slot description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[PlayerBiSLinkResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "201", description = "Best in slot set has been created"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("post"))),
|
|
||||||
tags = Array("best in slot"),
|
|
||||||
)
|
|
||||||
def createBiS: Route =
|
|
||||||
path("party" / Segment / "bis") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authPost(partyId)) { _ =>
|
|
||||||
put {
|
|
||||||
entity(as[PlayerBiSLinkResponse]) { bisLink =>
|
|
||||||
val playerId = bisLink.playerId.withPartyId(partyId)
|
|
||||||
onComplete(putBiS(playerId, bisLink.link)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Created, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@GET
|
|
||||||
@Path("party/{partyId}/bis")
|
|
||||||
@Produces(value = Array("application/json"))
|
|
||||||
@Operation(summary = "get best in slot", description = "Return the best in slot items",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
new Parameter(name = "nick", in = ParameterIn.QUERY, description = "player nick name to filter", example = "Siuan Sanche"),
|
|
||||||
new Parameter(name = "job", in = ParameterIn.QUERY, description = "player job to filter", example = "DNC"),
|
|
||||||
),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Best in slot",
|
|
||||||
content = Array(new Content(
|
|
||||||
array = new ArraySchema(schema = new Schema(implementation = classOf[PlayerResponse]))
|
|
||||||
))),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("get"))),
|
|
||||||
tags = Array("best in slot"),
|
|
||||||
)
|
|
||||||
def getBiS: Route =
|
|
||||||
path("party" / Segment / "bis") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authGet(partyId)) { _ =>
|
|
||||||
get {
|
|
||||||
parameters("nick".as[String].?, "job".as[String].?) { (maybeNick, maybeJob) =>
|
|
||||||
val playerId = PlayerId(partyId, maybeNick, maybeJob)
|
|
||||||
onComplete(bis(partyId, playerId)) {
|
|
||||||
case Success(response) => complete(response.map(PlayerResponse.fromPlayer))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@POST
|
|
||||||
@Path("party/{partyId}/bis")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Operation(summary = "modify best in slot", description = "Add or remove an item from the best in slot",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "action and piece description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[PieceActionResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "202", description = "Best in slot set has been modified"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("post"))),
|
|
||||||
tags = Array("best in slot"),
|
|
||||||
)
|
|
||||||
def modifyBiS: Route =
|
|
||||||
path("party" / Segment / "bis") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authPost(partyId)) { _ =>
|
|
||||||
post {
|
|
||||||
entity(as[PieceActionResponse]) { action =>
|
|
||||||
val playerId = action.playerIdResponse.withPartyId(partyId)
|
|
||||||
onComplete(doModifyBiS(action.action, playerId, action.piece.toPiece)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Accepted, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.http.scaladsl.model._
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import com.typesafe.scalalogging.StrictLogging
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json._
|
|
||||||
import spray.json._
|
|
||||||
|
|
||||||
trait HttpHandler extends StrictLogging { this: JsonSupport =>
|
|
||||||
|
|
||||||
implicit def exceptionHandler: ExceptionHandler = ExceptionHandler {
|
|
||||||
case other: Exception =>
|
|
||||||
logger.error("exception during request completion", other)
|
|
||||||
complete(StatusCodes.InternalServerError, ErrorResponse("unknown server error"))
|
|
||||||
}
|
|
||||||
|
|
||||||
implicit def rejectionHandler: RejectionHandler =
|
|
||||||
RejectionHandler.default
|
|
||||||
.mapRejectionResponse {
|
|
||||||
case response @ HttpResponse(_, _, entity: HttpEntity.Strict, _) =>
|
|
||||||
val message = ErrorResponse(entity.data.utf8String).toJson
|
|
||||||
response.copy(entity = HttpEntity(ContentTypes.`application/json`, message.compactPrint))
|
|
||||||
case other => other
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,147 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.model.{HttpEntity, StatusCodes}
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.util.Timeout
|
|
||||||
import io.swagger.v3.oas.annotations.enums.ParameterIn
|
|
||||||
import io.swagger.v3.oas.annotations.media.{ArraySchema, Content, Schema}
|
|
||||||
import io.swagger.v3.oas.annotations.responses.ApiResponse
|
|
||||||
import io.swagger.v3.oas.annotations.{Operation, Parameter}
|
|
||||||
import io.swagger.v3.oas.annotations.parameters.RequestBody
|
|
||||||
import io.swagger.v3.oas.annotations.security.SecurityRequirement
|
|
||||||
import javax.ws.rs.{Consumes, GET, POST, PUT, Path, Produces}
|
|
||||||
import me.arcanis.ffxivbis.http.{Authorization, LootHelper}
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json._
|
|
||||||
import me.arcanis.ffxivbis.models.PlayerId
|
|
||||||
|
|
||||||
import scala.util.{Failure, Success}
|
|
||||||
|
|
||||||
@Path("api/v1")
|
|
||||||
class LootEndpoint(override val storage: ActorRef)(implicit timeout: Timeout)
|
|
||||||
extends LootHelper(storage) with Authorization with JsonSupport with HttpHandler {
|
|
||||||
|
|
||||||
def route: Route = getLoot ~ modifyLoot
|
|
||||||
|
|
||||||
@GET
|
|
||||||
@Path("party/{partyId}/loot")
|
|
||||||
@Produces(value = Array("application/json"))
|
|
||||||
@Operation(summary = "get loot list", description = "Return the looted items",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
new Parameter(name = "nick", in = ParameterIn.QUERY, description = "player nick name to filter", example = "Siuan Sanche"),
|
|
||||||
new Parameter(name = "job", in = ParameterIn.QUERY, description = "player job to filter", example = "DNC"),
|
|
||||||
),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Loot list",
|
|
||||||
content = Array(new Content(
|
|
||||||
array = new ArraySchema(schema = new Schema(implementation = classOf[PlayerResponse]))
|
|
||||||
))),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("get"))),
|
|
||||||
tags = Array("loot"),
|
|
||||||
)
|
|
||||||
def getLoot: Route =
|
|
||||||
path("party" / Segment / "loot") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authGet(partyId)) { _ =>
|
|
||||||
get {
|
|
||||||
parameters("nick".as[String].?, "job".as[String].?) { (maybeNick, maybeJob) =>
|
|
||||||
val playerId = PlayerId(partyId, maybeNick, maybeJob)
|
|
||||||
onComplete(loot(partyId, playerId)) {
|
|
||||||
case Success(response) => complete(response.map(PlayerResponse.fromPlayer))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@POST
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Path("party/{partyId}/loot")
|
|
||||||
@Operation(summary = "modify loot list", description = "Add or remove an item from the loot list",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "action and piece description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[PieceActionResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "202", description = "Loot list has been modified"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("post"))),
|
|
||||||
tags = Array("loot"),
|
|
||||||
)
|
|
||||||
def modifyLoot: Route =
|
|
||||||
path("party" / Segment / "loot") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authPost(partyId)) { _ =>
|
|
||||||
post {
|
|
||||||
entity(as[PieceActionResponse]) { action =>
|
|
||||||
val playerId = action.playerIdResponse.withPartyId(partyId)
|
|
||||||
onComplete(doModifyLoot(action.action, playerId, action.piece.toPiece)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Accepted, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@PUT
|
|
||||||
@Path("party/{partyId}/loot")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Produces(value = Array("application/json"))
|
|
||||||
@Operation(summary = "suggest loot", description = "Suggest loot piece to party",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "piece description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[PieceResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Players with counters ordered by priority to get this item",
|
|
||||||
content = Array(new Content(
|
|
||||||
array = new ArraySchema(schema = new Schema(implementation = classOf[PlayerIdWithCountersResponse])),
|
|
||||||
))),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("get"))),
|
|
||||||
tags = Array("loot"),
|
|
||||||
)
|
|
||||||
def suggestLoot: Route =
|
|
||||||
path("party" / Segment / "loot") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authGet(partyId)) { _ =>
|
|
||||||
put {
|
|
||||||
entity(as[PieceResponse]) { piece =>
|
|
||||||
onComplete(suggestPiece(partyId, piece.toPiece)) {
|
|
||||||
case Success(response) => complete(response.map(PlayerIdWithCountersResponse.fromPlayerId))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,106 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.model.{HttpEntity, StatusCodes}
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.util.Timeout
|
|
||||||
import io.swagger.v3.oas.annotations.enums.ParameterIn
|
|
||||||
import io.swagger.v3.oas.annotations.media.{ArraySchema, Content, Schema}
|
|
||||||
import io.swagger.v3.oas.annotations.responses.ApiResponse
|
|
||||||
import io.swagger.v3.oas.annotations.{Operation, Parameter}
|
|
||||||
import io.swagger.v3.oas.annotations.parameters.RequestBody
|
|
||||||
import io.swagger.v3.oas.annotations.security.SecurityRequirement
|
|
||||||
import javax.ws.rs.{Consumes, GET, POST, Path, Produces}
|
|
||||||
import me.arcanis.ffxivbis.http.{Authorization, PlayerHelper}
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json._
|
|
||||||
import me.arcanis.ffxivbis.models.PlayerId
|
|
||||||
|
|
||||||
import scala.util.{Failure, Success}
|
|
||||||
|
|
||||||
@Path("api/v1")
|
|
||||||
class PlayerEndpoint(override val storage: ActorRef, ariyala: ActorRef)(implicit timeout: Timeout)
|
|
||||||
extends PlayerHelper(storage, ariyala) with Authorization with JsonSupport with HttpHandler {
|
|
||||||
|
|
||||||
def route: Route = getParty ~ modifyParty
|
|
||||||
|
|
||||||
@GET
|
|
||||||
@Path("party/{partyId}")
|
|
||||||
@Produces(value = Array("application/json"))
|
|
||||||
@Operation(summary = "get party", description = "Return the players who belong to the party",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
new Parameter(name = "nick", in = ParameterIn.QUERY, description = "player nick name to filter", example = "Siuan Sanche"),
|
|
||||||
new Parameter(name = "job", in = ParameterIn.QUERY, description = "player job to filter", example = "DNC"),
|
|
||||||
),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Players list",
|
|
||||||
content = Array(new Content(
|
|
||||||
array = new ArraySchema(schema = new Schema(implementation = classOf[PlayerResponse])),
|
|
||||||
))),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("get"))),
|
|
||||||
tags = Array("party"),
|
|
||||||
)
|
|
||||||
def getParty: Route =
|
|
||||||
path("party" / Segment) { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authGet(partyId)) { _ =>
|
|
||||||
get {
|
|
||||||
parameters("nick".as[String].?, "job".as[String].?) { (maybeNick, maybeJob) =>
|
|
||||||
val playerId = PlayerId(partyId, maybeNick, maybeJob)
|
|
||||||
onComplete(getPlayers(partyId, playerId)) {
|
|
||||||
case Success(response) => complete(response.map(PlayerResponse.fromPlayer))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@POST
|
|
||||||
@Path("party/{partyId}")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Operation(summary = "modify party", description = "Add or remove a player from party list",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "player description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[PlayerActionResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "202", description = "Party has been modified"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("post"))),
|
|
||||||
tags = Array("party"),
|
|
||||||
)
|
|
||||||
def modifyParty: Route =
|
|
||||||
path("party" / Segment) { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authPost(partyId)) { _ =>
|
|
||||||
entity(as[PlayerActionResponse]) { action =>
|
|
||||||
val player = action.playerIdResponse.toPlayer.copy(partyId = partyId)
|
|
||||||
onComplete(doModifyPlayer(action.action, player)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Accepted, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server.Route
|
|
||||||
import akka.util.Timeout
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json.JsonSupport
|
|
||||||
|
|
||||||
class RootApiV1Endpoint(storage: ActorRef, ariyala: ActorRef)
|
|
||||||
(implicit timeout: Timeout)
|
|
||||||
extends JsonSupport with HttpHandler {
|
|
||||||
|
|
||||||
private val biSEndpoint = new BiSEndpoint(storage, ariyala)
|
|
||||||
private val lootEndpoint = new LootEndpoint(storage)
|
|
||||||
private val playerEndpoint = new PlayerEndpoint(storage, ariyala)
|
|
||||||
private val userEndpoint = new UserEndpoint(storage)
|
|
||||||
|
|
||||||
def route: Route =
|
|
||||||
handleExceptions(exceptionHandler) {
|
|
||||||
handleRejections(rejectionHandler) {
|
|
||||||
biSEndpoint.route ~ lootEndpoint.route ~ playerEndpoint.route ~ userEndpoint.route
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,166 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1
|
|
||||||
|
|
||||||
import akka.actor.ActorRef
|
|
||||||
import akka.http.scaladsl.model.{HttpEntity, StatusCodes}
|
|
||||||
import akka.http.scaladsl.server.Directives._
|
|
||||||
import akka.http.scaladsl.server._
|
|
||||||
import akka.util.Timeout
|
|
||||||
import io.swagger.v3.oas.annotations.enums.ParameterIn
|
|
||||||
import io.swagger.v3.oas.annotations.media.{ArraySchema, Content, Schema}
|
|
||||||
import io.swagger.v3.oas.annotations.responses.ApiResponse
|
|
||||||
import io.swagger.v3.oas.annotations.{Operation, Parameter}
|
|
||||||
import io.swagger.v3.oas.annotations.parameters.RequestBody
|
|
||||||
import io.swagger.v3.oas.annotations.security.SecurityRequirement
|
|
||||||
import javax.ws.rs.{Consumes, DELETE, GET, POST, PUT, Path, Produces}
|
|
||||||
import me.arcanis.ffxivbis.http.{Authorization, UserHelper}
|
|
||||||
import me.arcanis.ffxivbis.http.api.v1.json._
|
|
||||||
import me.arcanis.ffxivbis.models.Permission
|
|
||||||
|
|
||||||
import scala.util.{Failure, Success}
|
|
||||||
|
|
||||||
@Path("api/v1")
|
|
||||||
class UserEndpoint(override val storage: ActorRef)(implicit timeout: Timeout)
|
|
||||||
extends UserHelper(storage) with Authorization with JsonSupport {
|
|
||||||
|
|
||||||
def route: Route = createParty ~ createUser ~ deleteUser ~ getUsers
|
|
||||||
|
|
||||||
@PUT
|
|
||||||
@Path("party")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Operation(summary = "create new party", description = "Create new party with specified ID",
|
|
||||||
requestBody = new RequestBody(description = "party administrator description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[UserResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Party has been created"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "406", description = "Party with the specified ID already exists"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
tags = Array("party"),
|
|
||||||
)
|
|
||||||
def createParty: Route =
|
|
||||||
path("party") {
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
put {
|
|
||||||
entity(as[UserResponse]) { user =>
|
|
||||||
onComplete(newPartyId) {
|
|
||||||
case Success(partyId) =>
|
|
||||||
val admin = user.toUser.copy(partyId = partyId, permission = Permission.admin)
|
|
||||||
onComplete(addUser(admin, isHashedPassword = false)) {
|
|
||||||
case Success(_) => complete(PartyIdResponse(partyId))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@POST
|
|
||||||
@Path("party/{partyId}/users")
|
|
||||||
@Consumes(value = Array("application/json"))
|
|
||||||
@Operation(summary = "create new user", description = "Add an user to the specified party",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
requestBody = new RequestBody(description = "user description", required = true,
|
|
||||||
content = Array(new Content(schema = new Schema(implementation = classOf[UserResponse])))),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "201", description = "User has been created"),
|
|
||||||
new ApiResponse(responseCode = "400", description = "Invalid parameters were supplied"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("admin"))),
|
|
||||||
tags = Array("users"),
|
|
||||||
)
|
|
||||||
def createUser: Route =
|
|
||||||
path("party" / Segment / "users") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authAdmin(partyId)) { _ =>
|
|
||||||
post {
|
|
||||||
entity(as[UserResponse]) { user =>
|
|
||||||
val withPartyId = user.toUser.copy(partyId = partyId)
|
|
||||||
onComplete(addUser(withPartyId, isHashedPassword = false)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Accepted, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@DELETE
|
|
||||||
@Path("party/{partyId}/users/{username}")
|
|
||||||
@Operation(summary = "remove user", description = "Remove an user from the specified party",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
new Parameter(name = "username", in = ParameterIn.PATH, description = "username to remove", example = "siuan"),
|
|
||||||
),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "202", description = "User has been removed"),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("admin"))),
|
|
||||||
tags = Array("users"),
|
|
||||||
)
|
|
||||||
def deleteUser: Route =
|
|
||||||
path("party" / Segment / "users" / Segment) { (partyId, username) =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authAdmin(partyId)) { _ =>
|
|
||||||
delete {
|
|
||||||
onComplete(removeUser(partyId, username)) {
|
|
||||||
case Success(_) => complete(StatusCodes.Accepted, HttpEntity.Empty)
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@GET
|
|
||||||
@Path("party/{partyId}/users")
|
|
||||||
@Produces(value = Array("application/json"))
|
|
||||||
@Operation(summary = "get users", description = "Return the list of users belong to party",
|
|
||||||
parameters = Array(
|
|
||||||
new Parameter(name = "partyId", in = ParameterIn.PATH, description = "unique party ID", example = "abcdefgh"),
|
|
||||||
),
|
|
||||||
responses = Array(
|
|
||||||
new ApiResponse(responseCode = "200", description = "Users list",
|
|
||||||
content = Array(new Content(
|
|
||||||
array = new ArraySchema(schema = new Schema(implementation = classOf[UserResponse])),
|
|
||||||
))),
|
|
||||||
new ApiResponse(responseCode = "401", description = "Supplied authorization is invalid"),
|
|
||||||
new ApiResponse(responseCode = "403", description = "Access is forbidden"),
|
|
||||||
new ApiResponse(responseCode = "500", description = "Internal server error"),
|
|
||||||
),
|
|
||||||
security = Array(new SecurityRequirement(name = "basic auth", scopes = Array("admin"))),
|
|
||||||
tags = Array("users"),
|
|
||||||
)
|
|
||||||
def getUsers: Route =
|
|
||||||
path("party" / Segment / "users") { partyId =>
|
|
||||||
extractExecutionContext { implicit executionContext =>
|
|
||||||
authenticateBasicBCrypt(s"party $partyId", authAdmin(partyId)) { _ =>
|
|
||||||
get {
|
|
||||||
onComplete(users(partyId)) {
|
|
||||||
case Success(response) => complete(response.map(UserResponse.fromUser))
|
|
||||||
case Failure(exception) => throw exception
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,13 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Evgeniy Alekseev.
|
|
||||||
*
|
|
||||||
* This file is part of ffxivbis
|
|
||||||
* (see https://github.com/arcan1s/ffxivbis).
|
|
||||||
*
|
|
||||||
* License: 3-clause BSD, see https://opensource.org/licenses/BSD-3-Clause
|
|
||||||
*/
|
|
||||||
package me.arcanis.ffxivbis.http.api.v1.json
|
|
||||||
|
|
||||||
object ApiAction extends Enumeration {
|
|
||||||
val add, remove = Value
|
|
||||||
}
|
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user