Initial iol

This commit is contained in:
Roberto Alsina 2023-07-01 19:59:14 -03:00
parent f067d97095
commit f0ab7c90dd
12 changed files with 200 additions and 0 deletions

2
.gitignore vendored
View File

@ -165,3 +165,5 @@ build
.secrets
shard.lock
template

View File

@ -27,4 +27,12 @@ functions:
image: ralsina/c-busqueda:latest
build_args:
ADDITIONAL_PACKAGE: gnuplot
iol:
lang: python3-fastapi
handler: ./iol
image: ralsina/iol:latest
secrets:
- iol-pass
- iol-user
- iol-api-secret

0
iol/__init__.py Normal file
View File

64
iol/handler.py Normal file
View File

@ -0,0 +1,64 @@
import logging
import time
from functools import lru_cache
import requests as r
from fastapi import APIRouter
BASE_URL = "https://api.invertironline.com/"
USER = open("/var/openfaas/secrets/iol-user").read().strip()
PASS = open("/var/openfaas/secrets/iol-pass").read().strip()
SECRET = open("/var/openfaas/secrets/iol-api-secret").read().strip()
router = APIRouter()
def get_ttl_hash(seconds=3600):
"""Return the same value withing `seconds` time period"""
return round(time.time() / seconds)
@lru_cache
def get_token(ttl_hash=None):
logging.error("getting token")
url = BASE_URL + "token"
data = {"username": USER, "password": PASS, "grant_type": "password"}
response = r.post(url, data=data)
access_token = response.json()["access_token"]
refresh_token = response.json()["refresh_token"]
return access_token, refresh_token
def refresh_token(refresh_token):
url = BASE_URL + "token"
data = {"refresh_token": refresh_token, "grant_type": "refresh_token"}
response = r.post(url, data=data)
access_token = response.json()["access_token"]
refresh_token = response.json()["refresh_token"]
return access_token, refresh_token
paises = {
"US": "estados_Unidos",
"AR": "argentina",
}
@lru_cache
def get(url, ttl_hash=None):
logging.error("getting data")
access, refresh = get_token(ttl_hash=get_ttl_hash())
response = r.get(url, headers={"Authorization": "Bearer " + access, "Accept": "application/json"})
return response.json()
@router.get("/{secret}/accion/{pais}/{accion}")
def get_accion(secret, pais, accion):
if secret != SECRET:
raise Exception("BAD")
pais = pais.upper()
accion = accion.upper()
access, refresh = get_token(ttl_hash=get_ttl_hash())
url = (
BASE_URL
+ f"/api/v2/Cotizaciones/acciones/{pais}/Todos?cotizacionInstrumentoModel.instrumento=acciones&cotizacionInstrumentoModel.pais={paises[pais]}&api_key={access}"
)
data = get(url, ttl_hash=get_ttl_hash())
return [a for a in data["titulos"] if a["simbolo"] == accion][0]

3
iol/requirements.txt Normal file
View File

@ -0,0 +1,3 @@
fastapi
requests
uvicorn[standard]

View File

@ -0,0 +1,49 @@
FROM --platform=${TARGETPLATFORM:-linux/amd64} ghcr.io/openfaas/of-watchdog:0.9.10 as watchdog
FROM --platform=${TARGETPLATFORM:-linux/amd64} python:3.11-alpine
COPY --from=watchdog /fwatchdog /usr/bin/fwatchdog
RUN chmod +x /usr/bin/fwatchdog
ARG ADDITIONAL_PACKAGE
RUN apk --no-cache add musl-dev gcc make ${ADDITIONAL_PACKAGE}
# Add non root user
RUN addgroup -S app && adduser app -S -G app
RUN chown app /home/app
USER app
ENV PATH=$PATH:/home/app/.local/bin
WORKDIR /home/app/
COPY requirements.txt .
USER root
RUN pip install -r requirements.txt
USER app
COPY index.py .
RUN mkdir -p function
RUN touch ./function/__init__.py
WORKDIR /home/app/function/
COPY function/requirements.txt .
RUN pip install --user -r requirements.txt
WORKDIR /home/app/
USER root
# remove build dependencies
RUN apk del musl-dev gcc make
COPY function function
RUN chown -R app:app ./
USER app
ENV fprocess="uvicorn index:app --workers 1 --host 0.0.0.0 --port 8000"
ENV cgi_headers="true"
ENV mode="http"
ENV upstream_url="http://127.0.0.1:8000"
HEALTHCHECK --interval=5s CMD [ -e /tmp/.lock ] || exit 1
CMD ["fwatchdog"]

View File

@ -0,0 +1,51 @@
# author: Justin Guese, 11.3.22, justin@datafortress.cloud
from fastapi import HTTPException
from fastapi import APIRouter
from pydantic import BaseModel
from typing import Dict, List
from os import environ
import glob
# reads in secrets to environment variables, such that they can be
# easily used with environ["SECRET_NAME"]
def readSecretToEnv(secretpath):
secretname = secretpath.split('/')[-1]
with open(secretpath, "r") as f:
environ[secretname] = f.read()
for secret in glob.glob("/var/openfaas/secrets/*"):
readSecretToEnv(secret)
router = APIRouter()
# just as an example
class User(BaseModel):
id: int
name: str
age: int
colleagues: List[str]
class ResponseModel(BaseModel):
data: Dict
# user: User
# otherStuff: str
@router.post("/", response_model = ResponseModel, tags=["Main Routes"])
def handle(request: Dict):
"""handle a request to the function
Args:
req (dict): request body
"""
try:
res = ResponseModel(data=request)
except Exception as e:
raise HTTPException(status_code=500, detail=str(repr(e)))
return res
@router.get("/", response_model = ResponseModel, tags=["Main Routes"])
def get():
return ResponseModel(data={"message": "Hello from OpenFAAS!"})
# again just as an example, delete this if not required
@router.get("/users/{user_id}", response_model = User, tags=["Main Routes"])
def getUser(user_id: int):
return User(id = user_id, name="Exampleuser", age=20, colleagues=["Colleague 1", "Colleague 2"])

View File

@ -0,0 +1,19 @@
# author: Justin Guese, 11.3.22, justin@datafortress.cloud
from os import environ
import glob
from fastapi import FastAPI, Request
from fastapi.openapi.docs import get_swagger_ui_html
from function.handler import router
app = FastAPI()
app.include_router(router)
# required to render /docs path
@app.get("/docs", include_in_schema=False)
async def custom_swagger_ui_html(req: Request):
root_path = req.scope.get("root_path", "").rstrip("/")
openapi_url = root_path + app.openapi_url
return get_swagger_ui_html(
openapi_url=openapi_url,
title="API",
)

View File

@ -0,0 +1,2 @@
fastapi
uvicorn

View File

@ -0,0 +1,2 @@
language: python3-fastapi
fprocess: uvicorn index:app --workers 1 --host 0.0.0.0 --port 8000