mirror of
https://github.com/jaypyles/Scraperr.git
synced 2025-11-21 16:46:21 +00:00
* feat: working new data view * feat: working new data view * fix: remove unused deps * fix: typing * chore: cleanup code
215 lines
6.8 KiB
Python
215 lines
6.8 KiB
Python
# STL
|
|
import datetime
|
|
import uuid
|
|
import traceback
|
|
from io import StringIO
|
|
import csv
|
|
import logging
|
|
import random
|
|
|
|
# PDM
|
|
from fastapi import Depends, APIRouter
|
|
from fastapi.encoders import jsonable_encoder
|
|
from fastapi.responses import JSONResponse, StreamingResponse
|
|
from api.backend.scheduler import scheduler
|
|
from apscheduler.triggers.cron import CronTrigger # type: ignore
|
|
|
|
# LOCAL
|
|
from api.backend.job import insert, update_job, delete_jobs
|
|
from api.backend.models import (
|
|
DeleteCronJob,
|
|
UpdateJobs,
|
|
DownloadJob,
|
|
DeleteScrapeJobs,
|
|
Job,
|
|
CronJob,
|
|
)
|
|
from api.backend.schemas import User
|
|
from api.backend.auth.auth_utils import get_current_user
|
|
from api.backend.utils import clean_text, format_list_for_query
|
|
from api.backend.job.models.job_options import FetchOptions
|
|
|
|
from api.backend.database.common import query
|
|
|
|
from api.backend.job.cron_scheduling.cron_scheduling import (
|
|
delete_cron_job,
|
|
get_cron_job_trigger,
|
|
insert_cron_job,
|
|
get_cron_jobs,
|
|
insert_job_from_cron_job,
|
|
)
|
|
|
|
from api.backend.job.utils.clean_job_format import clean_job_format
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
job_router = APIRouter()
|
|
|
|
|
|
@job_router.post("/update")
|
|
async def update(update_jobs: UpdateJobs, _: User = Depends(get_current_user)):
|
|
"""Used to update jobs"""
|
|
await update_job(update_jobs.ids, update_jobs.field, update_jobs.value)
|
|
|
|
|
|
@job_router.post("/submit-scrape-job")
|
|
async def submit_scrape_job(job: Job):
|
|
LOG.info(f"Recieved job: {job}")
|
|
try:
|
|
job.id = uuid.uuid4().hex
|
|
|
|
job_dict = job.model_dump()
|
|
insert(job_dict)
|
|
|
|
return JSONResponse(content={"id": job.id})
|
|
except Exception as e:
|
|
LOG.error(f"Exception occurred: {traceback.format_exc()}")
|
|
return JSONResponse(content={"error": str(e)}, status_code=500)
|
|
|
|
|
|
@job_router.post("/retrieve-scrape-jobs")
|
|
async def retrieve_scrape_jobs(
|
|
fetch_options: FetchOptions, user: User = Depends(get_current_user)
|
|
):
|
|
LOG.info(f"Retrieving jobs for account: {user.email}")
|
|
ATTRIBUTES = "chat" if fetch_options.chat else "*"
|
|
|
|
try:
|
|
job_query = f"SELECT {ATTRIBUTES} FROM jobs WHERE user = ?"
|
|
results = query(job_query, (user.email,))
|
|
return JSONResponse(content=jsonable_encoder(results[::-1]))
|
|
except Exception as e:
|
|
LOG.error(f"Exception occurred: {e}")
|
|
return JSONResponse(content=[], status_code=500)
|
|
|
|
|
|
@job_router.get("/job/{id}")
|
|
async def job(id: str, user: User = Depends(get_current_user)):
|
|
LOG.info(f"Retrieving jobs for account: {user.email}")
|
|
|
|
try:
|
|
job_query = "SELECT * FROM jobs WHERE user = ? AND id = ?"
|
|
results = query(job_query, (user.email, id))
|
|
return JSONResponse(content=jsonable_encoder(results))
|
|
except Exception as e:
|
|
LOG.error(f"Exception occurred: {e}")
|
|
return JSONResponse(content={"error": str(e)}, status_code=500)
|
|
|
|
|
|
@job_router.post("/download")
|
|
async def download(download_job: DownloadJob):
|
|
LOG.info(f"Downloading job with ids: {download_job.ids}")
|
|
|
|
try:
|
|
job_query = (
|
|
f"SELECT * FROM jobs WHERE id IN {format_list_for_query(download_job.ids)}"
|
|
)
|
|
results = query(job_query, tuple(download_job.ids))
|
|
|
|
csv_buffer = StringIO()
|
|
csv_writer = csv.writer(csv_buffer, quotechar='"', quoting=csv.QUOTE_ALL)
|
|
|
|
headers = ["id", "url", "element_name", "xpath", "text", "user", "time_created"]
|
|
csv_writer.writerow(headers)
|
|
|
|
for result in results:
|
|
for res in result["result"]:
|
|
for url, elements in res.items():
|
|
for element_name, values in elements.items():
|
|
for value in values:
|
|
text = clean_text(value.get("text", "")).strip()
|
|
if text:
|
|
csv_writer.writerow(
|
|
[
|
|
result.get("id", "")
|
|
+ "-"
|
|
+ str(random.randint(0, 1000000)),
|
|
url,
|
|
element_name,
|
|
value.get("xpath", ""),
|
|
text,
|
|
result.get("user", ""),
|
|
result.get("time_created", ""),
|
|
]
|
|
)
|
|
|
|
_ = csv_buffer.seek(0)
|
|
response = StreamingResponse(
|
|
csv_buffer,
|
|
media_type="text/csv",
|
|
)
|
|
response.headers["Content-Disposition"] = "attachment; filename=export.csv"
|
|
return response
|
|
|
|
except Exception as e:
|
|
LOG.error(f"Exception occurred: {e}")
|
|
traceback.print_exc()
|
|
return {"error": str(e)}
|
|
|
|
|
|
@job_router.get("/job/{id}/convert-to-csv")
|
|
async def convert_to_csv(id: str):
|
|
try:
|
|
job_query = f"SELECT * FROM jobs WHERE id = ?"
|
|
results = query(job_query, (id,))
|
|
|
|
return JSONResponse(content=clean_job_format(results))
|
|
except Exception as e:
|
|
LOG.error(f"Exception occurred: {e}")
|
|
traceback.print_exc()
|
|
return {"error": str(e)}
|
|
|
|
|
|
@job_router.post("/delete-scrape-jobs")
|
|
async def delete(delete_scrape_jobs: DeleteScrapeJobs):
|
|
result = await delete_jobs(delete_scrape_jobs.ids)
|
|
return (
|
|
JSONResponse(content={"message": "Jobs successfully deleted."})
|
|
if result
|
|
else JSONResponse({"error": "Jobs not deleted."})
|
|
)
|
|
|
|
|
|
@job_router.post("/schedule-cron-job")
|
|
async def schedule_cron_job(cron_job: CronJob):
|
|
if not cron_job.id:
|
|
cron_job.id = uuid.uuid4().hex
|
|
|
|
if not cron_job.time_created:
|
|
cron_job.time_created = datetime.datetime.now()
|
|
|
|
if not cron_job.time_updated:
|
|
cron_job.time_updated = datetime.datetime.now()
|
|
|
|
insert_cron_job(cron_job)
|
|
|
|
queried_job = query("SELECT * FROM jobs WHERE id = ?", (cron_job.job_id,))
|
|
|
|
scheduler.add_job(
|
|
insert_job_from_cron_job,
|
|
get_cron_job_trigger(cron_job.cron_expression),
|
|
id=cron_job.id,
|
|
args=[queried_job[0]],
|
|
)
|
|
|
|
return JSONResponse(content={"message": "Cron job scheduled successfully."})
|
|
|
|
|
|
@job_router.post("/delete-cron-job")
|
|
async def delete_cron_job_request(request: DeleteCronJob):
|
|
if not request.id:
|
|
return JSONResponse(
|
|
content={"error": "Cron job id is required."}, status_code=400
|
|
)
|
|
|
|
delete_cron_job(request.id, request.user_email)
|
|
scheduler.remove_job(request.id)
|
|
|
|
return JSONResponse(content={"message": "Cron job deleted successfully."})
|
|
|
|
|
|
@job_router.get("/cron-jobs")
|
|
async def get_cron_jobs_request(user: User = Depends(get_current_user)):
|
|
cron_jobs = get_cron_jobs(user.email)
|
|
return JSONResponse(content=jsonable_encoder(cron_jobs))
|