星图|投稿任务对接

KwaiShop

Posted by Claire on August 4, 2023

一、什么是星图?

抖音星图是抖音电商蓝图下,为品牌方、MCN公司和明星/达人服务并收取分成,在这可以实现订单接收、签约管理、项目汇总、数据查看的平台,当下比较多的使用就是一些MCN公司为旗下艺人揽活的。

商户可以在平台按照需求发布任务,那么任务有分一口价任务、按转化结算的任务,本文要涵盖的是对接按线索转化类任务。

二、什么是线索转化?

线索转化的逻辑就是:星图随巨量广告投放的时候,会添加埋点参数,根据需求对接并追踪埋点参数,在符合逻辑的业务节点回调星图,标记为一个线索。

三、对接中的一些疑问?

对接过程中遇到过很多问题,比如:


Q:对于混杂的文档,都没讲全,版本以及更新时间不一致,如何获取的最新版本对接文档? A:联系直客,直客提供的文档下可以提问,会有星图值班人员回复,加上飞书后可以更仔细得咨询。 不过直客也不一定靠谱,还是最终最好找到星图侧的人。 这是我当前获取到最新的对接文档:https://bytedance.feishu.cn/docx/WOQRd9htbohMfrxDLxrctbznn5g (这个文档一开始看没有权限复制有点坑,要申请加白才能复制)


Q:文档上说的转化ID是巨量老版本转化追踪吗? A:是的。


Q:老版本转化追踪都已经迁移切换,都不推荐使用,无人技术支撑了,怎么办? A:星图回传转化ID如果已经不能在巨量平台完成联调测试,那么请直客加白(或星图值班号),加白后可以无需转化ID。


Q:星图的回传会影响巨量的回传吗? A:不会,他们回传的clickid类型不同,注意区分即可。


Q:星图回传接口与巨量回传接口是需要相同还是可以不同? A:星图与巨量的回传可以理解为完全不相干的两个流程,可能就是老接口一致会给人混淆。 巨量当前老版本转化追踪不提供技术支撑了(工单即使反馈问题也解决不了),那必然就是建议使用新接口进行回传。巨量的回传为了优化投流策略,效果追踪。 星图全部都是要求老接口的,那么就是走老接口回传。星图的回传是为了线索类结算。


Q:巨量旧版本转化追踪进行联调,提示“该用户id不存在” A:请直客对你们的广告账户加白后就不需要依赖填写转化ID了,也就不需要做这个联调了。这个错误提示基本就是不允许进行旧版本的联调测试了,巨量侧不支持了。


通过以上这些含老血踩出来的经验,希望能给你理清一些思路。 那么下面就是用巨量转化追踪老接口,根据星图的需求,回传指标数据即可。 以下是根据小程序、落地页投放,API回传,进行表单提交线索转化结算的流程(如果是应用类SDK回传的再自行研究吧)。

四、如何对接开发?

通常为打开落地页/小程序时,落地页url上携带的clickid参数(这是最核心的,可能部分投放还带有其它参数),可将clickid参数放在如下demo中的callback=xxx中,完成数据回传。参数埋点是如何获取,并传递到后端服务的,这边就不详细描述了。

这个clickid在实际投放巨量广告的时候能够看到,巨量绑定的clickid是“E…”开头的,需要注意星图绑定的clickid是“B…”开头的。所以在做巨量回传和星图回传的时候注意区分clickid。

按照以上思路呢,基本已经获取到了星图侧传递过来的clickid了,那么下面看如何回传与测试。

继续参考这个文档:https://bytedance.feishu.cn/docx/WOQRd9htbohMfrxDLxrctbznn5g

回传事件类型:

  • event_type=0表示激活
  • event_type=1表示注册
  • event_type=2表示付费,对于按付费分佣结算(CPS),必须回传付费金额,可在props中增加pay_amount参数,单位为分,需通过unicode编码
  • event_type=3表示表单提交,投稿任务需回传加密的手机号进行结算,因此需增加phone_num参数。注意手机号为高敏感信息,有单独密钥
  • event_type=72,表示无效线索,可通过 props 的 invalid_type 字段回传无效线索的类型,需对props进行encode,并放到URL中
    • repeat_form_submission,重复提交表单(同地址/同手机号/同身份证重复申请办理)。判断规则:30天内同地址/同手机号/同身份证重复申请办理2次及以上的表单
    • identity_verification,身份信息校验不匹配。判断规则:三方平台国证通判断身份证与姓名不匹配的表单
    • wrong_address,地址乱填。判断规则:字数较少(如6字以内)、明显为无效地址的表单
    • wrong_phone_number,空号/联系电话虚假。判断规则:通过三方平台外呼判断手机为空号的表单
    • address_verification,IP地址与收货地址不一致。判断规则:通过ip地址与收货地址进行校验后,匹配不成功的表单
    • map_verification,地图校验不一致。判断规则:通过高德地图系统校验,匹配不成功的表单
# 线索场景
https://ad.oceanengine.com/track/activate/?callback=xxxx&event_type=3&phone_num=xxxx

# 激活场景
https://ad.oceanengine.com/track/activate/?callback=xxxx&event_type=0

# 注册场景
https://ad.oceanengine.com/track/activate/?callback=xxxx&event_type=1

# 付费场景
https://ad.oceanengine.com/track/activate/?callback=xxxx&event_type=2&props=%7B%22pay_amount%22:%20%221000%22%7D

测试采用表单数据回传方式,那么需要增加phone_num字段且需要加密

  • java方式,以下每次输出结果均会不同
public static void main(String[] args) throws Exception {
    String encryptNum = generatePhoneNum("13312345678");
}

private static String generatePhoneNum(String phoneNum) throws Exception {
    String publicKeyStr = "-----BEGIN PUBLIC KEY-----\n" +
            "MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCmnVrcCkHV8b+2BaMSC95DY0z9\n" +
            "9B7MgFoHMxXaZn0k6fvUXJTgxXtYleGFdoifHSVsrJIB0P1V4hazTIyDLnHGFOCp\n" +
            "UnYkLE2L8M/l0msAiKmShlxQAPB9IBNLOLFcp3qubLjfGGB1xfsDXHY9dAxYY8XC\n" +
            "edrYgXwkAoi3/dL+UQIDAQAB\n" +
            "-----END PUBLIC KEY-----";
    String publicKeyPem = publicKeyStr.replace("-----BEGIN PUBLIC KEY-----", "")
            .replaceAll("\\n", "")
            .replace("-----END PUBLIC KEY-----", "");
    byte[] keyContentAsBytes = Base64.getDecoder().decode(publicKeyPem);
    KeyFactory fact = KeyFactory.getInstance("RSA");
    X509EncodedKeySpec pubKeySpec = new X509EncodedKeySpec(keyContentAsBytes);
    PublicKey publicKey = fact.generatePublic(pubKeySpec);
    String phoneNumAfterEncrypt = encrypt(phoneNum, Base64.getEncoder().encodeToString(publicKey.getEncoded()));
    System.out.println(phoneNumAfterEncrypt);
    String encodePhoneNumAfterEncrypt = URLEncoder.encode(phoneNumAfterEncrypt, StandardCharsets.UTF_8.toString());
     System.out.println(encodePhoneNumAfterEncrypt);
    return encodePhoneNumAfterEncrypt;
}

public static String encrypt(String str, String publicKey) throws Exception {
    byte[] decoded = Base64.getDecoder().decode(publicKey);
    RSAPublicKey pubKey = (RSAPublicKey) KeyFactory.getInstance("RSA").generatePublic(new X509EncodedKeySpec(decoded));
    Cipher cipher = Cipher.getInstance("RSA");
    cipher.init(Cipher.ENCRYPT_MODE, pubKey);
    return Base64.getEncoder().encodeToString(cipher.doFinal(str.getBytes(StandardCharsets.UTF_8)));
}
  • python方式,版本使用3.7,官方提供demo未描述python版本无法跑通,以下每次输出结果均会不同
# python3.7环境下安装:
# pip install pycryptodome --user

from Cryptodome.PublicKey import RSA
from Cryptodome.Cipher import PKCS1_v1_5 as PKCS1_cipher
import base64

def gen_num(phone_num):
    # 这个PEM是有格式的,不要改动,否则报错:ValueError: Not a valid PEM pre boundary
    pub_key = RSA.importKey("""-----BEGIN PUBLIC KEY-----
MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCmnVrcCkHV8b+2BaMSC95DY0z9
9B7MgFoHMxXaZn0k6fvUXJTgxXtYleGFdoifHSVsrJIB0P1V4hazTIyDLnHGFOCp
UnYkLE2L8M/l0msAiKmShlxQAPB9IBNLOLFcp3qubLjfGGB1xfsDXHY9dAxYY8XC
edrYgXwkAoi3/dL+UQIDAQAB
-----END PUBLIC KEY-----""")
    cipher = PKCS1_cipher.new(pub_key)
    rsa_text = base64.b64encode(cipher.encrypt(bytes(phone_num.encode("utf8"))))
    return rsa_text.decode('utf-8')

if __name__=='__main__':
    print(gen_num("13312345678")) # RhLLAnzd7wQx0w5T+5Lzn6gie+FlVrDfP3dqohxDn/OVsDmEiqbgtFDoIC6/ILY3TDgH3pmeqUy1QIh9X1xLPbiYQKe4mnujeAyh+G5Q7SvGb7GJLLdCsI2akCnv0joOuR6CxbGQ3qrWxhR7YjNWztqTp8KCVnO+gaYo3Qd7ndc=

五、星图侧如何联调测试?

联调工具只能测试落地页,可以用落地页测试来确认接口加解密逻辑的正确性。在星图管理后台,在我的任务 -> 工具 -> 分析工具 -> 监测联调 -> 线索回传联调。

  1. 填入落地页链接
  2. 点击落地页链接,在星图页面内直接跳转
  3. 转化回传数据至指定地址,这个xingtu域名的是测试地址,正式的采用ad巨量的旧转化回传地址

    回传需要除了需要区分出B开头的clickid,此外还需要注意:

    • callback需要base64后加上star-
    • event_type必须传3
    • phone_num是加密后的留资手机号,phone_num需要转义(这一步上面demo中已经包含)
  4. 确保123步都操作完成后,点击“查看联调结果”

若显示“成功收到1条线索,解析成功1条线索”,即代表流程畅通,可正常在星图下单开始任务

正式回传样例

https://ad.oceanengine.com/track/activate/?callback=star-9e9adf4383d907c42921d3ae539370f2&event_type=3&phone_num=ChOInt8EQsQqIi8FAcstJraXIkdQvJ2jpWOQCsNI7wDaxQtggHzVOLoSGtHG25SN9CF5svCHpyQbR/ha2msOLZl1rx4PYas3hdsmu7mB9phCr8G+Q+lSaUcl30kc5SabRgJDHHEXCJ5RAkd5FHSttWGHl8ZgbgGxeIH86+Cs4Vo=

接收成功的返回值:

{
 "code":0,
 "ret":0,
 "msg":"success"
}