钩子规范bak

钩子服务本质上是一个实现了飞布钩子规范的WEB服务,可以用任意后端语言实现。

如果你希望实现其他语言的 hook SDK,需要遵从如下协议。

根据用途划分,钩子可分为4大类:局部钩子、全局钩子、授权钩子、文件钩子。

局部钩子(OPERATION钩子)

局部钩子目的是扩展OPEARTION的能力,分别在“OPEARTION执行”前后执行,主要用途是参数校验和副作用触发,如创建文章后发送邮件通知审核。

详情见如下流程图。

前置钩子在 "执行OPERATION"前执行,可校验参数或修改输入参数。

// 自定义参数校验
func PreResolve(hook *base.HookRequest, body generated.Todo__CreateOneTodoBody) (res generated.Todo__CreateOneTodoBody, err error) {
    if body.Input.Title == "" {
	return nil, errors.New("标题不能为空")
    }
    return body, nil
}

后置钩子在 "执行OPERATION" 后执行,可触发自定义操作或修改响应结果。

// 执行自定义操作
func PostResolve(hook *base.HookRequest, body generated.Todo__CreateOneTodoBody) (res generated.Todo__CreateOneTodoBody, err error) {
    fmt.Println("我要发一封邮件xxx,标题是:", body.Input.Title, data)
    return body, nil
}

除了上述局部钩子,还有两个特殊的局部钩子:自定义处理钩子和模拟钩子。

// 若该钩子有返回值,那么将跳过“执行OPERATION”,直接返回当前钩子的返回值 
func CustomResolve(hook *base.HookRequest, body generated.$HOOK_NAME$Body) (res generated.$HOOK_NAME$Body, err error) {
    hook.Logger().Info("CustomResolve")
    return body, nil
}

全局钩子

全局钩子目的是改写原有的request和response,分别在请求最初、opertaion触发前后,可以用来实现全局的参数/响应改写或发送全局的通知,包括预执行、前置、后置钩子。


// 在最初请求,可以修改body和header(请使用OriginBody)
func BeforeOriginRequest(hook *base.HttpTransportHookRequest, body *plugins.HttpTransportBody) (*base.ClientRequest, error) {
    return modifyForPayNotify(body)
}

func modifyForPayNotify(body *plugins.HttpTransportBody) (*base.ClientRequest, error) {
    u, err := url.Parse(body.Request.RequestURI)
    if err != nil {
	return nil, err
    }

    if u.Path != "/operations/Payment/PayNotify" {
	return body.Request, nil
    }

    // 1. 从body中取值
    modifyBody, err := sjson.Set("{}", "data", string(body.Request.OriginBody))
    if err != nil {
	return nil, err
    }

    // 2. 从url中取值
    for key, valArr := range u.Query() {
	modifyBody, _ = sjson.Set(modifyBody, key, valArr[0])
    }
    body.Request.Body = []byte(modifyBody)
    return body.Request, nil
}

授权钩子

认证钩子目的是在认证成功后执行自定义操作,可以用来实现用户信息同步,用户信息改写,用户重新校验(需要请求携带参数revalidate)

// 在认证后做自定义处理,比如同步用户信息等
func PostAuthentication(hook *base.AuthenticationHookRequest) error {
    hook.Context.Logger().Infof("用户%s已同步", hook.User.NickName)
    return nil
}

文件钩子

文件钩子目的是在文件上传前后进行自定义操作,可以用来实现重写文件名和自定义处理

// 改写上传后的文件名,默认随机字符串
func PreUpload(request *base.PreUploadHookRequest, body *plugins.UploadBody[any]) (*base.UploadHookResponse, error) {
    return &base.UploadHookResponse{FileKey: body.File.Name}, nil
}

数据代理

飞布服务不仅可以按照约定调用钩子服务,钩子也可以调用其它钩子,此时飞布服务变身为数据代理。

最后更新于