Pakete polars und fastAPI hinzugefügt, Database Klasse verbessert, FastAPI route erstellt.
parent
219ad57a35
commit
e2734a3db5
|
@ -25,3 +25,6 @@ pandas = ">=2.2.3,<3"
|
||||||
plotly = ">=5.24.1,<6"
|
plotly = ">=5.24.1,<6"
|
||||||
duckdb = ">=1.1.2,<2"
|
duckdb = ">=1.1.2,<2"
|
||||||
python-dotenv = ">=1.0.1,<2"
|
python-dotenv = ">=1.0.1,<2"
|
||||||
|
fastapi = ">=0.115.4,<0.116"
|
||||||
|
polars = ">=0.20.26,<2"
|
||||||
|
pyarrow = ">=18.0.0,<19"
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
import polars as pl
|
||||||
|
from fastapi import FastAPI, Response
|
||||||
|
|
||||||
|
import data
|
||||||
|
|
||||||
|
d = data.load()
|
||||||
|
|
||||||
|
app = FastAPI()
|
||||||
|
|
||||||
|
|
||||||
|
@app.get("/")
|
||||||
|
def read_root():
|
||||||
|
return {"Hello": "World"}
|
||||||
|
|
||||||
|
@app.get("/items/{item_id}")
|
||||||
|
def read_item(item_id: int):
|
||||||
|
ext = d.extractions_for(item_id).pl()
|
||||||
|
out = ext.with_columns(pl.col("calendar").str.extract_all(r"([0-9]{4}-[0-9]{2}-[0-9]{2})|[0-2]").alias("calendar_data"))
|
||||||
|
out = out.drop(['calendar', 'property_id'])
|
||||||
|
return Response(content=out.write_json(), media_type="application/json")
|
|
@ -4,7 +4,10 @@ from dotenv import load_dotenv
|
||||||
|
|
||||||
from data import database
|
from data import database
|
||||||
|
|
||||||
load_dotenv(dotenv_path='../.env')
|
dirname = os.path.dirname(__file__)
|
||||||
|
envfile = os.path.join(dirname, '../.env')
|
||||||
|
|
||||||
|
load_dotenv(dotenv_path=envfile)
|
||||||
|
|
||||||
def load():
|
def load():
|
||||||
return database.Database(path=os.getenv('DATABASE'))
|
return database.Database(path=os.getenv('DATABASE'))
|
|
@ -1,3 +1,5 @@
|
||||||
|
from threading import Thread, current_thread
|
||||||
|
|
||||||
import duckdb
|
import duckdb
|
||||||
|
|
||||||
|
|
||||||
|
@ -18,7 +20,8 @@ class Database:
|
||||||
).fetchone()
|
).fetchone()
|
||||||
|
|
||||||
def __init__(self, path):
|
def __init__(self, path):
|
||||||
self.connection = duckdb.connect(database = path)
|
duckdb_connection = duckdb.connect(database = path, read_only=True)
|
||||||
|
self.connection = duckdb_connection.cursor()
|
||||||
|
|
||||||
# Install spatial extension if not already installed
|
# Install spatial extension if not already installed
|
||||||
spatial_installed = self.check_duckdb_extensions(extension='spatial')
|
spatial_installed = self.check_duckdb_extensions(extension='spatial')
|
||||||
|
@ -26,6 +29,7 @@ class Database:
|
||||||
self.connection.sql("INSTALL spatial")
|
self.connection.sql("INSTALL spatial")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def db_overview(self):
|
def db_overview(self):
|
||||||
return self.connection.sql("DESCRIBE;").show()
|
return self.connection.sql("DESCRIBE;").show()
|
||||||
|
|
||||||
|
@ -49,7 +53,7 @@ class Database:
|
||||||
consultancy_d.properties
|
consultancy_d.properties
|
||||||
GROUP BY
|
GROUP BY
|
||||||
date;
|
date;
|
||||||
""").show()
|
""")
|
||||||
|
|
||||||
def properties_per_region(self):
|
def properties_per_region(self):
|
||||||
return self.connection.sql("""
|
return self.connection.sql("""
|
||||||
|
@ -65,7 +69,7 @@ class Database:
|
||||||
GROUP BY
|
GROUP BY
|
||||||
properties.seed_id,
|
properties.seed_id,
|
||||||
regions.name
|
regions.name
|
||||||
""").show()
|
""")
|
||||||
|
|
||||||
def properties_unreachable(self):
|
def properties_unreachable(self):
|
||||||
return self.connection.sql("""
|
return self.connection.sql("""
|
||||||
|
@ -178,5 +182,46 @@ class Database:
|
||||||
|
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
def properties_exceptions(self):
|
||||||
|
return self.connection.sql("""
|
||||||
|
SELECT
|
||||||
|
JSON_EXTRACT(exception, '$.status') AS exception_status,
|
||||||
|
COUNT(JSON_EXTRACT(exception, '$.status')) AS exception_count
|
||||||
|
FROM
|
||||||
|
consultancy_d.exceptions
|
||||||
|
WHERE
|
||||||
|
type != 'property'
|
||||||
|
GROUP BY
|
||||||
|
JSON_EXTRACT(exception, '$.status')
|
||||||
|
""")
|
||||||
|
|
||||||
|
def extractions(self):
|
||||||
|
return self.connection.sql("""
|
||||||
|
SELECT
|
||||||
|
JSON_EXTRACT(exception, '$.status') AS exception_status,
|
||||||
|
COUNT(JSON_EXTRACT(exception, '$.status')) AS exception_count
|
||||||
|
FROM
|
||||||
|
consultancy_d.extractions
|
||||||
|
WHERE
|
||||||
|
type != 'property'
|
||||||
|
GROUP BY
|
||||||
|
extractions.date
|
||||||
|
""")
|
||||||
|
|
||||||
|
def extractions_for(self, property_id):
|
||||||
|
return self.connection.sql(f"""
|
||||||
|
SELECT
|
||||||
|
JSON_EXTRACT(body, '$.content.days') as calendar,
|
||||||
|
property_id,
|
||||||
|
created_at
|
||||||
|
FROM
|
||||||
|
consultancy_d.extractions
|
||||||
|
WHERE
|
||||||
|
type == 'calendar' AND
|
||||||
|
property_id = {property_id}
|
||||||
|
ORDER BY
|
||||||
|
property_id
|
||||||
|
""")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>Document</title>
|
||||||
|
<script src=" https://cdn.jsdelivr.net/npm/echarts@5.5.1/dist/echarts.min.js "></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div style="width: 100%; height: 800px;" id="main"></div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
|
||||||
|
var chartDom = document.getElementById('main');
|
||||||
|
var myChart = echarts.init(chartDom);
|
||||||
|
var option;
|
||||||
|
|
||||||
|
function getVirtualData(year) {
|
||||||
|
const date = +echarts.time.parse(year + '-01-01');
|
||||||
|
const end = +echarts.time.parse(+year + 1 + '-01-01');
|
||||||
|
const dayTime = 3600 * 24 * 1000;
|
||||||
|
const data = [];
|
||||||
|
for (let time = date; time < end; time += dayTime) {
|
||||||
|
data.push([
|
||||||
|
echarts.time.format(time, '{yyyy}-{MM}-{dd}', false),
|
||||||
|
Math.floor(Math.random() * 1000)
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(getVirtualData('2017'))
|
||||||
|
|
||||||
|
option = {
|
||||||
|
tooltip: {
|
||||||
|
position: 'top'
|
||||||
|
},
|
||||||
|
visualMap: {
|
||||||
|
min: 0,
|
||||||
|
max: 1000,
|
||||||
|
calculable: true,
|
||||||
|
orient: 'horizontal',
|
||||||
|
left: 'center',
|
||||||
|
top: 'top'
|
||||||
|
},
|
||||||
|
calendar: [
|
||||||
|
{
|
||||||
|
range: '2024',
|
||||||
|
cellSize: ['auto', 20]
|
||||||
|
},
|
||||||
|
],
|
||||||
|
series: [
|
||||||
|
{
|
||||||
|
type: 'heatmap',
|
||||||
|
coordinateSystem: 'calendar',
|
||||||
|
calendarIndex: 0,
|
||||||
|
data: getVirtualData('2017')
|
||||||
|
},
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
option && myChart.setOption(option);
|
||||||
|
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -1,4 +1,16 @@
|
||||||
|
import polars as pl
|
||||||
|
|
||||||
import data
|
import data
|
||||||
|
|
||||||
inst = data.load()
|
inst = data.load()
|
||||||
inst.properties_distance().show()
|
test = inst.extractions_for(1).pl()
|
||||||
|
|
||||||
|
out = test.with_columns(
|
||||||
|
pl.col("calendar").str.extract_all(r"([0-9]{4}-[0-9]{2}-[0-9]{2})|[0-2]").alias("extracted_cal"),
|
||||||
|
)
|
||||||
|
out = out.drop(['calendar', 'property_id'])
|
||||||
|
|
||||||
|
print(out.to_dict(as_series=True))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue