forked from kompetenzinventar/ki-backend
Compare commits
13 Commits
feature-pr
...
main
Author | SHA1 | Date | |
---|---|---|---|
d10706e301 | |||
fc01bec163
|
|||
3bd9b03002
|
|||
ace1b1ed85
|
|||
09669cf369
|
|||
5dc0f7153d | |||
5d259635a2
|
|||
b09b072261 | |||
7e8c5a7de0 | |||
a9f9c36eda
|
|||
a349eff9b0
|
|||
9e48953fc3 | |||
3dcba71a6d
|
17
.drone.yml
17
.drone.yml
@ -4,7 +4,7 @@ type: docker
|
||||
name: default
|
||||
|
||||
steps:
|
||||
- name: qa
|
||||
- name: qa_main
|
||||
image: python:3.8-alpine
|
||||
commands:
|
||||
- apk add --no-cache gcc g++ musl-dev python3-dev
|
||||
@ -12,3 +12,18 @@ steps:
|
||||
- pipenv install --dev
|
||||
- pipenv run flake8
|
||||
- pipenv run python -m unittest discover ki
|
||||
when:
|
||||
branch:
|
||||
- main
|
||||
|
||||
- name: qa_pr
|
||||
image: python:3.8-alpine
|
||||
commands:
|
||||
- apk add --no-cache gcc g++ musl-dev python3-dev
|
||||
- pip3 install pipenv
|
||||
- pipenv install --dev
|
||||
- pipenv run flake8
|
||||
- pipenv run python -m unittest discover ki
|
||||
when:
|
||||
event:
|
||||
- pull_request
|
||||
|
@ -6,3 +6,9 @@
|
||||
language: system
|
||||
files: ^.*\.py$
|
||||
exclude: ^(migrations).*$
|
||||
- id: unittest
|
||||
name: unittest
|
||||
entry: python -m unittest discover ki
|
||||
language: system
|
||||
exclude: .*
|
||||
always_run: true
|
||||
|
3
.style.yapf
Normal file
3
.style.yapf
Normal file
@ -0,0 +1,3 @@
|
||||
[style]
|
||||
based_on_style = pep8
|
||||
allow_split_before_dict_value = 0
|
1
Pipfile
1
Pipfile
@ -11,6 +11,7 @@ flask-sqlalchemy = "~=2.5.1"
|
||||
sqlalchemy = "~=1.4.18"
|
||||
waitress = "~=2.0.0"
|
||||
pyyaml = "~=5.4.1"
|
||||
flask-cors = "~=3.0.10"
|
||||
|
||||
[dev-packages]
|
||||
flake8 = "~=3.9.2"
|
||||
|
26
Pipfile.lock
generated
26
Pipfile.lock
generated
@ -1,7 +1,7 @@
|
||||
{
|
||||
"_meta": {
|
||||
"hash": {
|
||||
"sha256": "315632e3e0878039dcf4dfd73b8dc3f0285388bac45601cda9bca603b3a4ae18"
|
||||
"sha256": "11a821c6c1f072dcf7c39a020056fa289b7a5283aa33d96e2ed6860fbc023fa4"
|
||||
},
|
||||
"pipfile-spec": 6,
|
||||
"requires": {
|
||||
@ -21,7 +21,6 @@
|
||||
"sha256:a21fedebb3fb8f6bbbba51a11114f08c78709377051384c9c5ead5705ee93a51",
|
||||
"sha256:e78be5b919f5bb184e3e0e2dd1ca986f2362e29a2bc933c446fe89f39dbe4e9c"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'",
|
||||
"version": "==1.6.5"
|
||||
},
|
||||
"click": {
|
||||
@ -29,7 +28,6 @@
|
||||
"sha256:8c04c11192119b1ef78ea049e0a6f0463e4c48ef00a30160c704337586f3ad7a",
|
||||
"sha256:fba402a4a47334742d782209a7c79bc448911afe1149d07bdabdf480b3e2f4b6"
|
||||
],
|
||||
"markers": "python_version >= '3.6'",
|
||||
"version": "==8.0.1"
|
||||
},
|
||||
"flask": {
|
||||
@ -40,6 +38,14 @@
|
||||
"index": "pypi",
|
||||
"version": "==2.0.1"
|
||||
},
|
||||
"flask-cors": {
|
||||
"hashes": [
|
||||
"sha256:74efc975af1194fc7891ff5cd85b0f7478be4f7f59fe158102e91abb72bb4438",
|
||||
"sha256:b60839393f3b84a0f3746f6cdca56c1ad7426aa738b70d6c61375857823181de"
|
||||
],
|
||||
"index": "pypi",
|
||||
"version": "==3.0.10"
|
||||
},
|
||||
"flask-migrate": {
|
||||
"hashes": [
|
||||
"sha256:4d42e8f861d78cb6e9319afcba5bf76062e5efd7784184dd2a1cccd9de34a702",
|
||||
@ -116,7 +122,6 @@
|
||||
"sha256:5174094b9637652bdb841a3029700391451bd092ba3db90600dea710ba28e97c",
|
||||
"sha256:9e724d68fc22902a1435351f84c3fb8623f303fffcc566a4cb952df8c572cff0"
|
||||
],
|
||||
"markers": "python_version >= '3.6'",
|
||||
"version": "==2.0.1"
|
||||
},
|
||||
"jinja2": {
|
||||
@ -124,7 +129,6 @@
|
||||
"sha256:1f06f2da51e7b56b8f238affdd6b4e2c61e39598a378cc49345bc1bd42a978a4",
|
||||
"sha256:703f484b47a6af502e743c9122595cc812b0271f661722403114f71a79d0f5a4"
|
||||
],
|
||||
"markers": "python_version >= '3.6'",
|
||||
"version": "==3.0.1"
|
||||
},
|
||||
"mako": {
|
||||
@ -132,7 +136,6 @@
|
||||
"sha256:17831f0b7087c313c0ffae2bcbbd3c1d5ba9eeac9c38f2eb7b50e8c99fe9d5ab",
|
||||
"sha256:aea166356da44b9b830c8023cd9b557fa856bd8b4035d6de771ca027dfc5cc6e"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==1.1.4"
|
||||
},
|
||||
"markupsafe": {
|
||||
@ -172,7 +175,6 @@
|
||||
"sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51",
|
||||
"sha256:fa130dd50c57d53368c9d59395cb5526eda596d3ffe36666cd81a44d56e48872"
|
||||
],
|
||||
"markers": "python_version >= '3.6'",
|
||||
"version": "==2.0.1"
|
||||
},
|
||||
"python-dateutil": {
|
||||
@ -180,7 +182,6 @@
|
||||
"sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c",
|
||||
"sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==2.8.1"
|
||||
},
|
||||
"python-dotenv": {
|
||||
@ -241,7 +242,6 @@
|
||||
"sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926",
|
||||
"sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==1.16.0"
|
||||
},
|
||||
"sqlalchemy": {
|
||||
@ -293,7 +293,6 @@
|
||||
"sha256:1de1db30d010ff1af14a009224ec49ab2329ad2cde454c8a708130642d579c42",
|
||||
"sha256:6c1ec500dcdba0baa27600f6a22f6333d8b662d22027ff9f6202e3367413caa8"
|
||||
],
|
||||
"markers": "python_version >= '3.6'",
|
||||
"version": "==2.0.1"
|
||||
}
|
||||
},
|
||||
@ -310,7 +309,6 @@
|
||||
"sha256:9e600479b3b99e8af981ecdfc80a0296104ee610cab48a5ae4ffd0b668650eb1",
|
||||
"sha256:b449c9c6118fe8cca7fa5e00b9ec60ba08145d281d52164230a69211c5d597a1"
|
||||
],
|
||||
"markers": "python_full_version >= '3.6.1'",
|
||||
"version": "==3.3.0"
|
||||
},
|
||||
"distlib": {
|
||||
@ -340,7 +338,6 @@
|
||||
"sha256:18d0c531ee3dbc112fa6181f34faa179de3f57ea57ae2899754f16a7e0ff6421",
|
||||
"sha256:5b41f71471bc738e7b586308c3fca172f78940195cb3bf6734c1e66fdac49306"
|
||||
],
|
||||
"markers": "python_full_version >= '3.6.1'",
|
||||
"version": "==2.2.10"
|
||||
},
|
||||
"mccabe": {
|
||||
@ -370,7 +367,6 @@
|
||||
"sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068",
|
||||
"sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==2.7.0"
|
||||
},
|
||||
"pyflakes": {
|
||||
@ -378,7 +374,6 @@
|
||||
"sha256:7893783d01b8a89811dd72d7dfd4d84ff098e5eed95cfa8905b22bbffe52efc3",
|
||||
"sha256:f5bc8ecabc05bb9d291eb5203d6810b49040f6ff446a756326104746cc00c1db"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==2.3.1"
|
||||
},
|
||||
"pyyaml": {
|
||||
@ -421,7 +416,6 @@
|
||||
"sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926",
|
||||
"sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==1.16.0"
|
||||
},
|
||||
"toml": {
|
||||
@ -429,7 +423,6 @@
|
||||
"sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b",
|
||||
"sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"
|
||||
],
|
||||
"markers": "python_version >= '2.6' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==0.10.2"
|
||||
},
|
||||
"virtualenv": {
|
||||
@ -437,7 +430,6 @@
|
||||
"sha256:14fdf849f80dbb29a4eb6caa9875d476ee2a5cf76a5f5415fa2f1606010ab467",
|
||||
"sha256:2b0126166ea7c9c3661f5b8e06773d28f83322de7a3ff7d06f0aed18c9de6a76"
|
||||
],
|
||||
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||
"version": "==20.4.7"
|
||||
},
|
||||
"yapf": {
|
||||
|
26
README.md
26
README.md
@ -9,6 +9,7 @@
|
||||
- Python 3.8
|
||||
- [Pipenv](https://github.com/pypa/pipenv)
|
||||
|
||||
|
||||
### Entwicklungsumgebung aufbauen und starten
|
||||
|
||||
Ggf. vorher aufräumen
|
||||
@ -21,9 +22,8 @@ rm data/ki.sqlite
|
||||
cp env.dev .env
|
||||
pipenv install --dev
|
||||
pipenv shell
|
||||
export FLASK_APP=app.py
|
||||
flask db upgrade
|
||||
flask seed
|
||||
flask seed --dev
|
||||
flask run
|
||||
```
|
||||
|
||||
@ -40,26 +40,24 @@ pre-commit install
|
||||
```
|
||||
|
||||
|
||||
### Tests ausführen
|
||||
### `alembic` Befehle
|
||||
|
||||
`alembic` ist über [Flask-Migrate](https://flask-migrate.readthedocs.io/en/latest/index.html) eingebunden.
|
||||
Es wird über `flask db ...` aufgerufen.
|
||||
|
||||
|
||||
### QA
|
||||
|
||||
```
|
||||
python -m unittest discover ki
|
||||
```
|
||||
|
||||
# Code formatieren
|
||||
yapf -i --recursive ki/
|
||||
|
||||
### Linting
|
||||
|
||||
```
|
||||
# Code-Style prüfen
|
||||
flake8
|
||||
```
|
||||
|
||||
### Formatierung
|
||||
|
||||
Um ein einheitliches Quellcode-Erlebnis zu haben, kann der Code mit yapf neu formatiert werden:
|
||||
|
||||
```
|
||||
yapf -i --recursive ki/
|
||||
```
|
||||
|
||||
### Testbenutzer
|
||||
|
||||
|
14
app.py
14
app.py
@ -1,17 +1,27 @@
|
||||
import logging
|
||||
import os
|
||||
|
||||
from dotenv import load_dotenv, find_dotenv
|
||||
from flask import Flask
|
||||
from flask_cors import CORS
|
||||
from flask_sqlalchemy import SQLAlchemy
|
||||
from flask_migrate import Migrate
|
||||
|
||||
load_dotenv(find_dotenv())
|
||||
|
||||
loglevel = os.getenv("KI_LOGLEVEL", logging.WARNING)
|
||||
loglevel = int(loglevel)
|
||||
logging.basicConfig(level=loglevel)
|
||||
logging.debug("Hello from KI")
|
||||
|
||||
app = Flask(__name__)
|
||||
app.config['SQLALCHEMY_DATABASE_URI'] = os.getenv("SQLALCHEMY_DATABASE_URI")
|
||||
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
|
||||
app.config["SQLALCHEMY_DATABASE_URI"] = os.getenv("SQLALCHEMY_DATABASE_URI")
|
||||
app.config["SQLALCHEMY_TRACK_MODIFICATIONS"] = False
|
||||
app.config["KI_DATA_DIR"] = os.path.dirname(__file__) + "/data"
|
||||
app.config["KI_AUTH"] = os.getenv("KI_AUTH")
|
||||
app.config["CORS_ORIGINS"] = os.getenv("CORS_ORIGINS", "*")
|
||||
|
||||
CORS(app)
|
||||
db = SQLAlchemy(app)
|
||||
migrate = Migrate(app, db)
|
||||
|
||||
|
14
drone.yml
14
drone.yml
@ -1,14 +0,0 @@
|
||||
---
|
||||
kind: pipeline
|
||||
type: docker
|
||||
name: default
|
||||
|
||||
steps:
|
||||
- name: qa
|
||||
image: python3.8-alpine
|
||||
commands:
|
||||
- apk add --no-cache gcc g++ musl-dev python3-dev
|
||||
- pip3 install pipenv
|
||||
- pipenv install --system
|
||||
- flake8
|
||||
- python -m unittest discover ki
|
10
env.dev
10
env.dev
@ -1,3 +1,11 @@
|
||||
SQLALCHEMY_DATABASE_URI = 'sqlite:///data/ki.sqlite'
|
||||
SQLALCHEMY_DATABASE_URI=sqlite:///data/ki.sqlite
|
||||
|
||||
CORS_ORIGINS=*
|
||||
|
||||
FLASK_APP=app.py
|
||||
FLASK_ENV=development
|
||||
|
||||
KI_AUTH=file
|
||||
|
||||
# 10 = debug
|
||||
KI_LOGLEVEL=10
|
||||
|
1
ki/actions/__init__.py
Normal file
1
ki/actions/__init__.py
Normal file
@ -0,0 +1 @@
|
||||
from ki.actions.seed import seed # noqa
|
83
ki/actions/seed.py
Normal file
83
ki/actions/seed.py
Normal file
@ -0,0 +1,83 @@
|
||||
import csv
|
||||
import logging
|
||||
|
||||
from app import app, db
|
||||
from ki.models import Address, Contact, ContactType, Language, Skill, Profile, ProfileLanguage, ProfileSkill, User
|
||||
|
||||
|
||||
def seed(dev: bool):
|
||||
skill_seed_file_path = app.config["KI_DATA_DIR"] + "/seed_data/skills.csv"
|
||||
|
||||
logging.info("importing skills")
|
||||
|
||||
with open(skill_seed_file_path) as skills_file:
|
||||
skills_csv_reader = csv.DictReader(skills_file)
|
||||
|
||||
for skill in skills_csv_reader:
|
||||
id = int(skill["id"])
|
||||
db_skill = Skill.query.get(id)
|
||||
|
||||
if db_skill is None:
|
||||
db.session.add(Skill(id=int(skill["id"]), name=skill["name"]))
|
||||
|
||||
logging.info("importing languages")
|
||||
|
||||
iso_seed_file_path = app.config["KI_DATA_DIR"] + "/seed_data/iso_639_1.csv"
|
||||
|
||||
with open(iso_seed_file_path) as iso_file:
|
||||
iso_csv_reader = csv.DictReader(iso_file)
|
||||
|
||||
for iso in iso_csv_reader:
|
||||
id = iso["639-1"]
|
||||
db_language = Language.query.get(id)
|
||||
|
||||
if db_language is None:
|
||||
db.session.add(Language(id=iso["639-1"], name=iso["Sprache"]))
|
||||
|
||||
if dev:
|
||||
logging.info("seeding peter :)")
|
||||
|
||||
peter = User(auth_id="peter")
|
||||
db.session.add(peter)
|
||||
|
||||
peters_profile = Profile(nickname="peternichtlustig",
|
||||
pronouns="Herr Dr. Dr.",
|
||||
volunteerwork="Gartenverein",
|
||||
freetext="Ich mag Kaffee",
|
||||
user=peter)
|
||||
db.session.add(peters_profile)
|
||||
|
||||
matrix_type = ContactType(name="Matrix")
|
||||
db.session.add(matrix_type)
|
||||
|
||||
matrix_contact = Contact(profile=peters_profile,
|
||||
contacttype=matrix_type,
|
||||
content="@peter:wtf-eg.de")
|
||||
db.session.add(matrix_contact)
|
||||
|
||||
peters_address = Address(name="Peter Nichtlustig",
|
||||
street="Waldweg",
|
||||
house_number="23i",
|
||||
additional="Hinterhaus",
|
||||
postcode="13337",
|
||||
city="Bielefeld",
|
||||
country="Deutschland",
|
||||
profile=peters_profile)
|
||||
db.session.add(peters_address)
|
||||
|
||||
peters_python_skill = ProfileSkill(profile=peters_profile,
|
||||
skill_id=3,
|
||||
level=5)
|
||||
db.session.add(peters_python_skill)
|
||||
|
||||
peter_de = ProfileLanguage(profile=peters_profile,
|
||||
language_id="de",
|
||||
level=5)
|
||||
db.session.add(peter_de)
|
||||
|
||||
peter_fr = ProfileLanguage(profile=peters_profile,
|
||||
language_id="fr",
|
||||
level=3)
|
||||
db.session.add(peter_fr)
|
||||
|
||||
db.session.commit()
|
@ -1,39 +1,10 @@
|
||||
import csv
|
||||
import click
|
||||
|
||||
from ki.models import Language, Skill
|
||||
from app import app, db
|
||||
|
||||
|
||||
def seed():
|
||||
skill_seed_file_path = app.config["KI_DATA_DIR"] + "/seed_data/skills.csv"
|
||||
|
||||
print("importing skills")
|
||||
|
||||
with open(skill_seed_file_path) as skills_file:
|
||||
skills_csv_reader = csv.DictReader(skills_file)
|
||||
|
||||
for skill in skills_csv_reader:
|
||||
id = int(skill["id"])
|
||||
db_skill = Skill.query.get(id)
|
||||
|
||||
if db_skill is None:
|
||||
db.session.add(Skill(id=int(skill["id"]), name=skill["name"]))
|
||||
|
||||
iso_seed_file_path = app.config["KI_DATA_DIR"] + "/seed_data/iso_639_1.csv"
|
||||
|
||||
with open(iso_seed_file_path) as iso_file:
|
||||
iso_csv_reader = csv.DictReader(iso_file)
|
||||
|
||||
for iso in iso_csv_reader:
|
||||
id = iso["639-1"]
|
||||
db_language = Language.query.get(id)
|
||||
|
||||
if db_language is None:
|
||||
db.session.add(Language(id=iso["639-1"], name=iso["Sprache"]))
|
||||
|
||||
db.session.commit()
|
||||
from app import app
|
||||
from ki.actions import seed
|
||||
|
||||
|
||||
@app.cli.command("seed")
|
||||
def seed_command():
|
||||
seed()
|
||||
@click.option("--dev", is_flag=True)
|
||||
def seed_command(dev):
|
||||
seed(dev)
|
||||
|
64
ki/models.py
64
ki/models.py
@ -13,7 +13,7 @@ class User(db.Model):
|
||||
auth_id = Column(String(50), nullable=False, unique=True)
|
||||
profile_id = Column(Integer, ForeignKey("profile.id"), nullable=True)
|
||||
|
||||
tokens = relationship("Token", uselist=False, back_populates="user")
|
||||
tokens = relationship("Token", back_populates="user")
|
||||
profile = relationship("Profile", back_populates="user")
|
||||
|
||||
def to_dict(self):
|
||||
@ -42,10 +42,17 @@ class Profile(db.Model):
|
||||
|
||||
def to_dict(self):
|
||||
return {
|
||||
"user_id": self.user.id,
|
||||
"nickname": self.nickname,
|
||||
"pronouns": self.pronouns,
|
||||
"volunteerwork": self.volunteerwork,
|
||||
"freetext": self.freetext
|
||||
"freetext": self.freetext,
|
||||
"address": self.address.to_dict(),
|
||||
"contacts": list(
|
||||
map(lambda contact: contact.to_dict(), self.contacts)),
|
||||
"skills": list(map(lambda skill: skill.to_dict(), self.skills)),
|
||||
"languages": list(
|
||||
map(lambda language: language.to_dict(), self.languages))
|
||||
}
|
||||
|
||||
|
||||
@ -58,6 +65,9 @@ class Token(db.Model):
|
||||
|
||||
user = relationship("User", back_populates="tokens")
|
||||
|
||||
def to_dict(self):
|
||||
return {"user_id": self.user_id, "token": self.token}
|
||||
|
||||
|
||||
class Contact(db.Model):
|
||||
__tablename__ = "contact"
|
||||
@ -71,6 +81,14 @@ class Contact(db.Model):
|
||||
contacttype = relationship("ContactType")
|
||||
content = Column(String(200), nullable=False)
|
||||
|
||||
def to_dict(self):
|
||||
return {
|
||||
"id": self.id,
|
||||
"profile_id": self.profile_id,
|
||||
"contacttype": self.contacttype.to_dict(),
|
||||
"content": self.content
|
||||
}
|
||||
|
||||
|
||||
class ContactType(db.Model):
|
||||
__tablename__ = "contacttype"
|
||||
@ -78,6 +96,9 @@ class ContactType(db.Model):
|
||||
id = Column(Integer, primary_key=True)
|
||||
name = Column(String(25), nullable=False)
|
||||
|
||||
def to_dict(self):
|
||||
return {"id": self.id, "name": self.name}
|
||||
|
||||
|
||||
class Address(db.Model):
|
||||
__tablename__ = "address"
|
||||
@ -94,6 +115,19 @@ class Address(db.Model):
|
||||
profile_id = Column(Integer, ForeignKey("profile.id"), nullable=False)
|
||||
profile = relationship("Profile", back_populates="address")
|
||||
|
||||
def to_dict(self):
|
||||
return {
|
||||
"id": self.id,
|
||||
"name": self.name,
|
||||
"street": self.street,
|
||||
"house_number": self.house_number,
|
||||
"additional": self.additional,
|
||||
"postcode": self.postcode,
|
||||
"city": self.city,
|
||||
"country": self.country,
|
||||
"profile_id": self.profile_id
|
||||
}
|
||||
|
||||
|
||||
class Skill(db.Model):
|
||||
__tablename__ = "skill"
|
||||
@ -104,7 +138,11 @@ class Skill(db.Model):
|
||||
profiles = relationship("ProfileSkill", back_populates="skill")
|
||||
|
||||
def to_dict(self):
|
||||
return {"id": self.id, "name": self.name}
|
||||
return {
|
||||
"id": self.id,
|
||||
"name": self.name,
|
||||
"icon_url": "/skills/{}/icon".format(self.id)
|
||||
}
|
||||
|
||||
|
||||
class ProfileSkill(db.Model):
|
||||
@ -117,6 +155,13 @@ class ProfileSkill(db.Model):
|
||||
profile = relationship("Profile", back_populates="skills")
|
||||
skill = relationship("Skill", back_populates="profiles")
|
||||
|
||||
def to_dict(self):
|
||||
return {
|
||||
"profile_id": self.profile_id,
|
||||
"skill": self.skill.to_dict(),
|
||||
"level": self.level
|
||||
}
|
||||
|
||||
|
||||
class Language(db.Model):
|
||||
__tablename__ = "language"
|
||||
@ -127,7 +172,11 @@ class Language(db.Model):
|
||||
profiles = relationship("ProfileLanguage", back_populates="language")
|
||||
|
||||
def to_dict(self):
|
||||
return {"id": self.id, "name": self.name}
|
||||
return {
|
||||
"id": self.id,
|
||||
"name": self.name,
|
||||
"icon_url": "/languages/{}/icon".format(self.id)
|
||||
}
|
||||
|
||||
|
||||
class ProfileLanguage(db.Model):
|
||||
@ -139,3 +188,10 @@ class ProfileLanguage(db.Model):
|
||||
|
||||
profile = relationship("Profile", back_populates="languages")
|
||||
language = relationship("Language", back_populates="profiles")
|
||||
|
||||
def to_dict(self):
|
||||
return {
|
||||
"profile_id": self.profile_id,
|
||||
"language": self.language.to_dict(),
|
||||
"level": self.level
|
||||
}
|
||||
|
@ -115,7 +115,9 @@ def get_user_profile(user_id):
|
||||
if profile is None:
|
||||
return make_response({}, 404)
|
||||
|
||||
return make_response({"profile": profile.to_dict()})
|
||||
return make_response({
|
||||
"profile": profile.to_dict(),
|
||||
})
|
||||
|
||||
|
||||
@app.route("/users/<user_id>/profile", methods=["POST"])
|
||||
|
43
ki/test/test_login_endpoint.py
Normal file
43
ki/test/test_login_endpoint.py
Normal file
@ -0,0 +1,43 @@
|
||||
from alembic import command
|
||||
import json
|
||||
import unittest
|
||||
|
||||
from app import app, db, migrate
|
||||
from ki.actions import seed
|
||||
|
||||
|
||||
class TestLoginEndpoint(unittest.TestCase):
|
||||
def setUp(self):
|
||||
app.debug = True
|
||||
app.config["SQLALCHEMY_DATABASE_URI"] = "sqlite:///:memory:"
|
||||
self.client = app.test_client()
|
||||
|
||||
with app.app_context():
|
||||
config = migrate.get_config()
|
||||
command.upgrade(config, "head")
|
||||
|
||||
seed(True)
|
||||
|
||||
def tearDown(self):
|
||||
db.drop_all()
|
||||
db.engine.dispose()
|
||||
|
||||
def test_login(self):
|
||||
response1_data = self.login()
|
||||
response2_data = self.login()
|
||||
self.assertNotEqual(response1_data["token"], response2_data["token"])
|
||||
|
||||
def login(self):
|
||||
response = self.client.post("/users/login",
|
||||
data=json.dumps({
|
||||
"username": "peter",
|
||||
"password": "geheim"
|
||||
}),
|
||||
content_type="application/json")
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertIn("token", response.json)
|
||||
return response.json
|
||||
|
||||
|
||||
if __name__ == "main":
|
||||
unittest.main()
|
@ -3,11 +3,13 @@ import unittest
|
||||
import json
|
||||
|
||||
from app import app, db, migrate
|
||||
from ki.commands import seed
|
||||
from ki.models import Profile, User
|
||||
from ki.actions import seed
|
||||
from ki.models import User
|
||||
|
||||
|
||||
class TestProfileEndpoint(unittest.TestCase):
|
||||
maxDiff = None
|
||||
|
||||
def setUp(self):
|
||||
app.debug = True
|
||||
app.config["TESTING"] = True
|
||||
@ -18,16 +20,13 @@ class TestProfileEndpoint(unittest.TestCase):
|
||||
config = migrate.get_config()
|
||||
command.upgrade(config, "head")
|
||||
|
||||
seed()
|
||||
seed(True)
|
||||
|
||||
def tearDown(self):
|
||||
db.drop_all()
|
||||
db.engine.dispose()
|
||||
|
||||
def test_create_profile(self):
|
||||
user = User(auth_id="peter")
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
|
||||
def test_update_profile(self):
|
||||
login_data = {"username": "peter", "password": "geheim"}
|
||||
login_response = self.client.post("/users/login",
|
||||
data=json.dumps(login_data),
|
||||
@ -37,7 +36,7 @@ class TestProfileEndpoint(unittest.TestCase):
|
||||
self.assertIn("token", login_response.json)
|
||||
|
||||
data = {
|
||||
"pronouns": "Herr Dr. Dr.",
|
||||
"pronouns": "Monsieur",
|
||||
"volunteerwork": "ja",
|
||||
"freetext": "Hallo",
|
||||
}
|
||||
@ -45,8 +44,7 @@ class TestProfileEndpoint(unittest.TestCase):
|
||||
data=json.dumps(data),
|
||||
content_type="application/json",
|
||||
headers={
|
||||
"Authorization":
|
||||
"Bearer " +
|
||||
"Authorization": "Bearer " +
|
||||
login_response.json["token"]
|
||||
})
|
||||
|
||||
@ -54,20 +52,11 @@ class TestProfileEndpoint(unittest.TestCase):
|
||||
with app.app_context():
|
||||
user = User.query.filter(User.id == 1).first()
|
||||
profile = user.profile
|
||||
self.assertEqual("Herr Dr. Dr.", profile.pronouns)
|
||||
self.assertEqual("Monsieur", profile.pronouns)
|
||||
self.assertEqual("ja", profile.volunteerwork)
|
||||
self.assertEqual("Hallo", profile.freetext)
|
||||
|
||||
def test_get_profile(self):
|
||||
user = User(auth_id="peter")
|
||||
db.session.add(user)
|
||||
|
||||
profile = Profile(user=user)
|
||||
profile.nickname = "Popeter"
|
||||
db.session.add(profile)
|
||||
|
||||
db.session.commit()
|
||||
|
||||
login_data = {"username": "peter", "password": "geheim"}
|
||||
login_response = self.client.post("/users/login",
|
||||
data=json.dumps(login_data),
|
||||
@ -78,18 +67,65 @@ class TestProfileEndpoint(unittest.TestCase):
|
||||
|
||||
response = self.client.get("/users/1/profile",
|
||||
headers={
|
||||
"Authorization":
|
||||
"Bearer " + login_response.json["token"]
|
||||
"Authorization": "Bearer " +
|
||||
login_response.json["token"]
|
||||
})
|
||||
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertEqual(
|
||||
self.assertDictEqual(
|
||||
response.json, {
|
||||
"profile": {
|
||||
"freetext": "",
|
||||
"nickname": "Popeter",
|
||||
"pronouns": "",
|
||||
"volunteerwork": ""
|
||||
"user_id": 1,
|
||||
"nickname": "peternichtlustig",
|
||||
"pronouns": "Herr Dr. Dr.",
|
||||
"freetext": "Ich mag Kaffee",
|
||||
"volunteerwork": "Gartenverein",
|
||||
"address": {
|
||||
"additional": "Hinterhaus",
|
||||
"city": "Bielefeld",
|
||||
"country": "Deutschland",
|
||||
"house_number": "23i",
|
||||
"id": 1,
|
||||
"name": "Peter Nichtlustig",
|
||||
"postcode": "13337",
|
||||
"profile_id": 1,
|
||||
"street": "Waldweg"
|
||||
},
|
||||
"contacts": [{
|
||||
"id": 1,
|
||||
"profile_id": 1,
|
||||
"contacttype": {
|
||||
"id": 1,
|
||||
"name": "Matrix"
|
||||
},
|
||||
"content": "@peter:wtf-eg.de"
|
||||
}],
|
||||
"skills": [{
|
||||
"profile_id": 1,
|
||||
"skill": {
|
||||
"id": 3,
|
||||
"name": "Python",
|
||||
"icon_url": "/skills/3/icon"
|
||||
},
|
||||
"level": 5
|
||||
}],
|
||||
"languages": [{
|
||||
"profile_id": 1,
|
||||
"language": {
|
||||
"id": "de",
|
||||
"name": "Deutsch",
|
||||
"icon_url": "/languages/de/icon"
|
||||
},
|
||||
"level": 5
|
||||
}, {
|
||||
"profile_id": 1,
|
||||
"language": {
|
||||
"id": "fr",
|
||||
"name": "Französisch",
|
||||
"icon_url": "/languages/fr/icon"
|
||||
},
|
||||
"level": 3
|
||||
}]
|
||||
}
|
||||
})
|
||||
|
||||
|
@ -1,8 +1,8 @@
|
||||
from alembic import command
|
||||
import unittest
|
||||
|
||||
from app import app, migrate
|
||||
from ki.commands import seed
|
||||
from app import app, db, migrate
|
||||
from ki.actions import seed
|
||||
|
||||
|
||||
class TestSkillsEndpoint(unittest.TestCase):
|
||||
@ -15,7 +15,17 @@ class TestSkillsEndpoint(unittest.TestCase):
|
||||
config = migrate.get_config()
|
||||
command.upgrade(config, "head")
|
||||
|
||||
seed()
|
||||
seed(True)
|
||||
|
||||
def tearDown(self):
|
||||
db.drop_all()
|
||||
db.engine.dispose()
|
||||
|
||||
def test_skills_options(self):
|
||||
response = self.client.options("/skills")
|
||||
self.assertEqual(response.status_code, 200)
|
||||
self.assertIn("Access-Control-Allow-Origin", response.headers)
|
||||
self.assertEqual(response.headers["Access-Control-Allow-Origin"], "*")
|
||||
|
||||
def test_get_skills1(self):
|
||||
response = self.client.get("/skills?search=p")
|
||||
@ -24,12 +34,16 @@ class TestSkillsEndpoint(unittest.TestCase):
|
||||
{
|
||||
"skills": [{
|
||||
"id": 1,
|
||||
"name": "PHP"
|
||||
"name": "PHP",
|
||||
"icon_url": "/skills/1/icon"
|
||||
}, {
|
||||
"id": 3,
|
||||
"name": "Python"
|
||||
"name": "Python",
|
||||
"icon_url": "/skills/3/icon"
|
||||
}]
|
||||
}, response.json)
|
||||
self.assertIn("Access-Control-Allow-Origin", response.headers)
|
||||
self.assertEqual(response.headers["Access-Control-Allow-Origin"], "*")
|
||||
|
||||
|
||||
if __name__ == "main":
|
||||
|
Reference in New Issue
Block a user