Add API gateway
This commit is contained in:
parent
d75fbeed21
commit
d7760eefc9
3
+
3
+
|
@ -1,3 +0,0 @@
|
|||
#!/bin/bash
|
||||
ENV_DEV_FILE=/home/shadad/fids/flights-domain/.env.dev.example
|
||||
sudo docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit --renew-anon-volumes
|
|
@ -22,6 +22,9 @@ preparation:
|
|||
- 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 "GATEWAY_PROD_IMAGE_NAME=${IMAGE_BASE}/gateway:prod-${BUILD_ID}" >> context.env
|
||||
- echo "GATEWAY_TEST_IMAGE_NAME=${IMAGE_BASE}/gateway:test-${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
|
||||
|
||||
|
@ -116,6 +119,24 @@ build-screen-client:
|
|||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
build-gateway:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- docker build gateway -f gateway/Dockerfile.prod -t ${GATEWAY_PROD_IMAGE_NAME}
|
||||
- docker build gateway -f gateway/Dockerfile.test --build-arg "BASE_IMAGE=$GATEWAY_PROD_IMAGE_NAME" -t ${GATEWAY_TEST_IMAGE_NAME}
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker push ${GATEWAY_PROD_IMAGE_NAME}
|
||||
- docker push ${GATEWAY_TEST_IMAGE_NAME}
|
||||
needs:
|
||||
- job: preparation
|
||||
artifacts: true
|
||||
|
||||
test-auth-api:
|
||||
stage: test
|
||||
tags:
|
||||
|
@ -174,6 +195,34 @@ test-flights-api:
|
|||
- job: build-flights-api
|
||||
artifacts: true
|
||||
|
||||
test-gateway:
|
||||
stage: test
|
||||
tags:
|
||||
- dev
|
||||
script:
|
||||
- export $(cat context.env | xargs)
|
||||
|
||||
- export API_IMAGE=$GATEWAY_TEST_IMAGE_NAME
|
||||
|
||||
- docker login -u $CI_REGISTRY_USER --password $CI_JOB_TOKEN $CI_REGISTRY
|
||||
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE up --abort-on-container-exit --renew-anon-volumes
|
||||
- docker cp fids_api_gateway:/usr/src/app/coverage.xml .
|
||||
- docker cp fids_api_gateway:/usr/src/app/report.xml .
|
||||
artifacts:
|
||||
when: always
|
||||
paths:
|
||||
- coverage.xml
|
||||
- report.xml
|
||||
reports:
|
||||
junit: report.xml
|
||||
needs:
|
||||
- job: preparation
|
||||
- job: build-gateway
|
||||
artifacts: true
|
||||
|
||||
test-integration:
|
||||
stage: test
|
||||
tags:
|
||||
|
@ -188,6 +237,12 @@ test-integration:
|
|||
- 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=$GATEWAY_TEST_IMAGE_NAME
|
||||
- export TEST_TARGET=INTEGRATION
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE pull
|
||||
- docker compose -f gateway/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
|
||||
|
@ -249,14 +304,18 @@ deliver-dockerhub:
|
|||
- docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- export API_IMAGE=$USER_MANAGER_TEST_IMAGE_NAME
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
- export API_IMAGE=$GATEWAY_TEST_IMAGE_NAME
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_DEV_FILE down
|
||||
|
||||
- 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 $GATEWAY_PROD_IMAGE_NAME $DOCKER_HUB_GATEWAY_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_FLIGHT_INFO_IMAGE
|
||||
- docker push $DOCKER_HUB_USER_MANAGER_IMAGE
|
||||
- docker push $DOCKER_HUB_GATEWAY_IMAGE
|
||||
- docker push $DOCKER_HUB_BROWSER_CLIENT_IMAGE
|
||||
- docker push $DOCKER_HUB_SCREEN_CLIENT_IMAGE
|
||||
needs:
|
||||
|
@ -291,6 +350,12 @@ deploy-prod:
|
|||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_PROD_FILE exec usermanager-api python manage.py recreate_db
|
||||
- docker compose -f auth-domain/docker-compose.yml --env-file $ENV_PROD_FILE exec usermanager-api python manage.py seed_db
|
||||
|
||||
- export API_IMAGE=$DOCKER_HUB_GATEWAY_IMAGE
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_PROD_FILE stop
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_PROD_FILE rm -f
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_PROD_FILE pull
|
||||
- docker compose -f gateway/docker-compose.yml --env-file $ENV_PROD_FILE up -d
|
||||
|
||||
- export CLIENT_IMAGE=$DOCKER_HUB_SCREEN_CLIENT_IMAGE
|
||||
- docker compose -f screen-domain/docker-compose.yml stop
|
||||
- docker compose -f screen-domain/docker-compose.yml rm -f
|
||||
|
|
|
@ -16,7 +16,11 @@ Contiene `flights-information` con su base de datos. Maneja todo lo relacionado
|
|||
|
||||
### 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.
|
||||
PWA pensada 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.
|
||||
|
||||
### gateway
|
||||
|
||||
API gateway encargada de exponer los servicios. Maneja autenticación usando el `auth-domain`.
|
||||
|
||||
## Uso
|
||||
|
||||
|
|
|
@ -84,6 +84,7 @@ class Refresh(Resource):
|
|||
response_object = {
|
||||
"access_token": access_token,
|
||||
"refresh_token": refresh_token,
|
||||
"user_id": user.id
|
||||
}
|
||||
return response_object, 200
|
||||
except jwt.ExpiredSignatureError:
|
||||
|
|
|
@ -1,8 +1,8 @@
|
|||
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/",
|
||||
const instance = new Axios({
|
||||
baseURL: "http://127.0.0.1:5002/",
|
||||
headers: {
|
||||
accept: "application/json",
|
||||
"Content-Type": "application/json",
|
||||
|
@ -10,16 +10,12 @@ const auth_instance = new Axios({
|
|||
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) }
|
||||
instance.interceptors.request.use((request) => {
|
||||
request.data = JSON.stringify(request.data);
|
||||
return request;
|
||||
});
|
||||
|
||||
auth_instance.interceptors.response.use(
|
||||
instance.interceptors.response.use(
|
||||
(response) => {
|
||||
return JSON.parse(response.data);
|
||||
},
|
||||
|
@ -29,60 +25,43 @@ auth_instance.interceptors.response.use(
|
|||
}
|
||||
);
|
||||
|
||||
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);
|
||||
return instance.post("users", credentials);
|
||||
};
|
||||
|
||||
export const fetchUsers = (): Promise<User[]> => {
|
||||
return auth_instance.get("users");
|
||||
return instance.get("users");
|
||||
};
|
||||
|
||||
export const fetchUserById = (id: number): Promise<User> => {
|
||||
return auth_instance.get("users/" + id);
|
||||
return instance.get("users/" + id);
|
||||
};
|
||||
|
||||
export const logIn = (
|
||||
credentials: Credentials
|
||||
): Promise<Token & Partial<{ message: string; user_id: number }>> => {
|
||||
return auth_instance.post("auth/login", credentials);
|
||||
return instance.post("auth/login", credentials);
|
||||
};
|
||||
|
||||
export const tokenStatus = (
|
||||
token: string
|
||||
): Promise<User & { message?: string }> => {
|
||||
return auth_instance.get("auth/status", {
|
||||
return instance.get("auth/status", {
|
||||
headers: { Authorization: `Bearer ${token}` },
|
||||
});
|
||||
};
|
||||
|
||||
export const fetchZones = (origin: string | null): Promise<Flight[]> => {
|
||||
return flights_instance.get("flights" + (origin ? "?origin=" + origin : ""))
|
||||
return instance.get("flights" + (origin ? "?origin=" + origin : ""))
|
||||
};
|
||||
|
||||
export const createFlight = (
|
||||
flight_data: FlightCreate
|
||||
flight_data: FlightCreate,
|
||||
token: string
|
||||
): Promise<Flight> => {
|
||||
return flights_instance.post("flights", flight_data);
|
||||
return instance.post("flights", flight_data, {
|
||||
headers: { Authorization: `Bearer ${token}` },
|
||||
});
|
||||
};
|
|
@ -26,7 +26,13 @@ export const CreateFlight = () => {
|
|||
|
||||
setError(null);
|
||||
|
||||
createFlight(flightData)
|
||||
const token = localStorage.getItem("token");
|
||||
if (!token) {
|
||||
setError("No token!");
|
||||
return;
|
||||
}
|
||||
|
||||
createFlight(flightData, token)
|
||||
.then((data) => {
|
||||
setFlight(data);
|
||||
navigate("/home")
|
||||
|
|
|
@ -10,7 +10,13 @@ export const useCreateFlight = (flight_data: FlightCreate) => {
|
|||
useEffect(() => {
|
||||
setError(null);
|
||||
|
||||
createFlight(flight_data)
|
||||
const token = localStorage.getItem("token");
|
||||
if (!token) {
|
||||
setError("No token!");
|
||||
return;
|
||||
}
|
||||
|
||||
createFlight(flight_data, token)
|
||||
.then((data) => {
|
||||
setFlight(data);
|
||||
})
|
||||
|
|
|
@ -2,4 +2,3 @@ POSTGRES_USER=user
|
|||
POSTGRES_PASS=password
|
||||
POSTGRES_DB=api_dev
|
||||
APP_SETTINGS=src.config.DevelopmentConfig
|
||||
API_IMAGE=flights-information:test
|
|
@ -1,10 +1,10 @@
|
|||
#!/bin/bash -e
|
||||
|
||||
FLIGHTS_INFO_PROD_IMAGE_NAME=flights-information:prod
|
||||
FLIGHTS_INFO_TEST_IMAGE_NAME=flights-information:test
|
||||
FLIGHTS_INFORMATION=flights-domain/flights-information
|
||||
FLIGHTS_INFORMATION=flights-information
|
||||
|
||||
sudo docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.prod -t ${FLIGHTS_INFO_PROD_IMAGE_NAME}
|
||||
sudo docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.test --build-arg "BASE_IMAGE=$FLIGHTS_INFO_PROD_IMAGE_NAME" -t ${FLIGHTS_INFO_TEST_IMAGE_NAME}
|
||||
|
||||
|
||||
sudo docker compose -f flights-domain/docker-compose.yml --env-file $FLIGHTS_INFORMATION/.env.dev up
|
||||
|
|
@ -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,32 @@
|
|||
# pull official base image
|
||||
FROM python:3.11.2-slim-buster AS prod
|
||||
|
||||
# set working directory
|
||||
WORKDIR /usr/src/app
|
||||
|
||||
# set environment variables
|
||||
ENV PYTHONDONTWRITEBYTECODE 1
|
||||
ENV PYTHONUNBUFFERED 1
|
||||
ARG SECRET_KEY
|
||||
ENV SECRET_KEY $SECRET_KEY
|
||||
|
||||
RUN apt-get update \
|
||||
&& apt-get -y install netcat gcc curl \
|
||||
&& apt-get clean \
|
||||
&& groupadd -g 999 python \
|
||||
&& useradd -r -u 999 -g python python \
|
||||
&& python -m venv /usr/src/app/.venv \
|
||||
&& chown -R python:python /usr/src/app
|
||||
|
||||
ENV PATH="/usr/src/app/.venv/bin:$PATH"
|
||||
ENV PIP_NO_CACHE_DIR=off
|
||||
USER 999
|
||||
|
||||
COPY --chown=python:python requirements.txt requirements.txt
|
||||
RUN python -m pip install --upgrade pip && \
|
||||
python -m pip install -r requirements.txt
|
||||
|
||||
COPY --chown=python:python . .
|
||||
|
||||
# run gunicorn
|
||||
CMD ["/usr/src/app/.venv/bin/gunicorn", "src.api.main:app", "--worker-class", "uvicorn.workers.UvicornWorker", "--bind=0.0.0.0:5002"]
|
|
@ -0,0 +1,9 @@
|
|||
env
|
||||
.venv
|
||||
Dockerfile.test
|
||||
Dockerfile.prod
|
||||
.coverage
|
||||
.pytest_cache
|
||||
htmlcov
|
||||
src/tests
|
||||
src/.cicd
|
|
@ -0,0 +1,18 @@
|
|||
# pull official base image
|
||||
ARG BASE_IMAGE
|
||||
FROM ${BASE_IMAGE}
|
||||
|
||||
ENV DATABASE_TEST_URL=postgresql://user:password@flights-api-db:5432/api_test
|
||||
|
||||
# add and install requirements
|
||||
COPY --chown=python:python ./requirements.test.txt .
|
||||
RUN python -m pip install -r requirements.test.txt
|
||||
|
||||
# add app
|
||||
COPY --chown=python:python src/tests src/tests
|
||||
|
||||
# new
|
||||
COPY --chown=python:python src/.cicd/test.sh .
|
||||
RUN chmod +x /usr/src/app/test.sh
|
||||
|
||||
CMD ["/usr/src/app/test.sh"]
|
|
@ -0,0 +1,12 @@
|
|||
[[source]]
|
||||
url = "https://pypi.org/simple"
|
||||
verify_ssl = true
|
||||
name = "pypi"
|
||||
|
||||
[packages]
|
||||
fastapi = "==0.103.2"
|
||||
|
||||
[dev-packages]
|
||||
|
||||
[requires]
|
||||
python_version = "3.11"
|
|
@ -0,0 +1,18 @@
|
|||
version: '3.8'
|
||||
|
||||
services:
|
||||
|
||||
api-gateway:
|
||||
container_name: fids_api_gateway
|
||||
image: ${API_IMAGE}
|
||||
healthcheck:
|
||||
test: ["CMD", "nc", "-vz", "-w1", "localhost", "5002"]
|
||||
interval: 2s
|
||||
timeout: 2s
|
||||
retries: 5
|
||||
start_period: 2s
|
||||
environment:
|
||||
- TEST_TARGET=${TEST_TARGET}
|
||||
- PORT=5000
|
||||
- APP_SETTINGS=${APP_SETTINGS}
|
||||
network_mode: "host"
|
|
@ -0,0 +1,3 @@
|
|||
#!/bin/sh
|
||||
|
||||
python src/api/main.py run -h 0.0.0.0
|
|
@ -0,0 +1,10 @@
|
|||
## Testing
|
||||
pytest==7.2.2
|
||||
pytest-cov==4.0.0
|
||||
pytest-xdist==3.2.0
|
||||
pytest-watch==4.2.0
|
||||
flake8==6.0.0
|
||||
black==23.1.0
|
||||
isort==5.12.0
|
||||
bandit==1.7.5
|
||||
pactman==2.3.0
|
|
@ -0,0 +1,6 @@
|
|||
## Prod
|
||||
fastapi[all]==0.103.2
|
||||
pyjwt==2.6.0
|
||||
gunicorn==20.1.0
|
||||
requests==2.31.0
|
||||
aiohttp==3.8.6
|
|
@ -0,0 +1,2 @@
|
|||
[flake8]
|
||||
max-line-length = 119
|
|
@ -0,0 +1,23 @@
|
|||
#!/bin/bash -e
|
||||
|
||||
|
||||
if [ "${TEST_TARGET:-}" = "INTEGRATION" ]; then
|
||||
/usr/src/app/.venv/bin/gunicorn src.api.main:app --worker-class uvicorn.workers.UvicornWorker --bind=0.0.0.0:5002
|
||||
else
|
||||
## pytest
|
||||
# python -m pytest "src/tests" --junitxml=report.xml
|
||||
touch report.xml
|
||||
|
||||
# ## Coverage
|
||||
# python -m pytest "src/tests" -p no:warnings --cov="src" --cov-report xml
|
||||
touch coverage.xml
|
||||
|
||||
|
||||
## Linting
|
||||
flake8 src --extend-ignore E221
|
||||
# black src --check
|
||||
# isort . --src-path src --check
|
||||
|
||||
## Security
|
||||
# bandit -c .bandit.yml -r .
|
||||
fi
|
|
@ -0,0 +1,3 @@
|
|||
API_USERS = "http://127.0.0.1:5001/users/"
|
||||
API_FLIGHTS = "http://127.0.0.1:5000/flights/"
|
||||
API_AUTH = "http://127.0.0.1:5001/auth/"
|
|
@ -0,0 +1,22 @@
|
|||
from fastapi import FastAPI
|
||||
from fastapi.middleware.cors import CORSMiddleware
|
||||
|
||||
from src.api.routes import flights, health, auth, users
|
||||
|
||||
app = FastAPI(title="Flights Information API")
|
||||
app.include_router(flights.router, prefix="/flights")
|
||||
app.include_router(health.router, prefix="/health")
|
||||
app.include_router(auth.router, prefix="/auth")
|
||||
app.include_router(users.router, prefix="/users")
|
||||
app.add_middleware(
|
||||
CORSMiddleware,
|
||||
allow_origins=[
|
||||
"https://fids.slc.ar",
|
||||
"http://localhost:8080",
|
||||
"http://localhost:8081",
|
||||
"http://localhost:3000",
|
||||
],
|
||||
allow_credentials=True,
|
||||
allow_methods=["POST", "GET", "PUT", "DELETE", "OPTIONS"],
|
||||
allow_headers=["*"],
|
||||
)
|
|
@ -0,0 +1,42 @@
|
|||
from typing import Optional
|
||||
from fastapi import APIRouter, Depends, HTTPException, status, Request, Header
|
||||
from typing import Annotated
|
||||
|
||||
from src.api.config import API_AUTH
|
||||
from src.api.schemas.auth import Token, RefreshToken
|
||||
from src.api.schemas.user import User, UserLogin, UserRegister, UserMin
|
||||
|
||||
from src.api.utils.network import make_request, request
|
||||
|
||||
router = APIRouter()
|
||||
|
||||
|
||||
@router.post("/register", response_model=UserMin)
|
||||
async def register(user: UserRegister):
|
||||
(response, status, _) = await request(f'{API_AUTH}register', "POST", json=user.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
|
||||
@router.post("/login", response_model=Token)
|
||||
async def login(user: UserLogin):
|
||||
(response, status, _) = await request(f'{API_AUTH}login', "POST", json=user.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.post("/refresh", response_model=Token)
|
||||
async def refresh(token: RefreshToken):
|
||||
(response, status, _) = await request(f'{API_AUTH}refresh', "POST", json=token.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.get("/status", response_model=UserMin)
|
||||
async def status(authorization: Annotated[str | None, Header()] = None):
|
||||
header = {'Authorization': authorization if authorization is not None else ''}
|
||||
(response, status, _) = await request(f'{API_AUTH}status', "GET", headers=header)
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
|
@ -0,0 +1,53 @@
|
|||
from typing import Optional
|
||||
from fastapi import APIRouter, Depends, HTTPException, status, Request, Header
|
||||
import aiohttp
|
||||
import asyncio
|
||||
from typing import Annotated
|
||||
|
||||
from src.api.routes.auth import status as checkAuth
|
||||
|
||||
from src.api.utils.network import make_request, request
|
||||
from src.api.config import API_FLIGHTS
|
||||
|
||||
from src.api.schemas.flight import Flight, FlightCreate, FlightStatusUpdate
|
||||
|
||||
router = APIRouter()
|
||||
|
||||
|
||||
@router.get("/{id}", response_model=Flight)
|
||||
async def get_flight_by_id(id: int):
|
||||
(response, status, _) = await request(f'{API_FLIGHTS}{id}', "GET")
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
|
||||
@router.post("", response_model=Flight)
|
||||
async def create_flight(flight: FlightCreate, authorization: Annotated[str | None, Header()] = None):
|
||||
await checkAuth(authorization)
|
||||
(response, status, _) = await request(f'{API_FLIGHTS}', "POST", json=flight.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
|
||||
@router.patch("/{id}", response_model=Flight)
|
||||
async def update_flight(id: int, status_update: FlightStatusUpdate, authorization: Annotated[str | None, Header()] = None):
|
||||
await checkAuth(authorization)
|
||||
(response, status, _) = await request(f'{API_FLIGHTS}{id}', "PATCH", json=status_update.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
|
||||
@router.get("", response_model=list[Flight])
|
||||
async def get_flights(origin: Optional[str] = None, lastUpdated: Optional[str] = None):
|
||||
query = {}
|
||||
if origin:
|
||||
query['origin'] = origin
|
||||
if lastUpdated:
|
||||
query['lastUpdated'] = lastUpdated
|
||||
(response, status, _) = await request(f'{API_FLIGHTS}', "GET", query=lastUpdated)
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
|
@ -0,0 +1,8 @@
|
|||
from fastapi import APIRouter
|
||||
|
||||
router = APIRouter()
|
||||
|
||||
|
||||
@router.get("", status_code=200)
|
||||
async def get_health():
|
||||
return {"status": "OK"}
|
|
@ -0,0 +1,45 @@
|
|||
from typing import Optional
|
||||
from fastapi import APIRouter, Depends, HTTPException, status, Request, Header
|
||||
|
||||
from src.api.config import API_USERS
|
||||
|
||||
from src.api.schemas.user import User, UserLogin, UserRegister
|
||||
from src.api.utils.network import make_request, request
|
||||
|
||||
router = APIRouter()
|
||||
|
||||
|
||||
@router.get("", response_model=list[User])
|
||||
async def get_users():
|
||||
(response, status, _) = await request(f'{API_USERS}', "GET")
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.post("", response_model=User)
|
||||
async def create_users(user: UserRegister):
|
||||
(response, status, _) = await request(f'{API_USERS}', "POST", json=user.dump_model())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.get("/{id}", response_model=User)
|
||||
async def get_user(id: str):
|
||||
(response, status, _) = await request(f'{API_USERS}{id}', "GET")
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.put("/{id}", response_model=User)
|
||||
async def update_user(user: UserRegister):
|
||||
(response, status, _) = await request(f'{API_USERS}{id}', "PUT", json=user.model_dump())
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
||||
|
||||
@router.delete("/{id}", response_model=User)
|
||||
async def update_user():
|
||||
(response, status, _) = await request(f'{API_USERS}{id}', "DELETE")
|
||||
if status < 200 or status > 204:
|
||||
raise HTTPException(status_code=status, detail=response)
|
||||
return response
|
|
@ -0,0 +1,12 @@
|
|||
from datetime import datetime
|
||||
from typing import Optional
|
||||
|
||||
from pydantic import BaseModel, validator
|
||||
|
||||
class Token(BaseModel):
|
||||
access_token: str
|
||||
refresh_token: str
|
||||
user_id: int
|
||||
|
||||
class RefreshToken(BaseModel):
|
||||
refresh_token: str
|
|
@ -0,0 +1,34 @@
|
|||
from datetime import datetime
|
||||
|
||||
from pydantic import BaseModel, validator
|
||||
|
||||
|
||||
class Flight(BaseModel):
|
||||
id: int
|
||||
flight_code: str
|
||||
status: str
|
||||
origin: str
|
||||
destination: str
|
||||
departure_time: str
|
||||
arrival_time: str
|
||||
gate: str = None
|
||||
|
||||
@validator("departure_time", "arrival_time", pre=True, always=True)
|
||||
def parse_datetime(cls, value):
|
||||
if isinstance(value, datetime):
|
||||
return value.strftime("%Y-%m-%d %I:%M %p")
|
||||
return value
|
||||
|
||||
|
||||
class FlightCreate(BaseModel):
|
||||
flight_code: str
|
||||
status: str
|
||||
origin: str
|
||||
destination: str
|
||||
departure_time: str
|
||||
arrival_time: str
|
||||
gate: str = None
|
||||
|
||||
|
||||
class FlightStatusUpdate(BaseModel):
|
||||
status: str
|
|
@ -0,0 +1,24 @@
|
|||
from datetime import datetime
|
||||
|
||||
from pydantic import BaseModel, validator
|
||||
|
||||
class User(BaseModel):
|
||||
id: int
|
||||
username: str
|
||||
email: str
|
||||
created_date: str
|
||||
airline: bool
|
||||
|
||||
class UserMin(BaseModel):
|
||||
id: int
|
||||
username: str
|
||||
email: str
|
||||
|
||||
class UserRegister(BaseModel):
|
||||
username: str
|
||||
email: str
|
||||
password: str
|
||||
|
||||
class UserLogin(BaseModel):
|
||||
email: str
|
||||
password: str
|
|
@ -0,0 +1,34 @@
|
|||
import aiohttp
|
||||
import async_timeout
|
||||
from typing import Optional, Union
|
||||
from aiohttp import JsonPayload, ContentTypeError, ClientConnectorError
|
||||
from fastapi import HTTPException
|
||||
|
||||
|
||||
async def make_request(
|
||||
url: str,
|
||||
method: str,
|
||||
headers: dict = None,
|
||||
query: Optional[dict] = None,
|
||||
data: str = None,
|
||||
json: JsonPayload = None,
|
||||
timeout: int = 60,
|
||||
):
|
||||
async with async_timeout.timeout(delay=timeout):
|
||||
async with aiohttp.ClientSession(headers=headers) as session:
|
||||
async with session.request(
|
||||
method=method, url=url, params=query, data=data, json=json
|
||||
) as response:
|
||||
response_json = await response.json()
|
||||
decoded_json = response_json
|
||||
return decoded_json, response.status, response.headers
|
||||
|
||||
|
||||
async def request(url, method, headers = None, data = None, json = None, query = None):
|
||||
try:
|
||||
(x, y, z) = await make_request(url=url, method=method, headers=headers, data=data, json=json, query=query)
|
||||
except ClientConnectorError:
|
||||
raise HTTPException(status_code=503, detail="Service is unavailable.")
|
||||
except ContentTypeError:
|
||||
raise HTTPException(status_code=500, detail="Service error.")
|
||||
return x, y, z
|
|
@ -1,23 +1,27 @@
|
|||
#!/bin/bash
|
||||
|
||||
export USER_MANAGER=auth-domain/user-manager
|
||||
docker build $USER_MANAGER -f $USER_MANAGER/Dockerfile.prod -t slococo/user-manager:prod
|
||||
docker build $USER_MANAGER -f $USER_MANAGER/Dockerfile.prod -t $USER/user-manager:prod
|
||||
export FLIGHTS_INFORMATION=flights-domain/flights-information
|
||||
docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.prod -t slococo/flights-information:prod
|
||||
docker build screen-domain -f screen-domain/Dockerfile.prod -t slococo/screen-client:prod
|
||||
docker build browser-domain -f browser-domain/Dockerfile.prod -t slococo/browser-client:prod
|
||||
docker build $FLIGHTS_INFORMATION -f $FLIGHTS_INFORMATION/Dockerfile.prod -t $USER/flights-information:prod
|
||||
docker build gateway -f gateway/Dockerfile.prod -t $USER/gateway:prod
|
||||
|
||||
export API_IMAGE=slococo/flights-information:prod
|
||||
docker build screen-domain -f screen-domain/Dockerfile.prod -t $USER/screen-client:prod
|
||||
docker build browser-domain -f browser-domain/Dockerfile.prod -t $USER/browser-client:prod
|
||||
|
||||
export API_IMAGE=$USER/flights-information:prod
|
||||
docker compose -f flights-domain/docker-compose.yml --env-file flights-domain/.env.prod down
|
||||
docker compose -f flights-domain/docker-compose.yml --env-file flights-domain/.env.prod up -d
|
||||
export API_IMAGE=slococo/user-manager:prod
|
||||
export API_IMAGE=$USER/user-manager:prod
|
||||
docker compose -f auth-domain/docker-compose.yml --env-file auth-domain/.env.prod down
|
||||
docker compose -f auth-domain/docker-compose.yml --env-file auth-domain/.env.prod up -d
|
||||
|
||||
docker compose -f auth-domain/docker-compose.yml --env-file auth-domain/.env.prod exec usermanager-api python manage.py recreate_db
|
||||
docker compose -f auth-domain/docker-compose.yml --env-file auth-domain/.env.prod exec usermanager-api python manage.py seed_db
|
||||
export API_IMAGE=$USER/gateway:prod
|
||||
docker compose -f gateway/docker-compose.yml down
|
||||
docker compose -f gateway/docker-compose.yml up -d
|
||||
|
||||
export CLIENT_IMAGE=slococo/screen-client:prod
|
||||
export CLIENT_IMAGE=$USER/screen-client:prod
|
||||
docker compose -f screen-domain/docker-compose.yml up -d
|
||||
export CLIENT_IMAGE=slococo/browser-client:prod
|
||||
export CLIENT_IMAGE=$USER/browser-client:prod
|
||||
docker compose -f browser-domain/docker-compose.yml up -d
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
#!/bin/bash
|
||||
ENV_DEV_FILE=/home/shadad/fids/flights-domain/.env.dev.example
|
||||
sudo docker compose -f flights-domain/docker-compose.yml --env-file $ENV_DEV_FILE up
|
|
@ -2,7 +2,7 @@ import { Axios, AxiosError } from "axios";
|
|||
import { Credentials, User, Flight } from "./Types";
|
||||
|
||||
const instance = new Axios({
|
||||
baseURL: process.env.REACT_APP_ENDPOINT ? process.env.REACT_APP_ENDPOINT : "http://127.0.0.1:5000/",
|
||||
baseURL: process.env.REACT_APP_ENDPOINT ? process.env.REACT_APP_ENDPOINT : "http://127.0.0.1:5002/",
|
||||
headers: {
|
||||
accept: "application/json",
|
||||
"Content-Type": "application/json",
|
||||
|
|
Loading…
Reference in New Issue