mirror of
https://github.com/resf/distro-tools.git
synced 2024-11-24 22:21:26 +00:00
Add osv-gcs-exporter and fix before/after for OSV API (#6)
This commit is contained in:
parent
3be61b8f77
commit
31edffb4c5
77
apollo/osv-gcs-exporter/__main__.py
Normal file
77
apollo/osv-gcs-exporter/__main__.py
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
"""
|
||||||
|
Export OSV data to GCS.
|
||||||
|
|
||||||
|
Individual JSON files are exported to the following path:
|
||||||
|
gs://<BUCKET>/<ADVISORY_ID>.json
|
||||||
|
"""
|
||||||
|
import argparse
|
||||||
|
import requests
|
||||||
|
import json
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
from google.cloud import storage
|
||||||
|
|
||||||
|
|
||||||
|
def main(args):
|
||||||
|
# Connect to GCS
|
||||||
|
client = storage.Client()
|
||||||
|
bucket = client.get_bucket(args.bucket)
|
||||||
|
|
||||||
|
time_after = None
|
||||||
|
if args.nightly:
|
||||||
|
time_after = (datetime.now() - timedelta(hours=26)).isoformat("T")
|
||||||
|
|
||||||
|
page = 1
|
||||||
|
while True:
|
||||||
|
r = requests.get(
|
||||||
|
"https://apollo.build.resf.org/api/v3/osv",
|
||||||
|
params={
|
||||||
|
"page": page,
|
||||||
|
"limit": 100,
|
||||||
|
"after": time_after,
|
||||||
|
},
|
||||||
|
timeout=60,
|
||||||
|
)
|
||||||
|
r.raise_for_status()
|
||||||
|
advisories = r.json()["advisories"]
|
||||||
|
if not advisories:
|
||||||
|
break
|
||||||
|
|
||||||
|
for advisory in advisories:
|
||||||
|
advisory_json = json.dumps(advisory)
|
||||||
|
|
||||||
|
file_name = f"{advisory['id']}.json"
|
||||||
|
|
||||||
|
# Skip if file already exists
|
||||||
|
blob = bucket.blob(file_name)
|
||||||
|
if blob.exists():
|
||||||
|
print(f"Skipping {file_name} (already exists)")
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Upload to GCS
|
||||||
|
blob = bucket.blob(file_name)
|
||||||
|
blob.upload_from_string(advisory_json)
|
||||||
|
print(f"Uploaded {file_name}")
|
||||||
|
|
||||||
|
page += 1
|
||||||
|
|
||||||
|
print("Done!")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
# Parse arguments
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument(
|
||||||
|
"--bucket",
|
||||||
|
help="GCS bucket to export to",
|
||||||
|
required=True,
|
||||||
|
)
|
||||||
|
# Flag for nightly job
|
||||||
|
parser.add_argument(
|
||||||
|
"--nightly",
|
||||||
|
help="Run as nightly job",
|
||||||
|
action="store_true",
|
||||||
|
)
|
||||||
|
|
||||||
|
# Send args to main
|
||||||
|
main(parser.parse_args())
|
1
apollo/osv-gcs-exporter/requirements.txt
Normal file
1
apollo/osv-gcs-exporter/requirements.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
google-cloud-storage==2.8.0
|
@ -1,3 +1,5 @@
|
|||||||
|
import datetime
|
||||||
|
|
||||||
from typing import TypeVar, Generic, Optional
|
from typing import TypeVar, Generic, Optional
|
||||||
|
|
||||||
from fastapi import APIRouter, Depends
|
from fastapi import APIRouter, Depends
|
||||||
@ -236,8 +238,8 @@ def to_osv_advisory(ui_url: str, advisory: Advisory) -> OSVAdvisory:
|
|||||||
async def get_advisories_osv(
|
async def get_advisories_osv(
|
||||||
params: Params = Depends(),
|
params: Params = Depends(),
|
||||||
product: Optional[str] = None,
|
product: Optional[str] = None,
|
||||||
before: Optional[str] = None,
|
before: Optional[datetime.datetime] = None,
|
||||||
after: Optional[str] = None,
|
after: Optional[datetime.datetime] = None,
|
||||||
cve: Optional[str] = None,
|
cve: Optional[str] = None,
|
||||||
synopsis: Optional[str] = None,
|
synopsis: Optional[str] = None,
|
||||||
keyword: Optional[str] = None,
|
keyword: Optional[str] = None,
|
||||||
|
Loading…
Reference in New Issue
Block a user