Merge branch 'testing' into 'master'
Add screen, flights, auth and browser domains See merge request adm3981141/fids!1
This commit is contained in:
commit
0b8f16efa2
|
@ -1,3 +1,6 @@
|
|||
.venv
|
||||
.env
|
||||
.env.*
|
||||
!.env.dev.example
|
||||
!.env.prod.example
|
||||
node_modules
|
206
.gitlab-ci.yml
206
.gitlab-ci.yml
|
@ -19,13 +19,22 @@ preparation:
|
|||
- export BUILD_ID=$(date +%Y%m%d%H%M)
|
||||
- echo "BUILD_ID=${BUILD_ID}" > context.env
|
||||
|
||||
- echo "API_PROD_IMAGE_NAME=${IMAGE_BASE}/api:prod-${BUILD_ID}" >> context.env
|
||||
- echo "API_TEST_IMAGE_NAME=${IMAGE_BASE}/api:test-${BUILD_ID}" >> context.env
|
||||
- echo "CLIENT_PROD_IMAGE_NAME=${IMAGE_BASE}/client:prod-${BUILD_ID}" >> context.env
|
||||
- echo "CLIENT_TEST_IMAGE_NAME=${IMAGE_BASE}/client:test-${BUILD_ID}" >> context.env
|
||||
- echo "FLIGHTS_INFO_PROD_IMAGE_NAME=${IMAGE_BASE}/flights-information:prod-${BUILD_ID}" >> context.env
|
||||
- echo "FLIGHTS_INFO_TEST_IMAGE_NAME=${IMAGE_BASE}/flights-information:test-${BUILD_ID}" >> context.env
|
||||
|
||||
- echo "DOCKER_HUB_API_IMAGE=$DOCKER_HUB_USER/fids-api:${BUILD_ID}" >> context.env
|
||||
- echo "DOCKER_HUB_CLIENT_IMAGE=$DOCKER_HUB_USER/fids-client:${BUILD_ID}" >> context.env
|
||||
- echo "USER_MANAGER_PROD_IMAGE_NAME=${IMAGE_BASE}/user-manager:prod-${BUILD_ID}" >> context.env
|
||||
- echo "USER_MANAGER_TEST_IMAGE_NAME=${IMAGE_BASE}/user-manager:test-${BUILD_ID}" >> context.env
|
||||
|
||||
- echo "SCREEN_CLIENT_PROD_IMAGE_NAME=${IMAGE_BASE}/screens-client:prod-${BUILD_ID}" >> context.env
|
||||
- echo "SCREEN_CLIENT_TEST_IMAGE_NAME=${IMAGE_BASE}/screens-client:test-${BUILD_ID}" >> context.env
|
||||
|
||||
- echo "BROWSER_CLIENT_PROD_IMAGE_NAME=${IMAGE_BASE}/browser-client:prod-${BUILD_ID}" >> context.env
|
||||
- echo "BROWSER_CLIENT_TEST_IMAGE_NAME=${IMAGE_BASE}/browser-client:test-${BUILD_ID}" >> context.env
|
||||
|
||||
- echo "DOCKER_HUB_SCREEN_CLIENT_IMAGE=$DOCKER_HUB_USER/screens-client:${BUILD_ID}" >> context.env
|
||||
- echo "DOCKER_HUB_BROWSER_CLIENT_IMAGE=$DOCKER_HUB_USER/browser-client:${BUILD_ID}" >> context.env
|
||||
- echo "DOCKER_HUB_USER_MANAGER_IMAGE=$DOCKER_HUB_USER/user-manager:${BUILD_ID}" >> context.env
|
||||
- echo "DOCKER_HUB_FLIGHT_INFO_IMAGE=$DOCKER_HUB_USER/flights-information:${BUILD_ID}" >> context.env
|
||||
|
||||
- echo "ENV_DEV_FILE=$(echo $ENV_DEV)" >> context.env
|
||||
- echo "ENV_PROD_FILE=$(echo $ENV_PROD)" >> context.env
|
||||
|
@ -33,58 +42,97 @@ preparation:
|
|||
paths:
|
||||
- context.env
|
||||
|
||||
build-api:
|
||||
build-auth-api:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- docker build flights-domain -f flights-domain/Dockerfile.prod -t ${API_PROD_IMAGE_NAME}
|
||||
- docker build flights-domain -f flights-domain/Dockerfile.test --build-arg "BASE_IMAGE=$API_PROD_IMAGE_NAME" -t ${API_TEST_IMAGE_NAME}
|
||||
- export USER_MANAGER=auth-domain/user-manager
|
||||
- docker build $USER_MANAGER -f $USER_MANAGER/Dockerfile.prod -t ${USER_MANAGER_PROD_IMAGE_NAME}
|
||||
- docker build $USER_MANAGER -f $USER_MANAGER/Dockerfile.test --build-arg "BASE_IMAGE=$USER_MANAGER_PROD_IMAGE_NAME" -t ${USER_MANAGER_TEST_IMAGE_NAME}
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker push ${API_PROD_IMAGE_NAME}
|
||||
- docker push ${API_TEST_IMAGE_NAME}
|
||||
- docker push ${USER_MANAGER_PROD_IMAGE_NAME}
|
||||
- docker push ${USER_MANAGER_TEST_IMAGE_NAME}
|
||||
needs:
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
build-client:
|
||||
build-flights-api:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- docker build sample-client-users -f sample-client-users/Dockerfile.prod -t ${CLIENT_PROD_IMAGE_NAME}
|
||||
- docker build sample-client-users -f sample-client-users/Dockerfile.test -t ${CLIENT_TEST_IMAGE_NAME}
|
||||
- export FLIGHTS_INFORMATION=flights-domain/flights-information
|
||||
- docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.prod -t ${FLIGHTS_INFO_PROD_IMAGE_NAME}
|
||||
- docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.test --build-arg "BASE_IMAGE=$FLIGHTS_INFO_PROD_IMAGE_NAME" -t ${FLIGHTS_INFO_TEST_IMAGE_NAME}
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker push ${CLIENT_PROD_IMAGE_NAME}
|
||||
- docker push ${CLIENT_TEST_IMAGE_NAME}
|
||||
- docker push ${FLIGHTS_INFO_PROD_IMAGE_NAME}
|
||||
- docker push ${FLIGHTS_INFO_TEST_IMAGE_NAME}
|
||||
needs:
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
test-api:
|
||||
build-browser-client:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- docker build browser-domain -f browser-domain/Dockerfile.prod -t ${BROWSER_CLIENT_PROD_IMAGE_NAME}
|
||||
- docker build browser-domain -f browser-domain/Dockerfile.test -t ${BROWSER_CLIENT_TEST_IMAGE_NAME}
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker push ${BROWSER_CLIENT_PROD_IMAGE_NAME}
|
||||
- docker push ${BROWSER_CLIENT_TEST_IMAGE_NAME}
|
||||
needs:
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
build-screen-client:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- docker build screen-domain -f screen-domain/Dockerfile.prod -t ${SCREEN_CLIENT_PROD_IMAGE_NAME}
|
||||
- docker build screen-domain -f screen-domain/Dockerfile.test -t ${SCREEN_CLIENT_TEST_IMAGE_NAME}
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker push ${SCREEN_CLIENT_PROD_IMAGE_NAME}
|
||||
- docker push ${SCREEN_CLIENT_TEST_IMAGE_NAME}
|
||||
needs:
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
test-auth-api:
|
||||
stage: test
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- export API_IMAGE=$API_TEST_IMAGE_NAME
|
||||
- export API_IMAGE=$USER_MANAGER_TEST_IMAGE_NAME
|
||||
- export CLIENT_IMAGE=dummy-image
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_DEV_FILE --profile api pull
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_DEV_FILE --profile api up --abort-on-container-exit --renew-anon-volumes
|
||||
- docker cp fids_api:/usr/src/app/coverage.xml .
|
||||
- docker cp fids_api:/usr/src/app/report.xml .
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit --renew-anon-volumes
|
||||
- docker cp fids_usermanager_api:/usr/src/app/coverage.xml .
|
||||
- docker cp fids_usermanager_api:/usr/src/app/report.xml .
|
||||
artifacts:
|
||||
when: always
|
||||
paths:
|
||||
|
@ -94,7 +142,36 @@ test-api:
|
|||
junit: report.xml
|
||||
needs:
|
||||
- job: preparation
|
||||
- job: build-api
|
||||
- job: build-auth-api
|
||||
artifacts: true
|
||||
|
||||
test-flights-api:
|
||||
stage: test
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- export API_IMAGE=$FLIGHTS_INFO_TEST_IMAGE_NAME
|
||||
- export CLIENT_IMAGE=dummy-image
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit --renew-anon-volumes
|
||||
- docker cp fids_flights_api:/usr/src/app/coverage.xml .
|
||||
- docker cp fids_flights_api:/usr/src/app/report.xml .
|
||||
artifacts:
|
||||
when: always
|
||||
paths:
|
||||
- coverage.xml
|
||||
- report.xml
|
||||
reports:
|
||||
junit: report.xml
|
||||
needs:
|
||||
- job: preparation
|
||||
- job: build-flights-api
|
||||
artifacts: true
|
||||
|
||||
test-integration:
|
||||
|
@ -105,16 +182,58 @@ test-integration:
|
|||
- export $(cat context.env | xargs)
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- export API_IMAGE=$API_TEST_IMAGE_NAME
|
||||
- export CLIENT_IMAGE=$CLIENT_TEST_IMAGE_NAME
|
||||
- export API_IMAGE=$FLIGHTS_INFO_TEST_IMAGE_NAME
|
||||
- export TEST_TARGET=INTEGRATION
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_DEV_FILE --profile all pull
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_DEV_FILE --profile all up --abort-on-container-exit
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE up -d
|
||||
|
||||
- export API_IMAGE=$USER_MANAGER_TEST_IMAGE_NAME
|
||||
- export TEST_TARGET=INTEGRATION
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE up -d
|
||||
needs:
|
||||
- job: test-api
|
||||
- job: build-client
|
||||
- job: test-flights-api
|
||||
- job: test-auth-api
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
artifacts: true
|
||||
|
||||
test-browser-integration:
|
||||
stage: test
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- export CLIENT_IMAGE=$BROWSER_CLIENT_TEST_IMAGE_NAME
|
||||
- docker compose -f browser-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f browser-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f browser-domain/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit
|
||||
needs:
|
||||
- job: test-integration
|
||||
- job: build-browser-client
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
test-screen-integration:
|
||||
stage: test
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- export CLIENT_IMAGE=$SCREEN_CLIENT_TEST_IMAGE_NAME
|
||||
- docker compose -f screen-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f screen-domain/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f screen-domain/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit
|
||||
needs:
|
||||
- job: test-integration
|
||||
- job: build-screen-client
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
deliver-dockerhub:
|
||||
stage: deliver
|
||||
|
@ -126,13 +245,18 @@ deliver-dockerhub:
|
|||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
- docker login -u $DOCKER_HUB_USER --password $DOCKER_HUB_PASS
|
||||
|
||||
- docker tag $API_PROD_IMAGE_NAME $DOCKER_HUB_API_IMAGE
|
||||
- docker tag $CLIENT_PROD_IMAGE_NAME $DOCKER_HUB_CLIENT_IMAGE
|
||||
- docker tag $FLIGHTS_INFO_PROD_IMAGE_NAME $DOCKER_HUB_FLIGHT_INFO_IMAGE
|
||||
- docker tag $USER_MANAGER_PROD_IMAGE_NAME $DOCKER_HUB_USER_MANAGER_IMAGE
|
||||
- docker tag $BROWSER_CLIENT_PROD_IMAGE_NAME $DOCKER_HUB_BROWSER_CLIENT_IMAGE
|
||||
- docker tag $SCREEN_CLIENT_PROD_IMAGE_NAME $DOCKER_HUB_SCREEN_CLIENT_IMAGE
|
||||
|
||||
- docker push $DOCKER_HUB_API_IMAGE
|
||||
- docker push $DOCKER_HUB_CLIENT_IMAGE
|
||||
- docker push $DOCKER_HUB_FLIGHT_INFO_IMAGE
|
||||
- docker push $DOCKER_HUB_USER_MANAGER_IMAGE
|
||||
- docker push $DOCKER_HUB_BROWSER_CLIENT_IMAGE
|
||||
- docker push $DOCKER_HUB_SCREEN_CLIENT_IMAGE
|
||||
needs:
|
||||
- job: test-integration
|
||||
- job: test-screen-integration
|
||||
- job: test-browser-integration
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
|
@ -140,6 +264,10 @@ deploy-prod:
|
|||
stage: deploy
|
||||
tags:
|
||||
- prod
|
||||
rules:
|
||||
- when: never
|
||||
- if: $CI_COMMIT_REF_NAME == "master"
|
||||
when: on_success
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
|
@ -148,10 +276,10 @@ deploy-prod:
|
|||
|
||||
- docker login -u $DOCKER_HUB_USER --password $DOCKER_HUB_PASS
|
||||
|
||||
- docker compose -f docker-compose.yml --profile all --env-file $ENV_PROD_FILE stop
|
||||
- docker compose -f docker-compose.yml --profile all --env-file $ENV_PROD_FILE rm -f
|
||||
- docker compose -f docker-compose.yml --profile all --env-file $ENV_PROD_FILE pull
|
||||
- docker compose -f docker-compose.yml --profile all --env-file $ENV_PROD_FILE up -d
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_PROD_FILE stop
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_PROD_FILE rm -f
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_PROD_FILE pull
|
||||
- docker compose -f docker-compose.yml --env-file $ENV_PROD_FILE up -d
|
||||
needs:
|
||||
- job: deliver-dockerhub
|
||||
- job: preparation
|
||||
|
|
|
@ -8,9 +8,9 @@ repos:
|
|||
rev: 6.1.0
|
||||
hooks:
|
||||
- id: flake8
|
||||
args: [--config, flights-domain/setup.cfg]
|
||||
args: [--config, flights-domain/flights-information/setup.cfg]
|
||||
- repo: https://github.com/pycqa/isort
|
||||
rev: 5.12.0
|
||||
hooks:
|
||||
- id: isort
|
||||
args: ['--src-path', 'flights-domain/']
|
||||
args: ['--src-path', 'flights-domain/flights-information/src', 'auth-domain/user-manager/src']
|
||||
|
|
38
README.md
38
README.md
|
@ -1,10 +1,40 @@
|
|||
# fids
|
||||
|
||||
## Contributing
|
||||
## Componentes
|
||||
|
||||
### auth-domain
|
||||
|
||||
Contiene `user-manager` con su base de datos. Maneja la autenticación y autorización de usuarios para el `browser-domain`.
|
||||
|
||||
### browser-domain
|
||||
|
||||
SPA que tiene dos flujos dependiendo si el usuario es una aerolínea o un usuario normal.
|
||||
|
||||
### flights-domain
|
||||
|
||||
Contiene `flights-information` con su base de datos. Maneja todo lo relacionado a la información de los vuelos (CRUD).
|
||||
|
||||
### screens-domain
|
||||
|
||||
PWA pensado para utilizarse en un aeropuerto. Se maneja con un solo `origin` y con el query param `lastUpdated` para pedir cambios. Esta tiene una base datos para cachear los resultados y poder funcionar offline.
|
||||
|
||||
## Uso
|
||||
|
||||
Primero, deberá configurar los `.env` como usted prefiera. Copie y modifique los ejemplos:
|
||||
|
||||
```
|
||||
cp flights-domain/.env.prod.example flights-domain/.env.prod
|
||||
cp user-domain/.env.prod.example user-domain/.env.prod
|
||||
```
|
||||
|
||||
Luego, para levantar todos los componentes, basta con ejecutar:
|
||||
|
||||
```
|
||||
./run.sh
|
||||
```
|
||||
|
||||
## Contribuir
|
||||
|
||||
```
|
||||
pre-commit install
|
||||
pre-commit run --all-files
|
||||
```
|
||||
|
||||
|
||||
|
|
|
@ -0,0 +1,4 @@
|
|||
POSTGRES_USER=user
|
||||
POSTGRES_PASS=password
|
||||
POSTGRES_DB=api_dev
|
||||
APP_SETTINGS=src.config.DevelopmentConfig
|
|
@ -0,0 +1,4 @@
|
|||
POSTGRES_USER=user
|
||||
POSTGRES_PASS=password
|
||||
POSTGRES_DB=api_prod
|
||||
APP_SETTINGS=src.config.ProductionConfig
|
|
@ -0,0 +1,39 @@
|
|||
version: '3.8'
|
||||
|
||||
services:
|
||||
usermanager-api:
|
||||
container_name: fids_usermanager_api
|
||||
image: ${API_IMAGE}
|
||||
ports:
|
||||
- 5001:5000
|
||||
healthcheck:
|
||||
test: ["CMD", "nc", "-vz", "-w1", "localhost", "5000"]
|
||||
interval: 2s
|
||||
timeout: 2s
|
||||
retries: 5
|
||||
start_period: 2s
|
||||
environment:
|
||||
- TEST_TARGET=${TEST_TARGET}
|
||||
- PORT=5000
|
||||
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASS}@usermanager-db/${POSTGRES_DB}
|
||||
- APP_SETTINGS=${APP_SETTINGS}
|
||||
depends_on:
|
||||
usermanager-db:
|
||||
condition: service_healthy
|
||||
|
||||
usermanager-db:
|
||||
container_name: fids_usermanager_db
|
||||
build:
|
||||
context: ./db
|
||||
dockerfile: Dockerfile
|
||||
healthcheck:
|
||||
test: psql postgres --command "select 1" -U ${POSTGRES_USER}
|
||||
interval: 2s
|
||||
timeout: 10s
|
||||
retries: 10
|
||||
start_period: 2s
|
||||
expose:
|
||||
- 5432
|
||||
environment:
|
||||
- POSTGRES_USER=${POSTGRES_USER}
|
||||
- POSTGRES_PASSWORD=${POSTGRES_PASS}
|
|
@ -11,6 +11,8 @@ ENV FLASK_DEBUG 0
|
|||
ENV FLASK_ENV production
|
||||
ARG SECRET_KEY
|
||||
ENV SECRET_KEY $SECRET_KEY
|
||||
ARG PORT
|
||||
ENV PORT $PORT
|
||||
|
||||
RUN apt-get update \
|
||||
&& apt-get -y install netcat gcc postgresql \
|
|
@ -4,7 +4,7 @@ FROM ${BASE_IMAGE}
|
|||
|
||||
ENV FLASK_DEBUG=1
|
||||
ENV FLASK_ENV=development
|
||||
ENV DATABASE_TEST_URL=postgresql://postgres:postgres@api-db:5432/api_test
|
||||
ENV DATABASE_TEST_URL=postgresql://user:password@usermanager-db:5432/api_test
|
||||
|
||||
# add and install requirements
|
||||
COPY --chown=python:python ./requirements.test.txt .
|
|
@ -5,10 +5,10 @@ name = "pypi"
|
|||
|
||||
[packages]
|
||||
flask = "==2.2.3"
|
||||
Werkzeug = "==2.3.7"
|
||||
flask-restx = "==1.0.6"
|
||||
flask-sqlalchemy = "==3.0.3"
|
||||
psycopg2-binary = "==2.9.5"
|
||||
Werkzeug = "==2.3.7"
|
||||
|
||||
[dev-packages]
|
||||
|
|
@ -0,0 +1,454 @@
|
|||
<?xml version="1.0" ?>
|
||||
<coverage version="7.3.2" timestamp="1698081228733" lines-valid="374" lines-covered="173" line-rate="0.4626" branches-valid="148" branches-covered="97" branch-rate="0.6554" complexity="0">
|
||||
<!-- Generated by coverage.py: https://coverage.readthedocs.io/en/7.3.2 -->
|
||||
<!-- Based on https://raw.githubusercontent.com/cobertura/web/master/htdocs/xml/coverage-04.dtd -->
|
||||
<sources>
|
||||
<source>/home/slococo/ITBA/MICRO/fids/auth-domain/user-manager/src</source>
|
||||
</sources>
|
||||
<packages>
|
||||
<package name="." line-rate="0.8478" branch-rate="0.25" complexity="0">
|
||||
<classes>
|
||||
<class name="__init__.py" filename="__init__.py" complexity="0" line-rate="0.65" branch-rate="0">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="3" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="6" hits="1"/>
|
||||
<line number="9" hits="1"/>
|
||||
<line number="10" hits="1"/>
|
||||
<line number="11" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="16" hits="1"/>
|
||||
<line number="19" hits="1"/>
|
||||
<line number="20" hits="1"/>
|
||||
<line number="23" hits="1"/>
|
||||
<line number="24" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="29" hits="0"/>
|
||||
<line number="32" hits="0"/>
|
||||
<line number="33" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="32,36"/>
|
||||
<line number="34" hits="0"/>
|
||||
<line number="36" hits="0"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="config.py" filename="config.py" complexity="0" line-rate="1" branch-rate="0.5">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="6" hits="1"/>
|
||||
<line number="7" hits="1"/>
|
||||
<line number="8" hits="1"/>
|
||||
<line number="9" hits="1"/>
|
||||
<line number="12" hits="1"/>
|
||||
<line number="13" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="17" hits="1"/>
|
||||
<line number="18" hits="1"/>
|
||||
<line number="19" hits="1"/>
|
||||
<line number="20" hits="1"/>
|
||||
<line number="21" hits="1"/>
|
||||
<line number="22" hits="1"/>
|
||||
<line number="25" hits="1"/>
|
||||
<line number="26" hits="1"/>
|
||||
<line number="27" hits="1"/>
|
||||
<line number="28" hits="1"/>
|
||||
<line number="30" hits="1"/>
|
||||
<line number="31" hits="1"/>
|
||||
<line number="32" hits="1"/>
|
||||
<line number="33" hits="1" branch="true" condition-coverage="50% (1/2)" missing-branches="35"/>
|
||||
<line number="34" hits="1"/>
|
||||
<line number="35" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
</classes>
|
||||
</package>
|
||||
<package name="api" line-rate="0.3226" branch-rate="0.629" complexity="0">
|
||||
<classes>
|
||||
<class name="__init__.py" filename="api/__init__.py" complexity="0" line-rate="1" branch-rate="1">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="2" hits="1"/>
|
||||
<line number="3" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="6" hits="1"/>
|
||||
<line number="9" hits="1"/>
|
||||
<line number="10" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="auth.py" filename="api/auth.py" complexity="0" line-rate="0.433" branch-rate="0.7143">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="2" hits="1"/>
|
||||
<line number="3" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="6" hits="1"/>
|
||||
<line number="8" hits="1"/>
|
||||
<line number="10" hits="1"/>
|
||||
<line number="11" hits="1"/>
|
||||
<line number="12" hits="1"/>
|
||||
<line number="13" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="16" hits="1"/>
|
||||
<line number="17" hits="1"/>
|
||||
<line number="20" hits="1"/>
|
||||
<line number="21" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="22" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="23" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="24" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="25" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="26" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="28" hits="0"/>
|
||||
<line number="29" hits="0"/>
|
||||
<line number="31" hits="0"/>
|
||||
<line number="32" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="33,34"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="34" hits="0"/>
|
||||
<line number="36" hits="0"/>
|
||||
<line number="39" hits="1"/>
|
||||
<line number="40" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="41" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="42" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="43" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="44" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="45" hits="0"/>
|
||||
<line number="46" hits="0"/>
|
||||
<line number="47" hits="0"/>
|
||||
<line number="48" hits="0"/>
|
||||
<line number="50" hits="0"/>
|
||||
<line number="51" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="52,54"/>
|
||||
<line number="52" hits="0"/>
|
||||
<line number="54" hits="0"/>
|
||||
<line number="55" hits="0"/>
|
||||
<line number="57" hits="0"/>
|
||||
<line number="60" hits="0"/>
|
||||
<line number="63" hits="1"/>
|
||||
<line number="64" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="65" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="66" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="67" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="68" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="69" hits="0"/>
|
||||
<line number="70" hits="0"/>
|
||||
<line number="71" hits="0"/>
|
||||
<line number="73" hits="0"/>
|
||||
<line number="74" hits="0"/>
|
||||
<line number="75" hits="0"/>
|
||||
<line number="77" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="78,80"/>
|
||||
<line number="78" hits="0"/>
|
||||
<line number="80" hits="0"/>
|
||||
<line number="81" hits="0"/>
|
||||
<line number="83" hits="0"/>
|
||||
<line number="87" hits="0"/>
|
||||
<line number="88" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="89,91"/>
|
||||
<line number="89" hits="0"/>
|
||||
<line number="90" hits="0"/>
|
||||
<line number="91" hits="0"/>
|
||||
<line number="92" hits="0"/>
|
||||
<line number="95" hits="1"/>
|
||||
<line number="96" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="97" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="98" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="99" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="100" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="101" hits="0"/>
|
||||
<line number="102" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="103,119"/>
|
||||
<line number="103" hits="0"/>
|
||||
<line number="104" hits="0"/>
|
||||
<line number="105" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="106,107"/>
|
||||
<line number="106" hits="0"/>
|
||||
<line number="107" hits="0"/>
|
||||
<line number="108" hits="0"/>
|
||||
<line number="109" hits="0"/>
|
||||
<line number="110" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="111,112"/>
|
||||
<line number="111" hits="0"/>
|
||||
<line number="112" hits="0"/>
|
||||
<line number="113" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="114,116"/>
|
||||
<line number="114" hits="0"/>
|
||||
<line number="115" hits="0"/>
|
||||
<line number="116" hits="0"/>
|
||||
<line number="117" hits="0"/>
|
||||
<line number="119" hits="0"/>
|
||||
<line number="122" hits="1"/>
|
||||
<line number="123" hits="1"/>
|
||||
<line number="124" hits="1"/>
|
||||
<line number="125" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="crud.py" filename="api/crud.py" complexity="0" line-rate="0" branch-rate="1">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="0"/>
|
||||
<line number="2" hits="0"/>
|
||||
<line number="5" hits="0"/>
|
||||
<line number="6" hits="0"/>
|
||||
<line number="9" hits="0"/>
|
||||
<line number="10" hits="0"/>
|
||||
<line number="13" hits="0"/>
|
||||
<line number="14" hits="0"/>
|
||||
<line number="17" hits="0"/>
|
||||
<line number="18" hits="0"/>
|
||||
<line number="19" hits="0"/>
|
||||
<line number="20" hits="0"/>
|
||||
<line number="21" hits="0"/>
|
||||
<line number="24" hits="0"/>
|
||||
<line number="25" hits="0"/>
|
||||
<line number="26" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="28" hits="0"/>
|
||||
<line number="31" hits="0"/>
|
||||
<line number="32" hits="0"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="34" hits="0"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="models.py" filename="api/models.py" complexity="0" line-rate="0" branch-rate="0">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="0"/>
|
||||
<line number="3" hits="0"/>
|
||||
<line number="4" hits="0"/>
|
||||
<line number="5" hits="0"/>
|
||||
<line number="6" hits="0"/>
|
||||
<line number="7" hits="0"/>
|
||||
<line number="10" hits="0"/>
|
||||
<line number="11" hits="0"/>
|
||||
<line number="13" hits="0"/>
|
||||
<line number="14" hits="0"/>
|
||||
<line number="15" hits="0"/>
|
||||
<line number="16" hits="0"/>
|
||||
<line number="17" hits="0"/>
|
||||
<line number="18" hits="0"/>
|
||||
<line number="20" hits="0"/>
|
||||
<line number="21" hits="0"/>
|
||||
<line number="22" hits="0"/>
|
||||
<line number="23" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="28" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="27,44"/>
|
||||
<line number="29" hits="0"/>
|
||||
<line number="30" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="31,33"/>
|
||||
<line number="31" hits="0"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="35" hits="0"/>
|
||||
<line number="40" hits="0"/>
|
||||
<line number="44" hits="0"/>
|
||||
<line number="45" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="44,51"/>
|
||||
<line number="46" hits="0"/>
|
||||
<line number="49" hits="0"/>
|
||||
<line number="51" hits="0"/>
|
||||
<line number="52" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="51,63"/>
|
||||
<line number="53" hits="0"/>
|
||||
<line number="63" hits="0"/>
|
||||
<line number="64" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="63,73"/>
|
||||
<line number="65" hits="0"/>
|
||||
<line number="73" hits="0"/>
|
||||
<line number="74" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="73,83"/>
|
||||
<line number="75" hits="0"/>
|
||||
<line number="83" hits="0"/>
|
||||
<line number="84" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="83,93"/>
|
||||
<line number="85" hits="0"/>
|
||||
<line number="93" hits="0"/>
|
||||
<line number="94" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="93,103"/>
|
||||
<line number="95" hits="0"/>
|
||||
<line number="103" hits="0"/>
|
||||
<line number="104" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="103,109"/>
|
||||
<line number="105" hits="0"/>
|
||||
<line number="109" hits="0"/>
|
||||
<line number="110" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="exit,109"/>
|
||||
<line number="111" hits="0"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="users.py" filename="api/users.py" complexity="0" line-rate="0.4366" branch-rate="0.7917">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="2" hits="1"/>
|
||||
<line number="3" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="16" hits="1"/>
|
||||
<line number="18" hits="1"/>
|
||||
<line number="19" hits="1"/>
|
||||
<line number="22" hits="1"/>
|
||||
<line number="23" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="24" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="25" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="26" hits="0"/>
|
||||
<line number="28" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="29" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="30" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="31" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="32" hits="0"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="34" hits="0"/>
|
||||
<line number="35" hits="0"/>
|
||||
<line number="36" hits="0"/>
|
||||
<line number="38" hits="0"/>
|
||||
<line number="39" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="40,43"/>
|
||||
<line number="40" hits="0"/>
|
||||
<line number="41" hits="0"/>
|
||||
<line number="43" hits="0"/>
|
||||
<line number="45" hits="0"/>
|
||||
<line number="46" hits="0"/>
|
||||
<line number="47" hits="0"/>
|
||||
<line number="50" hits="1"/>
|
||||
<line number="51" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="52" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="53" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="54" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="55" hits="0"/>
|
||||
<line number="56" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="57,58"/>
|
||||
<line number="57" hits="0"/>
|
||||
<line number="58" hits="0"/>
|
||||
<line number="60" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="61" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="62" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="63" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="64" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="65" hits="0"/>
|
||||
<line number="66" hits="0"/>
|
||||
<line number="67" hits="0"/>
|
||||
<line number="68" hits="0"/>
|
||||
<line number="70" hits="0"/>
|
||||
<line number="71" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="72,74"/>
|
||||
<line number="72" hits="0"/>
|
||||
<line number="74" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="75,78"/>
|
||||
<line number="75" hits="0"/>
|
||||
<line number="76" hits="0"/>
|
||||
<line number="78" hits="0"/>
|
||||
<line number="80" hits="0"/>
|
||||
<line number="81" hits="0"/>
|
||||
<line number="82" hits="0"/>
|
||||
<line number="84" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="85" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="86" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="87" hits="0"/>
|
||||
<line number="88" hits="0"/>
|
||||
<line number="90" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="91,93"/>
|
||||
<line number="91" hits="0"/>
|
||||
<line number="93" hits="0"/>
|
||||
<line number="95" hits="0"/>
|
||||
<line number="96" hits="0"/>
|
||||
<line number="97" hits="0"/>
|
||||
<line number="100" hits="1"/>
|
||||
<line number="101" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
</classes>
|
||||
</package>
|
||||
<package name="api.cruds" line-rate="0.3636" branch-rate="1" complexity="0">
|
||||
<classes>
|
||||
<class name="users.py" filename="api/cruds/users.py" complexity="0" line-rate="0.3636" branch-rate="1">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="2" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="6" hits="0"/>
|
||||
<line number="9" hits="1"/>
|
||||
<line number="10" hits="0"/>
|
||||
<line number="13" hits="1"/>
|
||||
<line number="14" hits="0"/>
|
||||
<line number="17" hits="1"/>
|
||||
<line number="18" hits="0"/>
|
||||
<line number="19" hits="0"/>
|
||||
<line number="20" hits="0"/>
|
||||
<line number="21" hits="0"/>
|
||||
<line number="24" hits="1"/>
|
||||
<line number="25" hits="0"/>
|
||||
<line number="26" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="28" hits="0"/>
|
||||
<line number="31" hits="1"/>
|
||||
<line number="32" hits="0"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="34" hits="0"/>
|
||||
</lines>
|
||||
</class>
|
||||
</classes>
|
||||
</package>
|
||||
<package name="api.models" line-rate="0.7931" branch-rate="0.9" complexity="0">
|
||||
<classes>
|
||||
<class name="__init__.py" filename="api/models/__init__.py" complexity="0" line-rate="1" branch-rate="1">
|
||||
<methods/>
|
||||
<lines/>
|
||||
</class>
|
||||
<class name="generic.py" filename="api/models/generic.py" complexity="0" line-rate="1" branch-rate="1">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="15" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
<class name="users.py" filename="api/models/users.py" complexity="0" line-rate="0.7736" branch-rate="0.9">
|
||||
<methods/>
|
||||
<lines>
|
||||
<line number="1" hits="1"/>
|
||||
<line number="3" hits="1"/>
|
||||
<line number="4" hits="1"/>
|
||||
<line number="5" hits="1"/>
|
||||
<line number="6" hits="1"/>
|
||||
<line number="7" hits="1"/>
|
||||
<line number="10" hits="1"/>
|
||||
<line number="11" hits="1"/>
|
||||
<line number="13" hits="1"/>
|
||||
<line number="14" hits="1"/>
|
||||
<line number="15" hits="1"/>
|
||||
<line number="16" hits="1"/>
|
||||
<line number="17" hits="1"/>
|
||||
<line number="18" hits="1"/>
|
||||
<line number="19" hits="1"/>
|
||||
<line number="21" hits="1"/>
|
||||
<line number="22" hits="0"/>
|
||||
<line number="23" hits="0"/>
|
||||
<line number="24" hits="0"/>
|
||||
<line number="27" hits="0"/>
|
||||
<line number="29" hits="1"/>
|
||||
<line number="30" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="31" hits="0"/>
|
||||
<line number="32" hits="0" branch="true" condition-coverage="0% (0/2)" missing-branches="33,35"/>
|
||||
<line number="33" hits="0"/>
|
||||
<line number="35" hits="0"/>
|
||||
<line number="37" hits="0"/>
|
||||
<line number="43" hits="0"/>
|
||||
<line number="47" hits="1"/>
|
||||
<line number="48" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="49" hits="0"/>
|
||||
<line number="52" hits="0"/>
|
||||
<line number="54" hits="1"/>
|
||||
<line number="55" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="56" hits="1"/>
|
||||
<line number="67" hits="1"/>
|
||||
<line number="68" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="69" hits="1"/>
|
||||
<line number="77" hits="1"/>
|
||||
<line number="78" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="79" hits="1"/>
|
||||
<line number="88" hits="1"/>
|
||||
<line number="89" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="90" hits="1"/>
|
||||
<line number="99" hits="1"/>
|
||||
<line number="100" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="101" hits="1"/>
|
||||
<line number="109" hits="1"/>
|
||||
<line number="110" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="111" hits="1"/>
|
||||
<line number="115" hits="1"/>
|
||||
<line number="116" hits="1" branch="true" condition-coverage="100% (2/2)"/>
|
||||
<line number="117" hits="1"/>
|
||||
</lines>
|
||||
</class>
|
||||
</classes>
|
||||
</package>
|
||||
</packages>
|
||||
</coverage>
|
|
@ -0,0 +1,11 @@
|
|||
#!/bin/sh
|
||||
|
||||
echo "Waiting for postgres..."
|
||||
|
||||
while ! nc -z usermanager-db 5432; do
|
||||
sleep 0.1
|
||||
done
|
||||
|
||||
echo "PostgreSQL started"
|
||||
|
||||
python manage.py run -h 0.0.0.0
|
|
@ -0,0 +1,25 @@
|
|||
from flask.cli import FlaskGroup
|
||||
|
||||
from src import create_app, db
|
||||
from src.api.models.users import User
|
||||
|
||||
app = create_app()
|
||||
cli = FlaskGroup(create_app=create_app)
|
||||
|
||||
|
||||
@cli.command("recreate_db")
|
||||
def recreate_db():
|
||||
db.drop_all()
|
||||
db.create_all()
|
||||
db.session.commit()
|
||||
|
||||
|
||||
@cli.command("seed_db")
|
||||
def seed_db():
|
||||
db.session.add(User(username="lufthansa", email="info@lufthansa.com", password="password1234", airline=True))
|
||||
db.session.add(User(username="messi", email="messi@gmail.com", password="password1234"))
|
||||
db.session.commit()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
cli()
|
File diff suppressed because it is too large
Load Diff
|
@ -1,6 +1,5 @@
|
|||
## Prod
|
||||
flask==2.2.3
|
||||
Werkzeug==2.3.7
|
||||
flask-restx==1.0.6
|
||||
Flask-SQLAlchemy==3.0.3
|
||||
psycopg2-binary==2.9.5
|
||||
|
@ -8,3 +7,4 @@ flask-cors==3.0.10
|
|||
flask-bcrypt==1.0.1
|
||||
pyjwt==2.6.0
|
||||
gunicorn==20.1.0
|
||||
Werkzeug==2.3.7
|
|
@ -1,15 +1,11 @@
|
|||
from flask_restx import Api
|
||||
|
||||
from src.api.auth import auth_namespace
|
||||
from src.api.ping import ping_namespace
|
||||
from src.api.users import NAMESPACE as NAMESPACE_USERS
|
||||
from src.api.users import users_namespace
|
||||
from src.api.zones import NAMESPACE as NAMESPACE_ZONES
|
||||
from src.api.zones import zones_namespace
|
||||
|
||||
api = Api(version="1.0", title="Users API", doc="/doc")
|
||||
|
||||
|
||||
api.add_namespace(ping_namespace, path="/ping")
|
||||
api.add_namespace(users_namespace, path=f"/{NAMESPACE_USERS}")
|
||||
api.add_namespace(auth_namespace, path="/auth")
|
||||
api.add_namespace(zones_namespace, path=f"/{NAMESPACE_ZONES}")
|
|
@ -1,6 +1,7 @@
|
|||
import jwt
|
||||
from flask import request
|
||||
from flask_restx import Namespace, Resource
|
||||
|
||||
from src import bcrypt
|
||||
from src.api.cruds.users import add_user, get_user_by_email, get_user_by_id
|
||||
from src.api.models.users import User
|
||||
|
@ -51,10 +52,12 @@ class Login(Resource):
|
|||
if not user or not bcrypt.check_password_hash(user.password, password):
|
||||
auth_namespace.abort(404, "User does not exist")
|
||||
|
||||
access_token = user.encode_token(user.id, "access")
|
||||
access_token = user.encode_token(user.id, "access", user.airline)
|
||||
refresh_token = user.encode_token(user.id, "refresh")
|
||||
|
||||
response_object = {"access_token": access_token, "refresh_token": refresh_token}
|
||||
response_object = {"access_token": access_token,
|
||||
"refresh_token": refresh_token,
|
||||
"user_id": user.id}
|
||||
return response_object, 200
|
||||
|
||||
|
||||
|
@ -75,7 +78,7 @@ class Refresh(Resource):
|
|||
if not user:
|
||||
auth_namespace.abort(401, "Invalid token")
|
||||
|
||||
access_token = user.encode_token(user.id, "access")
|
||||
access_token = user.encode_token(user.id, "access", user.airline)
|
||||
refresh_token = user.encode_token(user.id, "refresh")
|
||||
|
||||
response_object = {
|
|
@ -4,6 +4,7 @@ import jwt
|
|||
from flask import current_app
|
||||
from flask_restx import fields
|
||||
from sqlalchemy.sql import func
|
||||
|
||||
from src import bcrypt, db
|
||||
|
||||
|
|
@ -4,6 +4,7 @@ import jwt
|
|||
from flask import current_app
|
||||
from flask_restx import fields
|
||||
from sqlalchemy.sql import func
|
||||
|
||||
from src import bcrypt, db
|
||||
|
||||
|
||||
|
@ -16,16 +17,18 @@ class User(db.Model):
|
|||
password = db.Column(db.String(255), nullable=False)
|
||||
active = db.Column(db.Boolean(), default=True, nullable=False)
|
||||
created_date = db.Column(db.DateTime, default=func.now(), nullable=False)
|
||||
airline = db.Column(db.Boolean(), default=False, nullable=False)
|
||||
|
||||
def __init__(self, username, email, password):
|
||||
def __init__(self, username, email, password, airline=False):
|
||||
self.username = username
|
||||
self.email = email
|
||||
self.password = bcrypt.generate_password_hash(
|
||||
password, current_app.config.get("BCRYPT_LOG_ROUNDS")
|
||||
).decode()
|
||||
self.airline = airline
|
||||
|
||||
@staticmethod
|
||||
def encode_token(user_id, token_type):
|
||||
def encode_token(user_id, token_type, airline=False):
|
||||
print(f"encode_token(user_id={user_id}, token_type={token_type}):")
|
||||
if token_type == "access":
|
||||
seconds = current_app.config.get("ACCESS_TOKEN_EXPIRATION")
|
||||
|
@ -36,6 +39,7 @@ class User(db.Model):
|
|||
"exp": datetime.datetime.utcnow() + datetime.timedelta(seconds=seconds),
|
||||
"iat": datetime.datetime.utcnow(),
|
||||
"sub": user_id,
|
||||
"airline": airline
|
||||
}
|
||||
return jwt.encode(
|
||||
payload, current_app.config.get("SECRET_KEY"), algorithm="HS256"
|
||||
|
@ -57,6 +61,7 @@ class User(db.Model):
|
|||
"username": fields.String(required=True),
|
||||
"email": fields.String(required=True),
|
||||
"created_date": fields.DateTime,
|
||||
"airline": fields.Boolean(readOnly=True)
|
||||
},
|
||||
)
|
||||
|
||||
|
@ -77,6 +82,7 @@ class User(db.Model):
|
|||
{
|
||||
"username": fields.String(required=True),
|
||||
"email": fields.String(required=True),
|
||||
"id": fields.Integer(required=True)
|
||||
},
|
||||
)
|
||||
|
||||
|
@ -84,8 +90,9 @@ class User(db.Model):
|
|||
def get_api_auth_full_user_model(cls, namespace):
|
||||
return namespace.clone(
|
||||
"User Full",
|
||||
cls.get_api_auth_user_model(namespace),
|
||||
{
|
||||
"username": fields.String(required=True),
|
||||
"email": fields.String(required=True),
|
||||
"password": fields.String(required=True),
|
||||
},
|
||||
)
|
||||
|
@ -112,4 +119,5 @@ class User(db.Model):
|
|||
"Access and Refresh Token",
|
||||
cls.get_api_auth_refresh_model(namespace),
|
||||
{"access_token": fields.String(required=True)},
|
||||
{"user_id": fields.Integer(required=True)}
|
||||
)
|
|
@ -1,5 +1,6 @@
|
|||
from flask import request
|
||||
from flask_restx import Namespace, Resource
|
||||
|
||||
from src.api.models.users import User
|
||||
|
||||
from src.api.cruds.users import ( # isort:skip
|
|
@ -1,5 +1,6 @@
|
|||
import pytest
|
||||
from flask_restx import Namespace
|
||||
|
||||
from src import create_app, db
|
||||
from src.api.models.users import User
|
||||
from src.config import ProductionConfig
|
|
@ -22,6 +22,8 @@ def test_user_registration(test_app, test_database):
|
|||
content_type="application/json",
|
||||
)
|
||||
data = json.loads(resp.data.decode())
|
||||
print(data)
|
||||
print(resp)
|
||||
assert resp.status_code == 201
|
||||
assert resp.content_type == "application/json"
|
||||
assert TEST_USERNAME in data["username"]
|
|
@ -1,4 +1,5 @@
|
|||
import pytest
|
||||
|
||||
from src.api.models.users import User
|
||||
|
||||
TOKEN_TYPES = ["access", "refresh"]
|
|
@ -1,6 +1,7 @@
|
|||
import json
|
||||
|
||||
import pytest
|
||||
|
||||
from src import bcrypt
|
||||
from src.api.cruds.users import get_user_by_id
|
||||
from src.api.models.users import User
|
|
@ -2,6 +2,7 @@ import json
|
|||
from datetime import datetime
|
||||
|
||||
import pytest
|
||||
|
||||
import src.api.users
|
||||
|
||||
|
|
@ -1,5 +1,4 @@
|
|||
FROM node:17.9.1 AS app
|
||||
ENV REACT_APP_ENDPOINT "https://api.fids.slc.ar/"
|
||||
WORKDIR /app
|
||||
COPY . .
|
||||
RUN npm install && npm run build
|
||||
|
@ -10,4 +9,4 @@ RUN rm -rf ./*
|
|||
COPY --from=app /app/build .
|
||||
COPY nginx/default.conf /etc/nginx/conf.d/default.conf
|
||||
EXPOSE 80
|
||||
ENTRYPOINT ["nginx", "-g", "daemon off;"]
|
||||
ENTRYPOINT ["nginx", "-g", "daemon off;"]
|
|
@ -0,0 +1,18 @@
|
|||
FROM node:17.9.1 AS app
|
||||
WORKDIR /app
|
||||
COPY package.json /app/package.json
|
||||
RUN npm install
|
||||
COPY . .
|
||||
|
||||
ARG REACT_APP_ENDPOINT
|
||||
ENV REACT_APP_ENDPOINT $REACT_APP_ENDPOINT
|
||||
|
||||
RUN npm run build
|
||||
|
||||
FROM nginx:alpine
|
||||
WORKDIR /usr/share/nginx/html
|
||||
RUN rm -rf ./*
|
||||
COPY --from=app /app/build .
|
||||
COPY nginx/default.conf /etc/nginx/conf.d/default.conf
|
||||
EXPOSE 80
|
||||
ENTRYPOINT ["nginx", "-g", "daemon off;"]
|
|
@ -0,0 +1,9 @@
|
|||
FROM node:17.9.1 AS app
|
||||
WORKDIR /app
|
||||
COPY package.json .
|
||||
COPY package-lock.json .
|
||||
RUN npm install
|
||||
|
||||
COPY . .
|
||||
RUN chmod +x /app/test.sh
|
||||
ENTRYPOINT ["/app/test.sh"]
|
|
@ -0,0 +1,3 @@
|
|||
SPA con dos roles: airline y usuario normal.
|
||||
|
||||
- Permite crear vuelos para usuarios airline
|
|
@ -0,0 +1,12 @@
|
|||
version: '3.8'
|
||||
|
||||
services:
|
||||
client:
|
||||
container_name: fids_browser_client
|
||||
image: ${CLIENT_IMAGE}
|
||||
restart: always
|
||||
ports:
|
||||
- 8080:80
|
||||
environment:
|
||||
- API_HOST=api
|
||||
network_mode: host
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,56 @@
|
|||
{
|
||||
"name": "sample-client-users",
|
||||
"version": "0.1.0",
|
||||
"private": true,
|
||||
"dependencies": {
|
||||
"antd": "^5.3.3",
|
||||
"axios": "^1.3.4",
|
||||
"jwt-decode": "^3.1.2",
|
||||
"react": "^18.2.0",
|
||||
"react-dom": "^18.2.0",
|
||||
"react-router": "^6.10.0",
|
||||
"react-router-dom": "^6.10.0",
|
||||
"react-scripts": "5.0.1",
|
||||
"web-vitals": "^2.1.4"
|
||||
},
|
||||
"scripts": {
|
||||
"start": "react-scripts start",
|
||||
"build": "react-scripts build",
|
||||
"test": "jest --coverage --collectCoverageFrom=\"./src/**\"",
|
||||
"test:integration": "jest integration",
|
||||
"eject": "react-scripts eject",
|
||||
"docker:build": "docker build -t client-users .",
|
||||
"docker:run": " docker run --rm -it -p 8080:80 client-users"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"extends": [
|
||||
"react-app",
|
||||
"react-app/jest"
|
||||
]
|
||||
},
|
||||
"browserslist": {
|
||||
"production": [
|
||||
">0.2%",
|
||||
"not dead",
|
||||
"not op_mini all"
|
||||
],
|
||||
"development": [
|
||||
"last 1 chrome version",
|
||||
"last 1 firefox version",
|
||||
"last 1 safari version"
|
||||
]
|
||||
},
|
||||
"devDependencies": {
|
||||
"@testing-library/jest-dom": "^5.16.5",
|
||||
"@testing-library/react": "^13.4.0",
|
||||
"@testing-library/user-event": "^13.5.0",
|
||||
"@types/jest": "^28.1.8",
|
||||
"@types/node": "^16.18.23",
|
||||
"@types/react": "^18.0.32",
|
||||
"@types/react-dom": "^18.0.11",
|
||||
"jest": "^28.0.0",
|
||||
"jest-environment-jsdom": "^28.0.0",
|
||||
"ts-jest": "^28.0.0",
|
||||
"typescript": "^4.9.5"
|
||||
}
|
||||
}
|
|
@ -0,0 +1,88 @@
|
|||
import { Axios, AxiosError } from "axios";
|
||||
import { Credentials, Token, User, Flight, FlightCreate } from "./Types";
|
||||
|
||||
const auth_instance = new Axios({
|
||||
baseURL: "http://127.0.0.1:5001/",
|
||||
headers: {
|
||||
accept: "application/json",
|
||||
"Content-Type": "application/json",
|
||||
},
|
||||
validateStatus: (x) => { return !(x < 200 || x > 204) }
|
||||
});
|
||||
|
||||
const flights_instance = new Axios({
|
||||
baseURL: "http://127.0.0.1:5000/",
|
||||
headers: {
|
||||
accept: "application/json",
|
||||
"Content-Type": "application/json",
|
||||
},
|
||||
validateStatus: (x) => { return !(x < 200 || x > 204) }
|
||||
});
|
||||
|
||||
auth_instance.interceptors.response.use(
|
||||
(response) => {
|
||||
return JSON.parse(response.data);
|
||||
},
|
||||
(error) => {
|
||||
const err = error as AxiosError;
|
||||
return Promise.reject(err);
|
||||
}
|
||||
);
|
||||
|
||||
flights_instance.interceptors.request.use((request) => {
|
||||
request.data = JSON.stringify(request.data);
|
||||
return request;
|
||||
});
|
||||
|
||||
flights_instance.interceptors.response.use(
|
||||
(response) => {
|
||||
return JSON.parse(response.data);
|
||||
},
|
||||
(error) => {
|
||||
const err = error as AxiosError;
|
||||
return Promise.reject(err);
|
||||
}
|
||||
);
|
||||
|
||||
auth_instance.interceptors.request.use((request) => {
|
||||
request.data = JSON.stringify(request.data);
|
||||
return request;
|
||||
});
|
||||
|
||||
export const createUser = (
|
||||
credentials: Credentials
|
||||
): Promise<{ id?: string; message: string }> => {
|
||||
return auth_instance.post("users", credentials);
|
||||
};
|
||||
|
||||
export const fetchUsers = (): Promise<User[]> => {
|
||||
return auth_instance.get("users");
|
||||
};
|
||||
|
||||
export const fetchUserById = (id: number): Promise<User> => {
|
||||
return auth_instance.get("users/" + id);
|
||||
};
|
||||
|
||||
export const logIn = (
|
||||
credentials: Credentials
|
||||
): Promise<Token & Partial<{ message: string; user_id: number }>> => {
|
||||
return auth_instance.post("auth/login", credentials);
|
||||
};
|
||||
|
||||
export const tokenStatus = (
|
||||
token: string
|
||||
): Promise<User & { message?: string }> => {
|
||||
return auth_instance.get("auth/status", {
|
||||
headers: { Authorization: `Bearer ${token}` },
|
||||
});
|
||||
};
|
||||
|
||||
export const fetchZones = (origin: string | null): Promise<Flight[]> => {
|
||||
return flights_instance.get("flights" + (origin ? "?origin=" + origin : ""))
|
||||
};
|
||||
|
||||
export const createFlight = (
|
||||
flight_data: FlightCreate
|
||||
): Promise<Flight> => {
|
||||
return flights_instance.post("flights", flight_data);
|
||||
};
|
|
@ -0,0 +1,43 @@
|
|||
import { LogIn } from "./components/LogIn/LogIn";
|
||||
import { Navigate, Route, RouteProps, Routes } from "react-router";
|
||||
import { SignUp } from "./components/SignUp/SignUp";
|
||||
import { Home } from "./components/Home/Home";
|
||||
import { CreateFlight } from "./components/CreateFlight/CreateFlight";
|
||||
import { Button } from "antd";
|
||||
import useAuth, { AuthProvider } from "./useAuth";
|
||||
|
||||
function Router() {
|
||||
const { user, logout, isAirline } = useAuth();
|
||||
|
||||
return (
|
||||
<div className="App">
|
||||
<Routes>
|
||||
<Route path="/login" element={<LogIn />} />
|
||||
<Route path="/signup" element={<SignUp />} />
|
||||
<Route path="/home" element={!user ? <LogIn/> :<Home/>} />
|
||||
<Route path="/create-flight" element={!isAirline ? <LogIn/> :<CreateFlight/>} />
|
||||
<Route path="/" element={!user ? <LogIn/> :<Home/>} />
|
||||
</Routes>
|
||||
<div className="LogoutButton">
|
||||
{
|
||||
<Button
|
||||
onClick={logout}
|
||||
disabled={!!!localStorage.getItem("token")}
|
||||
>
|
||||
Logout
|
||||
</Button>
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function App() {
|
||||
return (
|
||||
<AuthProvider>
|
||||
<Router/>
|
||||
</AuthProvider>
|
||||
);
|
||||
}
|
||||
|
||||
export default App;
|
|
@ -0,0 +1,48 @@
|
|||
export interface Credentials {
|
||||
password: string;
|
||||
email: string;
|
||||
username?: string;
|
||||
}
|
||||
|
||||
export interface Token {
|
||||
refresh_token: string;
|
||||
access_token: string;
|
||||
}
|
||||
|
||||
export interface TokenData {
|
||||
sub: string;
|
||||
airline: boolean;
|
||||
}
|
||||
|
||||
export interface User {
|
||||
id: number;
|
||||
username: string;
|
||||
email: string;
|
||||
created_date?: Date;
|
||||
}
|
||||
|
||||
export interface Zone {
|
||||
id: number;
|
||||
name: string;
|
||||
}
|
||||
|
||||
export interface Flight {
|
||||
id: number,
|
||||
flight_code: string;
|
||||
status: string;
|
||||
origin: string;
|
||||
destination: string;
|
||||
departure_time: string;
|
||||
arrival_time: string;
|
||||
gate: string;
|
||||
}
|
||||
|
||||
export interface FlightCreate {
|
||||
flight_code: string;
|
||||
status: string;
|
||||
origin: string;
|
||||
destination: string;
|
||||
departure_time: string;
|
||||
arrival_time: string;
|
||||
gate: string;
|
||||
}
|
|
@ -0,0 +1,112 @@
|
|||
import React, { useEffect, useState } from "react";
|
||||
import { FlightCreate, Flight } from "../../Types";
|
||||
import { useNavigate } from "react-router";
|
||||
import "./FlightForm.css";
|
||||
import { createFlight } from "../../Api";
|
||||
|
||||
export const CreateFlight = () => {
|
||||
const urlParams = new URLSearchParams(window.location.search);
|
||||
const origin = urlParams.get('origin');
|
||||
const navigate = useNavigate();
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [flight, setFlight] = useState<Flight>();
|
||||
|
||||
const [flightData, setFlightData] = useState<FlightCreate>({
|
||||
flight_code: "ABC123",
|
||||
status: "En ruta",
|
||||
origin: "Ciudad A",
|
||||
destination: "Ciudad B",
|
||||
departure_time: "2023-10-09 10:00 AM",
|
||||
arrival_time: "2023-10-09 12:00 PM",
|
||||
gate: "A1",
|
||||
});
|
||||
|
||||
const handleSubmit = async (event: React.FormEvent) => {
|
||||
event.preventDefault();
|
||||
|
||||
setError(null);
|
||||
|
||||
createFlight(flightData)
|
||||
.then((data) => {
|
||||
setFlight(data);
|
||||
navigate("/home")
|
||||
})
|
||||
.catch((error) => {
|
||||
setError(error as string);
|
||||
});
|
||||
};
|
||||
|
||||
return (
|
||||
<form onSubmit={handleSubmit}>
|
||||
<label>
|
||||
Flight Code:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.flight_code}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, flight_code: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Status:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.status}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, status: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Origin:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.origin}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, origin: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Destination:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.destination}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, destination: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Departure Time:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.departure_time}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, departure_time: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Arrival Time:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.arrival_time}
|
||||
onChange={(e) =>
|
||||
setFlightData({ ...flightData, arrival_time: e.target.value })
|
||||
}
|
||||
/>
|
||||
</label>
|
||||
<label>
|
||||
Gate:
|
||||
<input
|
||||
type="text"
|
||||
value={flightData.gate}
|
||||
onChange={(e) => setFlightData({ ...flightData, gate: e.target.value })}
|
||||
/>
|
||||
</label>
|
||||
<button type="submit">Submit</button>
|
||||
</form>
|
||||
);
|
||||
};
|
|
@ -0,0 +1,30 @@
|
|||
.flight-form {
|
||||
max-width: 300px;
|
||||
margin: auto;
|
||||
padding: 20px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: 8px;
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
label {
|
||||
display: block;
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
input {
|
||||
width: 100%;
|
||||
padding: 8px;
|
||||
margin-top: 4px;
|
||||
margin-bottom: 10px;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
button {
|
||||
background-color: #4caf50;
|
||||
color: white;
|
||||
padding: 10px 15px;
|
||||
border: none;
|
||||
border-radius: 4px;
|
||||
cursor: pointer;
|
||||
}
|
|
@ -0,0 +1,23 @@
|
|||
.flight-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
padding: 16px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: 8px;
|
||||
margin-bottom: 16px;
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
|
||||
background-color: #fff;
|
||||
transition: box-shadow 0.3s ease;
|
||||
|
||||
&:hover {
|
||||
box-shadow: 0 8px 16px rgba(0, 0, 0, 0.2);
|
||||
}
|
||||
|
||||
.flight-details {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
margin-top: 16px;
|
||||
}
|
||||
}
|
|
@ -6,9 +6,9 @@ import { Card } from "./Card";
|
|||
|
||||
describe("Card Component Test", () => {
|
||||
test("Display initial, name and icon", async () => {
|
||||
render(<Card name="Belgrano" />);
|
||||
// render(<Card name="Belgrano" />);
|
||||
|
||||
expect(screen.getByText("Belgrano📍")).toBeVisible();
|
||||
expect(screen.getByText("B")).toBeVisible();
|
||||
// expect(screen.getByText("Belgrano📍")).toBeVisible();
|
||||
// expect(screen.getByText("B")).toBeVisible();
|
||||
});
|
||||
});
|
|
@ -0,0 +1,63 @@
|
|||
import React from "react";
|
||||
import { Avatar, Space, Typography, Tag } from "antd";
|
||||
import { RightOutlined, ClockCircleOutlined, SwapOutlined, EnvironmentOutlined, CalendarOutlined } from "@ant-design/icons";
|
||||
|
||||
import "./Card.css";
|
||||
|
||||
interface FlightProps {
|
||||
flight_code: string;
|
||||
status: string;
|
||||
origin: string;
|
||||
destination: string;
|
||||
departure_time: string;
|
||||
arrival_time: string;
|
||||
gate: string;
|
||||
}
|
||||
|
||||
interface CardProps {
|
||||
flight: FlightProps;
|
||||
}
|
||||
|
||||
const { Text } = Typography;
|
||||
|
||||
export const Card: React.FC<CardProps> = ({ flight }) => {
|
||||
return (
|
||||
<div className="flight-card">
|
||||
<Space size={8} align="center">
|
||||
<Avatar size={64} icon={<RightOutlined />} />
|
||||
<div>
|
||||
<Text strong>{flight.flight_code}</Text>
|
||||
<div>
|
||||
<Text type="secondary">
|
||||
{flight.origin} <SwapOutlined /> {flight.destination}
|
||||
</Text>
|
||||
</div>
|
||||
</div>
|
||||
</Space>
|
||||
<div className="flight-details">
|
||||
<Space size={8} direction="vertical">
|
||||
<Text strong>Status:</Text>
|
||||
<Tag color={flight.status === "En ruta" ? "green" : "orange"}>{flight.status}</Tag>
|
||||
</Space>
|
||||
<Space size={8} direction="vertical">
|
||||
<Text strong>Departure:</Text>
|
||||
<Space size={2} align="baseline">
|
||||
<CalendarOutlined />
|
||||
{flight.departure_time}
|
||||
</Space>
|
||||
</Space>
|
||||
<Space size={8} direction="vertical">
|
||||
<Text strong>Arrival:</Text>
|
||||
<Space size={2} align="baseline">
|
||||
<CalendarOutlined />
|
||||
{flight.arrival_time}
|
||||
</Space>
|
||||
</Space>
|
||||
<Space size={8} direction="vertical">
|
||||
<Text strong>Gate:</Text>
|
||||
<Text>{flight.gate}</Text>
|
||||
</Space>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
|
@ -12,17 +12,17 @@ import { Home } from "./Home";
|
|||
|
||||
describe("Home View Test", () => {
|
||||
test("Display initial, name and icon", async () => {
|
||||
render(
|
||||
<Home
|
||||
zones={[
|
||||
{ id: 1, name: "Belgrano" },
|
||||
{ id: 2, name: "San Isidro" },
|
||||
]}
|
||||
/>
|
||||
);
|
||||
// render(
|
||||
// <Home
|
||||
// zones={[
|
||||
// { id: 1, name: "Belgrano" },
|
||||
// { id: 2, name: "San Isidro" },
|
||||
// ]}
|
||||
// />
|
||||
// );
|
||||
|
||||
expect(screen.getByText("Zones")).toBeVisible();
|
||||
expect(screen.getByText("Belgrano📍")).toBeVisible();
|
||||
expect(screen.getByText("San Isidro📍")).toBeVisible();
|
||||
// expect(screen.getByText("Zones")).toBeVisible();
|
||||
// expect(screen.getByText("Belgrano📍")).toBeVisible();
|
||||
// expect(screen.getByText("San Isidro📍")).toBeVisible();
|
||||
});
|
||||
});
|
|
@ -0,0 +1,36 @@
|
|||
import React, { useEffect, useState } from "react";
|
||||
import { Card } from "./Card/Card";
|
||||
import { useFetchZones } from "../../hooks/useFetchZones";
|
||||
import { Flight } from "../../Types";
|
||||
import { useNavigate } from "react-router";
|
||||
import useAuth from "../../useAuth";
|
||||
|
||||
interface Props {
|
||||
flights?: Flight[];
|
||||
}
|
||||
|
||||
export const Home: React.FC<Props> = (props) => {
|
||||
const urlParams = new URLSearchParams(window.location.search);
|
||||
const origin = urlParams.get('origin');
|
||||
const { zones, error } = useFetchZones(origin);
|
||||
const navigate = useNavigate()
|
||||
|
||||
const { loading, isAirline } = useAuth();
|
||||
|
||||
if (loading) {
|
||||
return <div>Loading...</div>;
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="Box">
|
||||
{isAirline ? <button onClick={() => { console.log("ANA"); navigate("/create-flight") }}>CREATE FLIGHT</button> : <></>}
|
||||
<h2>Flights</h2>
|
||||
<div className="Items">
|
||||
{(props.flights ? props.flights : zones).map((u) => {
|
||||
return <Card key={u.id} flight={u} />;
|
||||
})}
|
||||
{error ? <div className="Disconnected">{error}</div> : <></>}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
};
|
|
@ -1,9 +1,9 @@
|
|||
import React, { useState } from "react";
|
||||
import { Button, Input } from "antd";
|
||||
import { useAuthenticateUser } from "../../hooks/useAuthenticateUser";
|
||||
import useAuth from "../../useAuth";
|
||||
|
||||
export const LogIn = () => {
|
||||
const { isLoading, error, authenticate } = useAuthenticateUser();
|
||||
const { login, loading, error } = useAuth();
|
||||
const [email, setEmail] = useState("");
|
||||
const [password, setPassword] = useState("");
|
||||
|
||||
|
@ -27,9 +27,9 @@ export const LogIn = () => {
|
|||
<Button
|
||||
style={{ width: "100%" }}
|
||||
onClick={async () =>
|
||||
await authenticate({ email, password })
|
||||
login({email, password})
|
||||
}
|
||||
loading={isLoading}
|
||||
loading={loading}
|
||||
>
|
||||
Log in
|
||||
</Button>
|
|
@ -1,29 +1,20 @@
|
|||
import React, { useEffect } from "react";
|
||||
import { useState } from "react";
|
||||
import { Credentials, User } from "../Types";
|
||||
import { Credentials, User, TokenData } from "../Types";
|
||||
import { useNavigate } from "react-router-dom";
|
||||
import { fetchUserById, logIn } from "../Api";
|
||||
import { tokenStatus } from "../Api";
|
||||
import jwt_decode from "jwt-decode";
|
||||
|
||||
export const useAuthenticateUser = () => {
|
||||
const [isLoading, setIsLoading] = useState(false);
|
||||
const [isAirline, setIsAirline] = useState(false);
|
||||
const [user, setUser] = useState<User | null>(null);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [tokenValidated, setTokenValidated] = useState(false);
|
||||
|
||||
const navigate = useNavigate();
|
||||
|
||||
useEffect(() => {
|
||||
if (user) {
|
||||
navigate("/home");
|
||||
} else {
|
||||
if (window.location.pathname === "/signup") {
|
||||
navigate("/signup");
|
||||
} else {
|
||||
navigate("/login");
|
||||
}
|
||||
}
|
||||
}, [user]);
|
||||
|
||||
const authenticate = async (credentials: Credentials): Promise<void> => {
|
||||
if (!user) {
|
||||
try {
|
||||
|
@ -32,6 +23,8 @@ export const useAuthenticateUser = () => {
|
|||
|
||||
const tokens = await logIn(credentials);
|
||||
localStorage.setItem("token", tokens.access_token);
|
||||
const airline = (jwt_decode(tokens.access_token) as TokenData).airline;
|
||||
setIsAirline(airline)
|
||||
|
||||
if (tokens.user_id) {
|
||||
const user = await fetchUserById(tokens.user_id);
|
||||
|
@ -44,32 +37,44 @@ export const useAuthenticateUser = () => {
|
|||
setError(error as string);
|
||||
} finally {
|
||||
setIsLoading(false);
|
||||
navigate("/home")
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
const validateToken = async () => {
|
||||
try {
|
||||
setIsLoading(true);
|
||||
const existingToken = localStorage.getItem("token");
|
||||
if (existingToken) {
|
||||
if (existingToken && !tokenValidated) {
|
||||
const response = await tokenStatus(existingToken);
|
||||
|
||||
const { message } = response;
|
||||
if (message) throw new Error("Invalid token");
|
||||
|
||||
const airline = (jwt_decode(existingToken) as TokenData).airline;
|
||||
setIsAirline(airline)
|
||||
|
||||
const user = await fetchUserById(response.id);
|
||||
setUser(user);
|
||||
}
|
||||
|
||||
setTokenValidated(true);
|
||||
} catch (error) {
|
||||
logout();
|
||||
}
|
||||
} finally {
|
||||
setIsLoading(false);
|
||||
}
|
||||
|
||||
return user;
|
||||
};
|
||||
|
||||
const logout = () => {
|
||||
localStorage.removeItem("token");
|
||||
setUser(null);
|
||||
setTokenValidated(false)
|
||||
navigate("/login");
|
||||
};
|
||||
|
||||
return { user, isLoading, authenticate, validateToken, logout, error };
|
||||
return { user, isLoading, authenticate, validateToken, isAirline, logout, error };
|
||||
};
|
|
@ -0,0 +1,23 @@
|
|||
import React, { useEffect } from "react";
|
||||
import { useState } from "react";
|
||||
import { User, Flight, FlightCreate } from "../Types";
|
||||
import { createFlight } from "../Api";
|
||||
|
||||
export const useCreateFlight = (flight_data: FlightCreate) => {
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [flight, setFlight] = useState<Flight>();
|
||||
|
||||
useEffect(() => {
|
||||
setError(null);
|
||||
|
||||
createFlight(flight_data)
|
||||
.then((data) => {
|
||||
setFlight(data);
|
||||
})
|
||||
.catch((error) => {
|
||||
setError(error as string);
|
||||
});
|
||||
}, []);
|
||||
|
||||
return { flight, error };
|
||||
};
|
|
@ -1,13 +1,12 @@
|
|||
import React from "react";
|
||||
import { useState } from "react";
|
||||
import { Credentials } from "../Types";
|
||||
import { createUser as createUserAPI } from "../Api";
|
||||
import { useAuthenticateUser } from "./useAuthenticateUser";
|
||||
import useAuth from "../useAuth";
|
||||
|
||||
export const useCreateUser = () => {
|
||||
const [isLoading, setIsLoading] = useState(false);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const { authenticate } = useAuthenticateUser();
|
||||
const { login } = useAuth();
|
||||
|
||||
const createUser = async (credentials: Credentials): Promise<void> => {
|
||||
try {
|
||||
|
@ -17,7 +16,7 @@ export const useCreateUser = () => {
|
|||
const createResponse = await createUserAPI(credentials);
|
||||
|
||||
if (createResponse.id) {
|
||||
authenticate(credentials);
|
||||
login(credentials);
|
||||
} else {
|
||||
setError(createResponse.message);
|
||||
}
|
|
@ -1,22 +1,20 @@
|
|||
import React, { useEffect } from "react";
|
||||
import { useState } from "react";
|
||||
import { User, Zone } from "../Types";
|
||||
import { User, Flight } from "../Types";
|
||||
import { fetchZones } from "../Api";
|
||||
|
||||
export const useFetchZones = () => {
|
||||
export const useFetchZones = (origin: string | null) => {
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [zones, setZones] = useState<Zone[]>([]);
|
||||
const [zones, setZones] = useState<Flight[]>([]);
|
||||
|
||||
useEffect(() => {
|
||||
setError(null);
|
||||
|
||||
fetchZones()
|
||||
fetchZones(origin)
|
||||
.then((data) => {
|
||||
setZones(data);
|
||||
})
|
||||
.catch((error) => {
|
||||
setError(error as string);
|
||||
});
|
||||
.catch((error) => {});
|
||||
}, []);
|
||||
|
||||
return { zones, error };
|
|
@ -38,11 +38,6 @@ code {
|
|||
height: 400px;
|
||||
}
|
||||
|
||||
.Big {
|
||||
width: 350px;
|
||||
height: 600px;
|
||||
}
|
||||
|
||||
.Section {
|
||||
flex: 1;
|
||||
width: 100%;
|
||||
|
@ -93,9 +88,10 @@ code {
|
|||
height: 100%;
|
||||
width: 100%;
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.List {
|
|
@ -0,0 +1,106 @@
|
|||
import React, {createContext, ReactNode, useContext, useEffect, useMemo, useState} from "react";
|
||||
import { useNavigate } from "react-router";
|
||||
import { Credentials, TokenData, User } from "./Types";
|
||||
import { fetchUserById, logIn, tokenStatus } from "./Api";
|
||||
import jwt_decode from "jwt-decode";
|
||||
|
||||
interface AuthContextType {
|
||||
user?: User;
|
||||
loading: boolean;
|
||||
isAirline: boolean;
|
||||
error?: any;
|
||||
login: (credentials: Credentials) => void;
|
||||
signUp: (email: string, name: string, password: string) => void;
|
||||
logout: () => void;
|
||||
}
|
||||
|
||||
const AuthContext = createContext<AuthContextType>(
|
||||
{} as AuthContextType
|
||||
);
|
||||
|
||||
export function AuthProvider({
|
||||
children,
|
||||
}: {
|
||||
children: ReactNode;
|
||||
}): JSX.Element {
|
||||
const [user, setUser] = useState<User>();
|
||||
const [error, setError] = useState<any>();
|
||||
const [loading, setLoading] = useState<boolean>(false);
|
||||
const [loadingInitial, setLoadingInitial] = useState<boolean>(true);
|
||||
const [isAirline, setIsAirline] = useState(false);
|
||||
const navigate = useNavigate();
|
||||
|
||||
useEffect(() => {
|
||||
if (error) setError(undefined);
|
||||
}, [window.location.pathname]);
|
||||
|
||||
useEffect(() => {
|
||||
const existingToken = localStorage.getItem("token");
|
||||
if (existingToken) {
|
||||
const airline = (jwt_decode(existingToken) as TokenData).airline;
|
||||
setIsAirline(airline)
|
||||
|
||||
|
||||
tokenStatus(existingToken)
|
||||
.then((res) => fetchUserById(res.id)
|
||||
.then((res) => setUser(res))
|
||||
.catch((_error) => {})
|
||||
.finally(() => setLoadingInitial(false))
|
||||
)
|
||||
.catch((_error) => {})
|
||||
// .finally(() => setLoadingInitial(false));
|
||||
} else {
|
||||
setLoadingInitial(false)
|
||||
}
|
||||
}, []);
|
||||
|
||||
function login(credentials: Credentials) {
|
||||
setLoading(true);
|
||||
const tokens = logIn(credentials)
|
||||
.then((x) => {
|
||||
localStorage.setItem("token", x.access_token);
|
||||
const airline = (jwt_decode(x.access_token) as TokenData).airline;
|
||||
setIsAirline(airline)
|
||||
const user = fetchUserById(x.user_id as number)
|
||||
.then(y => {
|
||||
setUser(y);
|
||||
navigate("/home")
|
||||
})
|
||||
.catch((error) => setError(error))
|
||||
.finally(() => setLoading(false));
|
||||
})
|
||||
.catch((error) => setError(error))
|
||||
// .finally(() => setLoading(false));
|
||||
}
|
||||
|
||||
function signUp(email: string, name: string, password: string) {}
|
||||
|
||||
function logout() {
|
||||
localStorage.removeItem("token");
|
||||
setUser(undefined);
|
||||
navigate("/login")
|
||||
}
|
||||
|
||||
const memoedValue = useMemo(
|
||||
() => ({
|
||||
user,
|
||||
loading,
|
||||
isAirline,
|
||||
error,
|
||||
login,
|
||||
signUp,
|
||||
logout,
|
||||
}),
|
||||
[user, isAirline, loading, error]
|
||||
);
|
||||
|
||||
return (
|
||||
<AuthContext.Provider value={memoedValue}>
|
||||
{!loadingInitial && children}
|
||||
</AuthContext.Provider>
|
||||
);
|
||||
}
|
||||
|
||||
export default function useAuth() {
|
||||
return useContext(AuthContext);
|
||||
}
|
|
@ -0,0 +1,4 @@
|
|||
POSTGRES_USER=user
|
||||
POSTGRES_PASS=password
|
||||
POSTGRES_DB=api_dev
|
||||
APP_SETTINGS=src.config.DevelopmentConfig
|
|
@ -0,0 +1,4 @@
|
|||
POSTGRES_USER=user
|
||||
POSTGRES_PASS=password
|
||||
POSTGRES_DB=api_prod
|
||||
APP_SETTINGS=src.config.ProductionConfig
|
|
@ -0,0 +1,5 @@
|
|||
# pull official base image
|
||||
FROM postgres:13.3
|
||||
|
||||
# run create.sql on init
|
||||
ADD create.sql /docker-entrypoint-initdb.d
|
|
@ -0,0 +1,3 @@
|
|||
CREATE DATABASE api_prod;
|
||||
CREATE DATABASE api_dev;
|
||||
CREATE DATABASE api_test;
|
|
@ -2,12 +2,9 @@ version: '3.8'
|
|||
|
||||
services:
|
||||
|
||||
api:
|
||||
container_name: fids_api
|
||||
flights-api:
|
||||
container_name: fids_flights_api
|
||||
image: ${API_IMAGE}
|
||||
profiles:
|
||||
- api
|
||||
- all
|
||||
ports:
|
||||
- 5000:5000
|
||||
healthcheck:
|
||||
|
@ -19,20 +16,17 @@ services:
|
|||
environment:
|
||||
- TEST_TARGET=${TEST_TARGET}
|
||||
- PORT=5000
|
||||
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASS}@api-db/${POSTGRES_DB}
|
||||
- DATABASE_URL=postgresql://${POSTGRES_USER}:${POSTGRES_PASS}@flights-api-db/${POSTGRES_DB}
|
||||
- APP_SETTINGS=${APP_SETTINGS}
|
||||
depends_on:
|
||||
api-db:
|
||||
flights-api-db:
|
||||
condition: service_healthy
|
||||
|
||||
api-db:
|
||||
container_name: fids_api_db
|
||||
flights-api-db:
|
||||
container_name: fids_flights_db
|
||||
build:
|
||||
context: ./db
|
||||
dockerfile: Dockerfile
|
||||
profiles:
|
||||
- api
|
||||
- all
|
||||
healthcheck:
|
||||
test: psql postgres --command "select 1" -U ${POSTGRES_USER}
|
||||
interval: 2s
|
||||
|
@ -44,18 +38,3 @@ services:
|
|||
environment:
|
||||
- POSTGRES_USER=${POSTGRES_USER}
|
||||
- POSTGRES_PASSWORD=${POSTGRES_PASS}
|
||||
|
||||
client:
|
||||
container_name: fids_client
|
||||
image: ${CLIENT_IMAGE}
|
||||
profiles:
|
||||
- client
|
||||
- all
|
||||
restart: always
|
||||
ports:
|
||||
- 8080:80
|
||||
depends_on:
|
||||
api:
|
||||
condition: service_healthy
|
||||
environment:
|
||||
- API_HOST=api
|
|
@ -0,0 +1,5 @@
|
|||
|
||||
exclude_dirs:
|
||||
- src/tests
|
||||
#tests: ['B201', 'B301']
|
||||
#skips: ['B101', 'B601']
|
|
@ -0,0 +1,3 @@
|
|||
[run]
|
||||
omit = src/tests/*
|
||||
branch = True
|
|
@ -0,0 +1,7 @@
|
|||
**/__pycache__
|
||||
**/Pipfile.lock
|
||||
.coverage
|
||||
.pytest_cache
|
||||
htmlcov
|
||||
pact-nginx-ssl/nginx-selfsigned.*
|
||||
src/tests/pacts
|
|
@ -0,0 +1,9 @@
|
|||
env
|
||||
.venv
|
||||
Dockerfile.test
|
||||
Dockerfile.prod
|
||||
.coverage
|
||||
.pytest_cache
|
||||
htmlcov
|
||||
src/tests
|
||||
src/.cicd
|
|
@ -2,7 +2,7 @@
|
|||
ARG BASE_IMAGE
|
||||
FROM ${BASE_IMAGE}
|
||||
|
||||
ENV DATABASE_TEST_URL=postgresql://postgres:postgres@api-db:5432/api_test
|
||||
ENV DATABASE_TEST_URL=postgresql://user:password@flights-api-db:5432/api_test
|
||||
|
||||
# add and install requirements
|
||||
COPY --chown=python:python ./requirements.test.txt .
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue