对于 fastapi apscheduler 集成以及支持动态添加任务的简单说明
参考使用
- 依赖
包含了fastapi,apscheduler sqlalchemy uvicorn
pip install fastapi apscheduler sqlalchemy uvicorn
- 代码简单说明
代码使用了基于数据库jobstores,同时演示了一个简单的编程模式添加job(动态加载模块方式)以及基于配置的job 初始化
yaml 格式配置的job 定义
version: v1
tasks:
- name: baidu
modulename: mytask
taskname: task
interval: 5
- name: google
modulename: mytaskv2
taskname: task
interval: 5
tasks:
fastapi 集成代码
from fastapi import FastAPI
from apscheduler.schedulers.asyncio import AsyncIOScheduler as Scheduler
from apscheduler.triggers.interval import IntervalTrigger
from importlib import import_module
from contextlib import asynccontextmanager
import logging
logging.basicConfig()
logging.getLogger('apscheduler').setLevel(logging.DEBUG)
from apscheduler.jobstores.sqlalchemy import SQLAlchemyJobStore
from sqlalchemy import create_engine
from config import config
from apscheduler.events import EVENT_ALL,JobEvent
# Database URL
DATABASE_URL = "sqlite:///./test.db"
# Create the database engine
engine = create_engine(DATABASE_URL)
# Configure the job store
jobstores = {
'default': SQLAlchemyJobStore(engine=engine)
}
scheduler = Scheduler(jobstores=jobstores)
def listener(event:JobEvent) -> None:
print(event)
@asynccontextmanager
async def lifespan(app: FastAPI):
myconfig = config()
app.state.scheduler = scheduler
scheduler.add_listener(listener, EVENT_ALL)
for task in myconfig.tasks:
module = import_module(task.get('modulename'))
interval = task.get('interval')
print("interval", interval)
scheduler.add_job(getattr(module, task.get('taskname')), IntervalTrigger(seconds=interval))
scheduler.start()
yield
app.state.scheduler.shutdown()
app = FastAPI(lifespan=lifespan)
@app.get("/")
async def read_root():
return {"message": "Hello World"}
@app.get("/lists")
async def tasklists():
return [ item.id for item in scheduler.get_jobs() ]
@app.get("/add")
async def addtask():
module = import_module("mytaskv3")
interval = 3
task = getattr(module, "task")
# 动态加载模式
job = scheduler.add_job(task,IntervalTrigger(seconds=interval),name="mydemo",id="mydemo")
return dict(id=job.id, name=job.name, next_run_time=job.next_run_time)
if __name__ == "__main__":
import uvicorn
try:
uvicorn.run(app, host="0.0.0.0", port=8000)
except KeyboardInterrupt as e:
print(e)
from apscheduler.schedulers.asyncio import AsyncIOScheduler as Scheduler
动态加载简单说明,核心是对于开发好的任务通过import_module 进行模块加载,之后通过属性获取实际的任务方法
说明
注意目前使用的apscheduler 版本是3.10.4 当前4.0 版本还处于预发布阶段,很多使用上变化还是比较大的,目前apscheduler 对于任务依赖上是缺少支持的,也是可以做,但是并不是很方便,此场景rocketry 是一个不错的选择,但是目前看着是缺少维护了,从使用上两个实际上可以配置起来使用,可以解决一些依赖的问题
参考资料
https://github.com/agronholm/apscheduler
https://fastapi.tiangolo.com/advanced/events/
https://apscheduler.readthedocs.io/en/3.x/versionhistory.html
https://github.com/Miksus/rocketry