更新时间:2025-12-23 17:29:06
| 日期 | 版本 | 修订内容摘要 |
|---|---|---|
| 2023-09-07 | v1.0.0 | 更新接口文档 |
| 2025-05-08 | v2.0.0 | 启用行为验证码V2 JS地址 |
创蓝云智注册账号,进行认证,并激活,之后在【应用管理】创建应用
以下代码示例,单击验证,激活验证码,并弹窗展示验证结果。
注意 该示例未展示调用票据校验 API 的逻辑。业务客户端完成验证码接入后,业务服务端需二次核查验证码票据结果(未接入票据校验,会导致黑产轻易伪造验证结果,失去验证码人机对抗效果),详情请参见:【服务端接入>Web 及 App接入】。
复制成功```<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Web 前端接入示例</title> <!-- 验证码程序依赖(必须)。请勿修改以下程序依赖,如通过其他手段规避加载,会导致验证码无法正常更新,对抗能力无法保证,甚至引起误拦截。 --> <script src="https://captcha.253.com/TJCaptcha.js"></script> </head> <body> <button id="CaptchaId" type="button">验证</button> </body> <script> // 定义回调函数 function callback(res) { // 第一个参数传入回调结果,结果如下: // ret Int 验证结果,0:验证成功。2:用户主动关闭验证码。 // ticket String 验证成功的票据,当且仅当 ret = 0 时 ticket 有值。 // CaptchaAppId String 验证码应用ID。 // bizState Any 自定义透传参数。 // randstr String 本次验证的随机串,后续票据校验时需传递该参数。 // verifyDuration Int 验证码校验接口耗时(ms)。 // actionDuration Int 操作校验成功耗时(用户动作+校验完成)(ms)。 // sid String 链路sid。 console.log('callback:', res); // res(用户主动关闭验证码)= {ret: 2, ticket: null} // res(验证成功) = {ret: 0, ticket: "String", randstr: "String"} // res(请求验证码发生错误,验证码自动返回trerror_前缀的容灾票据) = {ret: 0, ticket: "String", randstr: "String", errorCode: Number, errorMessage: "String"} // 此处代码仅为验证结果的展示示例,真实业务接入,建议基于ticket和errorCode情况做不同的业务处理 if (res.ret === 0) { // 复制结果至剪切板 var str = '【randstr】->【' + res.randstr + '】 【ticket】->【' + res.ticket + '】'; var ipt = document.createElement('input'); ipt.value = str; document.body.appendChild(ipt); ipt.select(); document.execCommand("Copy"); document.body.removeChild(ipt); alert('1. 返回结果(randstr、ticket)已复制到剪切板,ctrl+v 查看。2. 打开浏览器控制台,查看完整返回结果。'); } } // 定义验证码js加载错误处理函数 function loadErrorCallback() { var appid = '您的CaptchaAppId'; // 生成容灾票据或自行做其它处理 var ticket = 'trerror_1001_' + appid + '_' + Math.floor(new Date().getTime() / 1000); callback({ ret: 0, randstr: '@' + Math.random().toString(36).substr(2), ticket: ticket, errorCode: 1001, errorMessage: 'jsload_error' }); } // 定义验证码触发事件 window.onload = function () { document.getElementById('CaptchaId').onclick = function () { try { // 生成一个验证码对象 // CaptchaAppId:登录验证码控制台,从【验证管理】页面进行查看。如果未创建过验证,请先新建验证。注意:不可使用客户端类型为小程序的CaptchaAppId,会导致数据统计错误。 //callback:定义的回调函数 var captcha = new TencentCaptcha('您的验证码CaptchaAppId', callback, { userLanguage: 'zh-cn', showFn: (ret) => { const { duration, // 验证码渲染完成的耗时(ms) sid, // 链路sid } = ret; }, }); // 调用方法,显示验证码 captcha.show(); } catch (error) { // 加载异常,调用验证码js加载错误处理函数 loadErrorCallback(); } } } </script> </html>
Web 页面需动态引入验证码 JS,在业务需要验证时,唤起验证码进行验证。
复制成功<!-- 动态引入验证码JS示例 --> <script src="https://captcha.253.com/TJCaptcha.js"></script>
注意 必须动态引入验证码 JS。如通过其他手段规避动态加载,会导致验证码无法正常更新,对抗能力无法保证,甚至引起误拦截。 如果使用行为验证码V1,请将JS地址改为https://captcha.253.com/TCaptcha.js, 其他无需变更。
引入验证码 JS 后,验证码会在全局注册一个TencentCaptcha类,业务方可以使用这个类自行初始化验证码,并对验证码进行显示或者隐藏。
注意 触发验证码的元素不要使用id="TencentCaptcha",TencentCaptcha 属于系统默认 id,用来兼容验证码旧接入方式。
构造函数
复制成功new TencentCaptcha(CaptchaAppId, callback, options);
参数说明
| 参数名 | 值类型 | 说明 |
|---|---|---|
| CaptchaAppId | String | 行为验证码 CaptchaAppId:登录 控制台,在行为验证码概览页面进行查看。如果未创建过应用,请先新建应用。 注意:不可使用客户端类型为小程序的 CaptchaAppId,会导致数据统计错误。 |
| callback | Function | 行为验证码回调函数,详情请参见 callback 回调函数。 |
| options | Object | 验证码外观配置参数, 详情请参见 options 外观配置参数。 |
验证结束后,会调用业务传入的回调函数,并在第一个参数中传入回调结果。回调结果字段说明如下:
| 字段名 | 值类型 | 说明 |
|---|---|---|
| ret | Int | 验证结果,0:验证成功。2:用户主动关闭验证码。说明:容灾场景下验证结果返回:0,详情请参见 业务容灾方案(Web 及 App)。 |
| ticket | String | 验证成功的票据,当且仅当 ret = 0 时 ticket 有值。 |
| appid | String | 验证码应用 ID。 |
| bizState | Any | 自定义透传参数。 |
| randstr | String | 本次验证的随机串,后续票据校验时需传递该参数。 |
| errorCode | Number | 错误 code ,详情请参见 回调函数 errorCode 说明。 |
| errorMessage | String | 错误信息。 |
| verifyDuration | Number | 验证码校验接口耗时(ms) |
| actionDuration | Number | 用户操作校验成功耗时(ms) |
| sid | String | 链路 sid |
| errorCode | 说明 |
|---|---|
| 1001 | TJCaptcha.js 加载错误 |
| 1002 | 调用 show 方法超时 |
| 1003 | 中间 js 加载超时 |
| 1004 | 中间 js 加载错误 |
| 1005 | 中间 js 运行错误 |
| 1006 | 拉取验证码配置错误/超时(网络超时,欠费,CaptchaAppid 加密配置错误) |
| 1007 | iframe 加载超时 |
| 1008 | iframe 加载错误 |
| 1009 | jQuery 加载错误 |
| 1010 | 滑块 js 加载错误 |
| 1011 | 滑块 js 运行错误 |
| 1012 | 刷新连续错误3次 |
| 1013 | 验证网络连续错误3次 |
| 1085 | 无感验证超时/失败 |
options 参数用于对验证码进行定制外观设置,默认可以设置为空。
注意
复制成功.tcaptcha-transform{ transform: scale(0.9); }
| 配置名 | 值类型 | 说明 |
|---|---|---|
| bizState | Any | 自定义透传参数,业务可用该字段传递少量数据,该字段的内容会被带入 callback 回调的对象中。 |
| enableDarkMode | Boolean/String | 开启自适应深夜模式或强制深夜模式。(VTT 空间语义验证暂不支持该功能)开启自适应深夜模式: {"enableDarkMode": true}强制深夜模式: {"enableDarkMode": 'force'} |
| sdkOpts | Object | 示例 {"width": 140, "height": 140}仅支持移动端原生 webview 调用时传入,用来设置验证码 loading 加载弹窗的大小(注意,并非验证码弹窗大小)。 |
| ready | Function | 验证码加载完成的回调,回调参数为验证码实际的宽高(单位:px):{"sdkView": {"width": number,"height": number}}该参数仅为查看验证码宽高使用,请勿使用此参数直接设定宽高。 |
| needFeedBack | Boolean /String | 隐藏帮助按钮或自定义帮助按钮链接。(VTT 空间语义验证暂不支持自定义链接) 隐藏帮助按钮: {"needFeedBack": false }自定义帮助链接: {"needFeedBack": 'url地址' } |
| loading | Boolean | 是否在验证码加载过程中显示loading框。不指定该参数时,默认显示loading框。显示 loading 框: {"loading": true}不显示 loading 框: {"loading": false} (展示方式为嵌入式时不支持配置) |
| userLanguage | String | 指定验证码提示文案的语言,优先级高于控制台配置。(VTT 空间语义、文字点选验证暂不支持语言配置)支持传入值同 navigator.language 用户首选语言,大小写不敏感。详情参见 userLanguage 配置参数。 |
| type | String | 定义验证码展示方式。popup(默认)弹出式,以浮层形式居中弹出展示验证码。embed 嵌入式,以嵌入指定容器元素中的方式展示验证码。详情参见 嵌入式验证码参数配置示例。 |
| aidEncrypted | String | CaptchaAppId 加密校验串,可选参数。详情见 CaptchaAppid 加密校验能力接入指引。 |
| showFn | Function | duration 渲染耗时 + sid 回调函数。 |
嵌入式验证码参数配置示例
复制成功<!doctype html> <html lang="en"> <head> <meta charset="UTF-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <title>验证码-内嵌接入</title> <script src="https://captcha.253.com/TJCaptcha.js"></script> </head> <body> <!-- 验证码指定的内嵌容器 可自定义到任何位置 --> <div id="targetEmbed"></div> <script> function globalCallback(res) { console.log('captcha success', res); } function errorCallback(res) { console.log('errorCallback', res); } window.onload = function () { try { const captcha = new TencentCaptcha(document.getElementById('targetEmbed'), '用户CaptchaAppid', globalCallback, { type: 'embed', }); captcha.show(); } catch (error) { console.log('error', error); errorCallback(error); } }; </script> </body> </html>
| 参数名 | 说明 |
|---|---|
| zh-cn | 简体中文 |
| zh-hk | 繁体中文(中国香港) |
| zh-tw | 繁体中文(中国台湾) |
| en | 英文 |
| ar | 阿拉伯语 |
| cs-cz | 捷克语(捷克) |
| de | 德语 |
| en-in | 英语(印度) |
| es | 西班牙语 |
| es-la | 西班牙语(拉丁美洲) |
| fil | 菲律宾语 |
| fr | 法语 |
| he | 希伯来语 |
| hi | 印地语 |
| id | 印尼语 |
| it | 意大利语 |
| ja | 日语 |
| ko | 朝鲜语 / 韩语 |
| lo | 老挝语 |
| ms | 马来语 |
| my | 缅甸语 |
| nl | 荷兰语 |
| pl | 波兰语 |
| pt | 葡萄牙语 |
| pt-pt | 葡萄牙语(葡萄牙) |
| ro-ro | 罗马尼亚语(罗马尼亚) |
| ru | 俄语 |
| th | 泰语 |
| tr | 土耳其语 |
| uk-ua | 乌克兰语(乌克兰) |
| ur | 乌尔都语 |
| vi | 越南语 |
操作验证码的常用方法:
| 方法名 | 说明 | 传入参数 | 返回内容 |
|---|---|---|---|
| show | 显示验证码,可以反复调用。 | 无 | 无 |
| destroy | 隐藏验证码,可以反复调用。 | 无 | 无 |
| getTicket | 获取验证成功后的 ticket。 | 无 | Object:{"CaptchaAppId":"","ticket":""} |
为保障验证码 Captcha 服务端异常时不阻塞客户网站正常业务流程,建议参考如下方式接入验证码。
复制成功// 错误处理函数作用:在脚本加载或初始化错误时,保障事件流程正常 // 函数定义需在script加载前 function loadErrorCallback() { var appid = '' // 生成容灾票据或自行做其它处理 var ticket = 'trerror_1001_' + appid + Math.floor(new Date().getTime() / 1000); callback({ ret: 0, randstr: '@'+ Math.random().toString(36).substr(2), ticket:ticket, errorCode: 1001, errorMessage: 'jsload_error', }); }
复制成功try { // 生成一个验证码对象 var captcha = new TencentCaptcha('您的验证码CaptchaAppId', callback, {}); // 调用方法,显示验证码 captcha.show(); } catch (error) { // 加载异常,调用验证码js加载错误处理函数 loadErrorCallback(); }
复制成功function callback(res) { // res(用户主动关闭验证码)= {ret: 2, ticket: null} // res(验证成功) = {ret: 0, ticket: "String", randstr: "String"} // res(请求错误,返回trerror_前缀的容灾票据) = {ret: 0, ticket: "String", randstr: "String", errorCode: Number, errorMessage: "String"} if (res.ticket){ //根据errorCode情况做特殊处理 if(res.errorCode === xxxxx){ //自定义容灾逻辑(例如跳过这次验证) } } }
业务客户端完成验证码接入后,服务端需二次核查验证码票据结果(未接入票据校验,会导致黑产轻易伪造验证结果,失去验证码人机对抗效果),详情请参见:服务端接入>Web 客户端接入。
复制成功<!-- 验证码程序依赖(必须)。请勿修改以下程序依赖,如使用本地缓存,或通过其他手段规避加载,会导致验证码无法正常更新,对抗能力无法保证,甚至引起误拦截。 --> <script src="https://captcha.253.com/TJCaptcha.js"></script>
复制成功import React from 'react'; import logo from './logo.svg'; import './App.css'; interface ICaptchaResult { ret: number; ticket: string; randstr: string; CaptchaAppId?: string; bizState?: string; errorCode?: number; errorMessage?: string; } function App() { // 定义回调函数 function callback(res: ICaptchaResult) { // 第一个参数传入回调结果,结果如下: // ret Int 验证结果,0:验证成功。2:用户主动关闭验证码。 // ticket String 验证成功的票据,当且仅当 ret = 0 时 ticket 有值。 // CaptchaAppId String 验证码应用ID。 // bizState Any 自定义透传参数。 // randstr String 本次验证的随机串,后续票据校验时需传递该参数。 console.log('callback:', res); // res(用户主动关闭验证码)= {ret: 2, ticket: null} // res(验证成功) = {ret: 0, ticket: "String", randstr: "String"} // res(请求验证码发生错误,验证码自动返回trerror_前缀的容灾票据) = {ret: 0, ticket: "String", randstr: "String", errorCode: Number, errorMessage: "String"} // 此处代码仅为验证结果的展示示例,真实业务接入,建议基于ticket和errorCode情况做不同的业务处理 if (res.ret === 0) { // 复制结果至剪切板 var str = '【randstr】->【' + res.randstr + '】 【ticket】->【' + res.ticket + '】'; var ipt = document.createElement('input'); ipt.value = str; document.body.appendChild(ipt); ipt.select(); document.execCommand('Copy'); document.body.removeChild(ipt); alert('1. 返回结果(randstr、ticket)已复制到剪切板,ctrl+v 查看。 2. 打开浏览器控制台,查看完整返回结果。'); } } // 定义验证码js加载错误处理函数 function loadErrorCallback() { var appid = '您的CaptchaAppId'; // 生成容灾票据或自行做其它处理 var ticket = 'trerror_1001_' + appid + '_' + Math.floor(new Date().getTime() / 1000); callback({ ret: 0, randstr: '@' + Math.random().toString(36).substr(2), ticket: ticket, errorCode: 1001, errorMessage: 'jsload_error', }); } function onCaptchaShow() { try { // 生成一个验证码对象 // CaptchaAppId:登录验证码控制台,从【验证管理】页面进行查看。如果未创建过验证,请先新建验证。注意:不可使用客户端类型为小程序的CaptchaAppId,会导致数据统计错误。 // callback:定义的回调函数 const captcha = new TencentCaptcha('您的CaptchaAppId', callback, {}); // 调用方法,显示验证码 captcha.show(); } catch (error) { // 加载异常,调用验证码js加载错误处理函数 loadErrorCallback(); } } return ( <div className='App'> <button className='captcha-btn' onClick={onCaptchaShow}> 弹出验证码 </button> </div> ); } export default App;
调用验证码实例方法
TencentCaptcha 的实例提供一些操作验证码的常用方法:
| 方法名 | 说明 | 传入参数 | 返回内容 |
|---|---|---|---|
| show | 显示验证码,可以反复调用。 | 无 | 无 |
| destroy | 隐藏验证码,可以反复调用。 | 无 | 无 |
| getTicket | 获取验证成功后的 ticket。 | 无 | Object:{"CaptchaAppId":"","ticket":""} |
通过前端传递加密符(非必选能力,可根据安全性需求选择性接入),可以有效防止因 CaptchaAppid 泄露而造成的资源盗刷。 加密规则 接口通过 aidEncrypted 参数(即加密后的字符串标识),支持采用加密模式传递验证码业务 CaptchaAppid 进行校验。 当控制台强制校验开启时,触发加密模式。由客户的服务端进行加密后下发对应加密字符串到客户的前端,由客户前端将加密后的字符串传参到验证码侧。 加密步骤如下:

加密结果示例
| 类型 | 示例值 |
|---|---|
| CaptchaAppid | 123456789 |
| 时间戳 | 1710144972 |
| 过期时间 | 86400秒 |
| iv | 0123456789012345 |
| AppSecretKey | 1234567891011121314151516 |
| 循环填充的密钥 key | 12345678910111213141515161234567 |
| 加密的业务数据对象 | 123456789&1710144972&86400 |
| 加密后的最终字符串 aidEncrypted | MDEyMzQ1Njc4OTAxMjM0NWvZ11atw+1uzYmoIyt5rAQVPyMK9ZDavskPw5hcayeT |
代码示例
服务端加密 加密规则为:Base64(IV + AES256(CaptchaAppid&时间戳&密文过期时间,IV, Key) ),即使用随机生成16字节的 IV、及根据 AppSecretKey 循环填充后得到32字节的加密 Key,使用 AES256算法加密模式 CBC/PKCS7Padding,对明文数据 CaptchaAppid&时间戳&密文过期时间进行加密。
复制成功#!/usr/bin/env python # -*- coding: utf-8 -*- from Crypto.Cipher import AES from Crypto.Util.Padding import pad, unpad import base64 import time def encrypt(plaintext, key, iv): cipher = AES.new(key, AES.MODE_CBC, iv) # 创建一个新的AES cipher,CBC模式 ciphertext = cipher.encrypt(pad(plaintext.encode(), AES.block_size))# 对数据进行填充,然后加密。pad(plaintext.encode(), AES.block_size)将检查明文长度是否为16字节倍数,若非16字节倍数,将使用PKCS7填充方式将明文填充到16字节倍数。 ciphertextBase64 = base64.b64encode(iv + ciphertext).decode('utf-8') # iv拼接加密后的数据,并进行Base64返回后进行传输。 return ciphertextBase64 # 加密示例 AppSecretKey = b'1234567891011121314151516' #客户从控制台获取对应验证码账号下的AppSecretKey,25位 remainder = 32 % AppSecretKey.__len__() # 计算需要补充的密钥长度 key = AppSecretKey + AppSecretKey[:remainder] # 最终加密key,补充满32位。 Captchaappid = "123456789" # 客户自身的验证码CaptchaAppid curTime = 1710144972 # 获取当前的时间戳,示例暂设置成固定时间戳,客户应该设置成最新的时间戳,使用int(time.time()) expireTime = 86400 # 过期时间设置,这里暂设置成该值,客户根据自身需要设置 plaintext = Captchaappid + "&" + str(curTime) + "&" + str(expireTime) # 拼接待加密的业务数据对象,CaptchaAppid&时间戳&密文过期时间 iv = "0123456789012345".encode() # 随机生成16字节的IV,这里暂设置成该值,客户使用时应该用随机生成的数据,使用os.urandom(16) ciphertext = encrypt(plaintext, key, iv) # 加密 print("Ciphertext (Base64):", ciphertext) # 本示例数据将输出MDEyMzQ1Njc4OTAxMjM0NWvZ11atw+1uzYmoIyt5rAQVPyMK9ZDavskPw5hcayeT
前端接入示例
复制成功const encryptAppid = async () => { /** 从后端获取加密后的 CaptchaAppid字符串 */ const { aidEncrypted } = await fetch('/api/encryptAppid'); /** 回调函数 */ const callBack = (ret) => { console.log('ret', ret); }; /** 错误回调函数 */ const errorCb = (error) => { console.log('error', error); }; try { /** 将获取的加密字符串传入aidEncrypted参数 */ const captcha = new TencentCaptcha('123456789', callBack, { aidEncrypted: aidEncrypted }); captcha.show(); } catch (error) { errorCb(error); } };