一文读懂DolphinScheduler插件机制:如何轻松扩展任务类型与数据源

一、整体架构概览

DolphinScheduler 的插件体系基于 Java SPI(Service Provider Interface) 机制,配合 Google AutoService 自动生成注册文件,实现了零侵入的插件化扩展

arduino 复制代码
dolphinscheduler-spi                    ← 核心接口层(SPI基础设施)
dolphinscheduler-datasource-plugin      ← 数据源插件层
dolphinscheduler-task-plugin            ← Task插件层
dolphinscheduler-worker                 ← 插件消费层(Worker执行任务)

二、SPI 基础设施

scss 复制代码
PrioritySPI (接口)
  └── getIdentify(): SPIIdentify  ← 插件唯一标识 + 优先级
  └── compareTo(Integer)          ← 优先级比较

PrioritySPIFactory 是插件发现的核心引擎

less 复制代码
通过 Java 标准 ServiceLoader 扫描 classpath
for (T t : ServiceLoader.load(spiClass)) {
    if (map.containsKey(t.getIdentify().getName())) {
        resolveConflict(t);   // 同名插件按优先级决策,优先级相同则抛异常
    } else {
        map.put(t.getIdentify().getName(), t);
    }
}

插件注册方式:每个插件模块使用 @AutoService 注解,编译时自动在 META-INF/services/ 下生成 SPI 配置文件,无需手动维护

三、数据源插件详细流程

3.1 接口层次结构

scss 复制代码
DataSourceChannelFactory (SPI入口)
  └── getName()          ← 插件唯一名称,如 "MYSQL"
  └── create()           ← 创建 DataSourceChannel

DataSourceChannel (通道)
  └── createAdHocDataSourceClient()    ← 创建临时连接客户端
  └── createPooledDataSourceClient()   ← 创建连接池客户端

DataSourceClient (基础接口)
  └── getConnection(): Connection

PooledDataSourceClient extends DataSourceClient
  └── createDataSourcePool()           ← 创建 HikariCP 连接池

3.2 以 MySQL 为例的完整实现链

java 复制代码
MySQLDataSourceChannelFactory          ← @AutoService 注册
  └── create() → MySQLDataSourceChannel
        └── createPooledDataSourceClient() → MySQLPooledDataSourceClient
              └── extends BasePooledDataSourceClient
                    └── createDataSourcePool() → HikariDataSource
                          ├── setDriverClassName()
                          ├── setJdbcUrl()
                          ├── setUsername() / setPassword()
                          ├── setMinimumIdle() / setMaximumPoolSize()
                          └── setConnectionTestQuery()

四、Task 插件详细流程

4.1 接口层次结构

scss 复制代码
TaskChannelFactory (SPI入口) extends UiChannelFactory, PrioritySPI
  └── getName()          ← 插件类型名,如 "SHELL"
  └── create()           ← 创建 TaskChannel
  └── getParams()        ← 返回 UI 配置参数(前端渲染用)

TaskChannel (通道)
  └── createTask(TaskExecutionContext) → AbstractTask
  └── parseParameters(ParametersNode) → AbstractParameters
  └── getResources(parameters)        → ResourceParametersHelper
  └── cancelApplication(boolean)

AbstractTask (任务执行基类)
  └── init()             ← 初始化
  └── handle(callback)   ← 执行(抽象)
  └── cancel()           ← 取消(抽象)
  └── getExitStatus()    ← 根据 exitCode 返回状态

4.2 以 Shell 为例的完整实现链

sql 复制代码
ShellTaskChannelFactory                ← @AutoService 注册
  └── getName() → "SHELL"
  └── getParams() → [nodeName, runFlag, ...]  ← 前端UI参数
  └── create() → ShellTaskChannel
        └── createTask(ctx) → ShellTask
              └── handle(callback)
                    └── ShellCommandExecutor.run(shellActuatorBuilder)
                          └── 执行 Shell 脚本进程

五、两种插件的关键对

相关推荐
棒槌开发师1 小时前
动态组件设计(elpis)
架构
用户298698530145 小时前
Java 实现 Word 文档文本查找与高亮标注
java·后端
得物技术6 小时前
从表单到 Agent:得物社区活动搭建的 AI 实践之路
人工智能·架构·agent
Ausra无忧6 小时前
记录在公司把单服务器升级成多服务器架构流程
前端·后端·架构
宇宙之一粟6 小时前
乐企版式文件生成平台
java·后端·python
不好听6137 小时前
拆解 LLM Tool Use 的完整机制:从缸中大脑到 Agent 觉醒
架构·llm·agent
plainGeekDev7 小时前
MVC 写法 → MVVM
android·java·kotlin
starsstreaming7 小时前
200K 的窗口,跑完 400K 的任务:Claude Code 上下文压缩机制全拆解
架构