├── demo-album
├── app.js
├── pages
│ ├── album
│ │ ├── album.json
│ │ ├── album.wxml
│ │ ├── album.wxss
│ │ └── album.js
│ ├── preview
│ │ ├── preview.json
│ │ ├── preview.wxml
│ │ ├── preview.wxss
│ │ └── preview.js
│ └── index
│ │ ├── index.json
│ │ ├── index.wxml
│ │ └── index.js
├── lib
│ ├── api.js
│ ├── request.js
│ ├── cos.js
│ └── util.js
├── images
│ ├── wave.png
│ └── camera.png
├── app.wxss
├── config.js
├── sitemap.json
├── .gitignore
├── app.json
└── project.config.json
├── demo
├── pages
│ └── index
│ │ ├── index.json
│ │ ├── index.wxss
│ │ ├── index.wxml
│ │ └── index.js
├── config.js
├── sitemap.json
├── app.wxss
├── app.json
├── demo-ci.js
├── app.js
├── project.config.json
├── ciDemo
│ ├── mediaProcess.js
│ ├── other.js
│ ├── audit.js
│ ├── asr.js
│ ├── docPreview.js
│ └── fileProcess.js
├── demo-post-policy.js
├── tools.js
├── lib
│ ├── cos-auth.js
│ └── beacon_mp.min.js
└── demo-sdk.js
├── .babelrc
├── index.js
├── .gitignore
├── .npmignore
├── .prettierrc
├── scripts
└── patch-check.js
├── patches
└── fast-xml-parser+4.5.0.patch
├── src
├── event.js
├── async.js
├── session.js
├── cos.js
├── task.js
└── tracker.js
├── LICENSE
├── package.json
├── .github
└── workflows
│ └── auto-changelog.yml
├── README.md
├── lib
├── request.js
├── base64.js
└── crypto.js
└── server
└── sts.js
/demo-album/app.js:
--------------------------------------------------------------------------------
1 | App({});
--------------------------------------------------------------------------------
/demo/pages/index/index.json:
--------------------------------------------------------------------------------
1 | {}
--------------------------------------------------------------------------------
/demo-album/pages/album/album.json:
--------------------------------------------------------------------------------
1 | {}
--------------------------------------------------------------------------------
/demo-album/lib/api.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | };
--------------------------------------------------------------------------------
/demo-album/pages/preview/preview.json:
--------------------------------------------------------------------------------
1 | {
2 | }
--------------------------------------------------------------------------------
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": ["@babel/preset-env"]
3 | }
--------------------------------------------------------------------------------
/index.js:
--------------------------------------------------------------------------------
1 | var COS = require('./src/cos');
2 | module.exports = COS;
--------------------------------------------------------------------------------
/demo-album/images/wave.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/tencentyun/cos-wx-sdk-v5/HEAD/demo-album/images/wave.png
--------------------------------------------------------------------------------
/demo-album/images/camera.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/tencentyun/cos-wx-sdk-v5/HEAD/demo-album/images/camera.png
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .vscode
2 | .idea
3 | node_modules
4 | package-lock.json
5 | .DS_Store
6 | */.wechatide
7 | project.private.config.json
8 |
--------------------------------------------------------------------------------
/demo-album/app.wxss:
--------------------------------------------------------------------------------
1 | page {
2 | background-color: #fff;
3 | }
4 | .main {
5 | position: relative;
6 | z-index: 2;
7 | margin: 40rpx 80rpx;
8 | }
--------------------------------------------------------------------------------
/demo/config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | stsUrl: 'https://example.com/sts.php',
3 | Bucket: 'test-1250000000',
4 | Region: 'ap-guangzhou',
5 | };
6 |
--------------------------------------------------------------------------------
/demo-album/pages/index/index.json:
--------------------------------------------------------------------------------
1 | {
2 | "navigationBarBackgroundColor": "#2277da",
3 | "navigationBarTextStyle": "white",
4 | "backgroundColor": "white"
5 | }
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | .idea/
2 | .vscode/
3 | .github/
4 | coverage/
5 | demo/
6 | demo-album/
7 | server/
8 | patches/
9 | .babelrc
10 | .prettierrc
11 | build.js
12 | CHANGELOG.md
--------------------------------------------------------------------------------
/demo-album/config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | stsUrl: 'https://example.com/sts.php',
3 | Bucket: 'test-1250000000',
4 | Region: 'ap-guangzhou',
5 | albumDir: 'album/',
6 | };
--------------------------------------------------------------------------------
/demo/sitemap.json:
--------------------------------------------------------------------------------
1 | {
2 | "desc": "关于本文件的更多信息,请参考文档 https://developers.weixin.qq.com/miniprogram/dev/framework/sitemap.html",
3 | "rules": [{
4 | "action": "allow",
5 | "page": "*"
6 | }]
7 | }
--------------------------------------------------------------------------------
/demo-album/sitemap.json:
--------------------------------------------------------------------------------
1 | {
2 | "desc": "关于本文件的更多信息,请参考文档 https://developers.weixin.qq.com/miniprogram/dev/framework/sitemap.html",
3 | "rules": [{
4 | "action": "allow",
5 | "page": "*"
6 | }]
7 | }
--------------------------------------------------------------------------------
/demo-album/.gitignore:
--------------------------------------------------------------------------------
1 | # Windows
2 | [Dd]esktop.ini
3 | Thumbs.db
4 | $RECYCLE.BIN/
5 |
6 | # macOS
7 | .DS_Store
8 | .fseventsd
9 | .Spotlight-V100
10 | .TemporaryItems
11 | .Trashes
12 |
13 | # Node.js
14 | node_modules/
15 |
--------------------------------------------------------------------------------
/.prettierrc:
--------------------------------------------------------------------------------
1 | {
2 | "printWidth": 120,
3 | "tabWidth": 2,
4 | "useTabs": false,
5 | "semi": true,
6 | "singleQuote": true,
7 | "jsxSingleQuote": true,
8 | "jsxBracketSameLine": true,
9 | "trailingComma": "es5"
10 | }
11 |
--------------------------------------------------------------------------------
/demo/app.wxss:
--------------------------------------------------------------------------------
1 | /**app.wxss**/
2 | .container {
3 | height: 100%;
4 | display: flex;
5 | flex-direction: column;
6 | align-items: center;
7 | justify-content: space-between;
8 | padding: 0;
9 | box-sizing: border-box;
10 | }
11 |
--------------------------------------------------------------------------------
/demo/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "pages": [
3 | "pages/index/index"
4 | ],
5 | "window": {
6 | "backgroundTextStyle": "light",
7 | "navigationBarBackgroundColor": "#fff",
8 | "navigationBarTitleText": "COS 小程序 SDK DEMO",
9 | "navigationBarTextStyle": "black"
10 | },
11 | "sitemapLocation": "sitemap.json"
12 | }
--------------------------------------------------------------------------------
/demo-album/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "pages": [
3 | "pages/index/index",
4 | "pages/preview/preview",
5 | "pages/album/album"
6 | ],
7 | "window": {
8 | "navigationBarTitleText": "COS 上传示例",
9 | "windowBackground": "white",
10 | "navigationBarTextStyle": "white",
11 | "navigationBarBackgroundColor": "#373b3e"
12 | },
13 | "sitemapLocation": "sitemap.json"
14 | }
15 |
--------------------------------------------------------------------------------
/scripts/patch-check.js:
--------------------------------------------------------------------------------
1 | const fs = require('fs');
2 | const path = require('path');
3 | const exec = require('child_process').execSync;
4 |
5 | const patchesDir = path.join(__dirname, '../patches');
6 | const hasPatches = fs.existsSync(patchesDir);
7 |
8 | if (hasPatches) {
9 | // 执行补丁
10 | console.log('npx patch-package');
11 | exec('npx patch-package', { stdio: 'inherit' });
12 | } else {
13 | console.log('无补丁应用');
14 | }
--------------------------------------------------------------------------------
/demo-album/lib/request.js:
--------------------------------------------------------------------------------
1 |
2 |
3 | module.exports = (options) => {
4 | return new Promise((resolve, reject) => {
5 | options = Object.assign(options, {
6 | success(result) {
7 | if (result.statusCode === 200) {
8 | resolve(result.data);
9 | } else {
10 | reject(result);
11 | }
12 | },
13 | fail: reject,
14 | });
15 | wx.request(options);
16 | });
17 | };
--------------------------------------------------------------------------------
/demo-album/pages/index/index.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 简单上传
4 | 文件 3 天后删除
5 |
6 |
7 |
8 |
9 | 共享相册
10 |
11 |
12 |
13 | ● 上传对象到 COS 并获取下载地址
14 |
--------------------------------------------------------------------------------
/demo/demo-ci.js:
--------------------------------------------------------------------------------
1 | const asrDao = require('./ciDemo/asr');
2 | const auditDao = require('./ciDemo/audit');
3 | const docPreviewDao = require('./ciDemo/docPreview');
4 | const picProcessDao = require('./ciDemo/picProcess');
5 | const mediaProcessDao = require('./ciDemo/mediaProcess');
6 | const fileProcessDao = require('./ciDemo/fileProcess');
7 | const metaDao =require('./ciDemo/meta');
8 | const otherDao = require('./ciDemo/other');
9 |
10 | module.exports = {
11 | asrDao,
12 | auditDao,
13 | docPreviewDao,
14 | picProcessDao,
15 | mediaProcessDao,
16 | fileProcessDao,
17 | metaDao,
18 | otherDao
19 | };
--------------------------------------------------------------------------------
/patches/fast-xml-parser+4.5.0.patch:
--------------------------------------------------------------------------------
1 | diff --git a/node_modules/fast-xml-parser/src/xmlparser/OrderedObjParser.js b/node_modules/fast-xml-parser/src/xmlparser/OrderedObjParser.js
2 | index 70db055..17536b7 100644
3 | --- a/node_modules/fast-xml-parser/src/xmlparser/OrderedObjParser.js
4 | +++ b/node_modules/fast-xml-parser/src/xmlparser/OrderedObjParser.js
5 | @@ -184,7 +184,7 @@ function buildAttributesMap(attrStr, jPath, tagName) {
6 | }
7 |
8 | const parseXml = function(xmlData) {
9 | - xmlData = xmlData.replace(/\r\n?/g, "\n"); //TODO: remove this line
10 | + // xmlData = xmlData.replace(/\r\n?/g, "\n"); //TODO: remove this line
11 | const xmlObj = new xmlNode('!xml');
12 | let currentNode = xmlObj;
13 | let textData = "";
14 |
--------------------------------------------------------------------------------
/demo-album/project.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "description": "",
3 | "setting": {
4 | "urlCheck": true,
5 | "es6": true,
6 | "postcss": true,
7 | "minified": true,
8 | "newFeature": true,
9 | "babelSetting": {
10 | "ignore": [],
11 | "disablePlugins": [],
12 | "outputPath": ""
13 | },
14 | "coverView": false,
15 | "enhance": false,
16 | "showShadowRootInWxmlPanel": false,
17 | "packNpmRelationList": [],
18 | "ignoreUploadUnusedFiles": true
19 | },
20 | "compileType": "miniprogram",
21 | "condition": {},
22 | "editorSetting": {
23 | "tabIndent": "insertSpaces",
24 | "tabSize": 2
25 | },
26 | "libVersion": "2.28.0",
27 | "packOptions": {
28 | "ignore": [],
29 | "include": []
30 | },
31 | "appid": "wx0000000000000000"
32 | }
33 |
--------------------------------------------------------------------------------
/demo/app.js:
--------------------------------------------------------------------------------
1 | //app.js
2 | App({
3 | onLaunch: function () {
4 | //调用API从本地缓存中获取数据
5 | var logs = wx.getStorageSync('logs') || [];
6 | logs.unshift(Date.now())
7 | wx.setStorageSync('logs', logs)
8 | },
9 | getUserInfo:function(cb){
10 | var that = this
11 | if(this.globalData.userInfo){
12 | typeof cb == "function" && cb(this.globalData.userInfo);
13 | }else{
14 | //调用登录接口
15 | wx.login({
16 | success: function () {
17 | wx.getUserInfo({
18 | success: function (res) {
19 | that.globalData.userInfo = res.userInfo;
20 | typeof cb == "function" && cb(that.globalData.userInfo);
21 | }
22 | })
23 | }
24 | })
25 | }
26 | },
27 | globalData:{
28 | userInfo:null
29 | }
30 | });
--------------------------------------------------------------------------------
/demo-album/lib/cos.js:
--------------------------------------------------------------------------------
1 | var COS = require('./cos-wx-sdk-v5');
2 | var config = require('../config');
3 | var cos = new COS({
4 | getAuthorization: function (options, callback) {
5 | wx.request({
6 | method: 'GET',
7 | url: config.stsUrl, // 服务端签名,参考 server 目录下的两个签名例子
8 | dataType: 'json',
9 | success: function (result) {
10 | var data = result.data;
11 | callback({
12 | TmpSecretId: data.credentials && data.credentials.tmpSecretId,
13 | TmpSecretKey: data.credentials && data.credentials.tmpSecretKey,
14 | XCosSecurityToken: data.credentials && data.credentials.sessionToken,
15 | ExpiredTime: data.expiredTime,
16 | });
17 | }
18 | });
19 | },
20 | });
21 |
22 | module.exports = cos;
--------------------------------------------------------------------------------
/src/event.js:
--------------------------------------------------------------------------------
1 | var initEvent = function (cos) {
2 | var listeners = {};
3 | var getList = function (action) {
4 | !listeners[action] && (listeners[action] = []);
5 | return listeners[action];
6 | };
7 | cos.on = function (action, callback) {
8 | getList(action).push(callback);
9 | };
10 | cos.off = function (action, callback) {
11 | var list = getList(action);
12 | for (var i = list.length - 1; i >= 0; i--) {
13 | callback === list[i] && list.splice(i, 1);
14 | }
15 | };
16 | cos.emit = function (action, data) {
17 | var list = getList(action).map(function (cb) {
18 | return cb;
19 | });
20 | for (var i = 0; i < list.length; i++) {
21 | list[i](data);
22 | }
23 | };
24 | };
25 |
26 | var EventProxy = function () {
27 | initEvent(this);
28 | };
29 |
30 | module.exports.init = initEvent;
31 | module.exports.EventProxy = EventProxy;
32 |
--------------------------------------------------------------------------------
/demo-album/pages/preview/preview.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 | 上传成功
4 | {{url}}
5 |
6 | 复制 URL,可在浏览器中下载您上传的对象
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
--------------------------------------------------------------------------------
/demo/pages/index/index.wxss:
--------------------------------------------------------------------------------
1 | /**index.wxss**/
2 | .btn-content button {
3 | margin: 0 0 10px 0;
4 | border-radius: 0;
5 | }
6 | .title {
7 | display:block;
8 | box-sizing: border-box;
9 | padding: 0 5px;
10 | width: 100%;
11 | height: 30px;
12 | line-height: 30px;
13 | border-top: 1px solid #ccc;
14 | margin:auto;
15 | font-size:14px;
16 | color:#333;
17 | text-align: left;
18 | font-weight: bold;
19 | }
20 |
21 | .list-panel{
22 | width: 100%;
23 | }
24 |
25 | .sub-title{
26 | display:block;
27 | box-sizing: border-box;
28 | padding: 0 5px;
29 | width: 100%;
30 | height: 30px;
31 | line-height: 30px;
32 | font-size:12px;
33 | color:#333;
34 | text-align: left;
35 | font-weight: bold;
36 | }
37 |
38 | .list {
39 | margin-top: 10px;
40 | padding-bottom: 10px;
41 | width: 100%;
42 | }
43 |
44 | .button {
45 | float: left;
46 | margin: 0 3px 3px 0;
47 | text-align: left;
48 | font-size: 14px;
49 | height: 28px;
50 | line-height:28px;
51 | padding:0 10px;
52 | }
--------------------------------------------------------------------------------
/demo-album/lib/util.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | // 一维数组转二维数组
3 | listToMatrix(list, elementsPerSubArray) {
4 | let matrix = [], i, col, row;
5 | for (i = 0, row = -1; i < list.length; i += 1) {
6 | col = i % elementsPerSubArray;
7 | row = Math.floor(i / elementsPerSubArray);
8 | if (!matrix[row]) matrix[row] = [0, 0, 0];
9 | matrix[row][col] = list[i];
10 | }
11 | return matrix;
12 | },
13 | // 选中文件之后,计算一个随机的短文件名
14 | getRandFileName: function (filePath) {
15 | var extIndex = filePath.lastIndexOf('.');
16 | var extName = extIndex === -1 ? '' : filePath.substr(extIndex);
17 | return parseInt('' + Date.now() + Math.floor(Math.random() * 900 + 100), 10).toString(36) + extName;
18 | },
19 | // 对更多字符编码的 url encode 格式
20 | camSafeUrlEncode: function (str) {
21 | return encodeURIComponent(str)
22 | .replace(/!/g, '%21')
23 | .replace(/'/g, '%27')
24 | .replace(/\(/g, '%28')
25 | .replace(/\)/g, '%29')
26 | .replace(/\*/g, '%2A');
27 | },
28 | };
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2017-present 腾讯云
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "cos-wx-sdk-v5",
3 | "version": "1.8.0",
4 | "description": "小程序 SDK for [腾讯云对象存储服务](https://cloud.tencent.com/product/cos)",
5 | "main": "dist/cos-wx-sdk-v5.min.js",
6 | "scripts": {
7 | "prettier": "prettier --write src demo/demo-sdk.js demo/test.js demo/ciDemo",
8 | "dev": "cross-env NODE_ENV=development node build.js --mode=development",
9 | "build": "cross-env NODE_ENV=production node build.js --mode=production",
10 | "sts.js": "node server/sts.js",
11 | "postinstall": "node scripts/patch-check.js"
12 | },
13 | "repository": {
14 | "type": "git",
15 | "url": "http://github.com/tencentyun/cos-wx-sdk-v5.git"
16 | },
17 | "author": "carsonxu",
18 | "license": "ISC",
19 | "dependencies": {
20 | "fast-xml-parser": "4.5.0",
21 | "mime": "^2.4.6"
22 | },
23 | "devDependencies": {
24 | "@babel/core": "7.17.9",
25 | "@babel/preset-env": "7.16.11",
26 | "babel-loader": "8.2.5",
27 | "body-parser": "^1.18.3",
28 | "cross-env": "^7.0.3",
29 | "express": "^4.17.1",
30 | "patch-package": "^8.0.0",
31 | "prettier": "^3.0.1",
32 | "qcloud-cos-sts": "^3.0.2",
33 | "terser-webpack-plugin": "4.2.3",
34 | "webpack": "4.46.0",
35 | "webpack-cli": "4.10.0"
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/src/async.js:
--------------------------------------------------------------------------------
1 | var eachLimit = function (arr, limit, iterator, callback) {
2 | callback = callback || function () {};
3 | if (!arr.length || limit <= 0) {
4 | return callback();
5 | }
6 |
7 | var completed = 0;
8 | var started = 0;
9 | var running = 0;
10 |
11 | (function replenish() {
12 | if (completed >= arr.length) {
13 | return callback();
14 | }
15 |
16 | while (running < limit && started < arr.length) {
17 | started += 1;
18 | running += 1;
19 | iterator(arr[started - 1], function (err) {
20 | if (err) {
21 | callback(err);
22 | callback = function () {};
23 | } else {
24 | completed += 1;
25 | running -= 1;
26 | if (completed >= arr.length) {
27 | callback();
28 | } else {
29 | replenish();
30 | }
31 | }
32 | });
33 | }
34 | })();
35 | };
36 |
37 | var retry = function (times, iterator, callback) {
38 | var next = function (index) {
39 | iterator(function (err, data) {
40 | if (err && index < times) {
41 | next(index + 1);
42 | } else {
43 | callback(err, data);
44 | }
45 | });
46 | };
47 | if (times < 1) {
48 | callback();
49 | } else {
50 | next(1);
51 | }
52 | };
53 |
54 | var async = {
55 | eachLimit: eachLimit,
56 | retry: retry,
57 | };
58 |
59 | module.exports = async;
60 |
--------------------------------------------------------------------------------
/demo/pages/index/index.wxml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | 简单上传例子(推荐)
9 |
10 |
11 |
12 | 小程序完整 SDK 例子(功能齐全,文件较大)
13 |
14 | {{title[index]}} {{key}}
15 |
16 |
17 |
18 |
19 |
20 |
21 | 数据万象示例
22 |
23 | {{ciTitle[index]}} {{key}}
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
--------------------------------------------------------------------------------
/.github/workflows/auto-changelog.yml:
--------------------------------------------------------------------------------
1 | name: ChangeLog
2 |
3 | on:
4 | workflow_dispatch:
5 | release:
6 | types: [published]
7 |
8 | jobs:
9 | build:
10 |
11 | runs-on: ubuntu-latest
12 |
13 | steps:
14 | - uses: actions/setup-node@v2-beta
15 | with:
16 | node-version: '12'
17 | - uses: actions/checkout@v2
18 | with:
19 | fetch-depth: 0
20 |
21 | - name: Checkout Tool
22 | uses: actions/checkout@v2
23 | with:
24 | repository: konakonall/auto-changelog
25 | path: 'auto-changelog'
26 | - name: Build Tool
27 | run: |
28 | cd auto-changelog
29 | npm install
30 | npm link
31 |
32 | - name: Generate ChangeLog
33 | env: # Or as an environment variable
34 | TOKEN: ${{ secrets.GITHUB_TOKEN }}
35 | run: |
36 | auto-changelog --token $TOKEN
37 | - name: Cat ChangeLog
38 | run: cat CHANGELOG.md
39 |
40 | - name: Commit files
41 | env:
42 | CI_USER: "gouki0123"
43 | CI_EMAIL: "gouki0123@gmail.com"
44 | run: |
45 | git config --local user.email "$CI_EMAIL"
46 | git config --local user.name "$CI_USER"
47 | git add CHANGELOG.md && git commit -m 'Updated CHANGELOG.md' && echo "push=1" >> $GITHUB_ENV || echo "No changes to CHANGELOG.md"
48 |
49 | - name: Push changes
50 | if: env.push == 1
51 | env:
52 | CI_USER: "gouki0123"
53 | CI_TOKEN: ${{ secrets.GITHUB_TOKEN }}
54 | run: |
55 | git push "https://$CI_USER:$CI_TOKEN@github.com/$GITHUB_REPOSITORY.git" HEAD:master
56 |
--------------------------------------------------------------------------------
/demo/pages/index/index.js:
--------------------------------------------------------------------------------
1 | //index.js
2 | var cosDemoSdk = require('../../demo-sdk');
3 | var ciDemoSdk = require('../../demo-ci');
4 | var postUpload = require('../../demo-post-policy');
5 |
6 | var option = {
7 | data: {
8 | demoType: 'cos',
9 | listMap: {},
10 | title: {
11 | toolsDao: '工具函数',
12 | bucketDao: '存储桶操作',
13 | objectDao: '对象操作',
14 | advanceObjectDao: '高级操作',
15 | ciObjectDao: '数据万象示例',
16 | },
17 | ciListMap: {},
18 | ciTitle: {
19 | asrDao: '智能语音',
20 | auditDao: '内容审核',
21 | docPreviewDao: '文档预览',
22 | picProcessDao: '图片处理',
23 | mediaProcessDao: '媒体处理',
24 | fileProcessDao: '文件处理',
25 | metaDao: '智能检索',
26 | otherDao: '其他',
27 | },
28 | },
29 | switchCosDemo() {
30 | this.setData({ demoType: 'cos' });
31 | },
32 | switchCiDemo() {
33 | this.setData({ demoType: 'ci' });
34 | },
35 | };
36 |
37 | for (var key in cosDemoSdk) {
38 | if (cosDemoSdk.hasOwnProperty(key)) {
39 | var sublist = [];
40 | var subDemoSdk = cosDemoSdk[key];
41 | for (var subkey in subDemoSdk) {
42 | sublist.push(subkey);
43 | option[subkey] = subDemoSdk[subkey];
44 | }
45 | option.data.listMap[key] = sublist;
46 | }
47 | }
48 |
49 | for (var key in ciDemoSdk) {
50 | if (ciDemoSdk.hasOwnProperty(key)) {
51 | var sublist = [];
52 | var subDemoSdk = ciDemoSdk[key];
53 | for (var subkey in subDemoSdk) {
54 | sublist.push(subkey);
55 | option[subkey] = subDemoSdk[subkey];
56 | }
57 | option.data.ciListMap[key] = sublist;
58 | }
59 | }
60 |
61 | option.postUpload = postUpload;
62 |
63 | //获取应用实例
64 | Page(option);
65 |
--------------------------------------------------------------------------------
/demo-album/pages/preview/preview.wxss:
--------------------------------------------------------------------------------
1 | .preview {
2 | width: 100%;
3 | height: 100%;
4 | background: #fff;
5 | }
6 |
7 | .preview-success {
8 | display: inline-block;
9 | font-size: 40rpx;
10 | }
11 |
12 | .preview-url {
13 | border: 1px solid #ddd;
14 | padding: 16rpx;
15 | background: #f7f7f7;
16 | word-break:break-all;
17 | font-size: 32rpx;
18 | border-radius: 5rpx;
19 | margin-top: 40rpx;
20 | }
21 |
22 | .preview-tips {
23 | margin-top: 20rpx;
24 | font-size: 30rpx;
25 | color: #888;
26 | }
27 |
28 | .preview-button-wrap {
29 | margin-top: 30rpx;
30 | text-align: left;
31 | }
32 |
33 | .preview-button {
34 | display: inline-block;
35 | position:relative;
36 | text-align: center;
37 | box-sizing:border-box;
38 | font-size:32rpx;
39 | text-decoration:none;
40 | -webkit-tap-highlight-color:transparent;
41 | overflow:hidden;
42 | width:280rpx;
43 | height:85rpx;
44 | background-color:#fff;
45 | color:#2277da;
46 | line-height:85rpx;
47 | border-radius: 5rpx;
48 | }
49 |
50 | .preview-button:first-child {
51 | margin-right: 20rpx;
52 | }
53 |
54 | .preview-outer {
55 | margin-top: 50rpx;
56 | position: relative;
57 | text-align: center;
58 | width: 590rpx;
59 | }
60 |
61 | .preview-inner {
62 | padding-top: 50rpx;
63 | position: relative;
64 | text-align: center;
65 | width: 590rpx;
66 | }
67 |
68 | .preview-image {
69 | display:block;
70 | margin: 0 auto;
71 | width: 590rpx;
72 | height: 590rpx;
73 | }
74 |
75 | .preview-video {
76 | margin-top: 30rpx;
77 | display:block;
78 | margin: 0 auto;
79 | width: 590rpx;
80 | height: 415rpx;
81 | }
--------------------------------------------------------------------------------
/demo/project.config.json:
--------------------------------------------------------------------------------
1 | {
2 | "description": "项目配置文件,详见文档:https://developers.weixin.qq.com/miniprogram/dev/devtools/projectconfig.html",
3 | "setting": {
4 | "urlCheck": false,
5 | "es6": false,
6 | "enhance": false,
7 | "postcss": false,
8 | "preloadBackgroundData": false,
9 | "minified": false,
10 | "newFeature": true,
11 | "coverView": true,
12 | "nodeModules": false,
13 | "autoAudits": false,
14 | "showShadowRootInWxmlPanel": true,
15 | "scopeDataCheck": false,
16 | "uglifyFileName": false,
17 | "checkInvalidKey": true,
18 | "checkSiteMap": true,
19 | "uploadWithSourceMap": true,
20 | "compileHotReLoad": false,
21 | "useMultiFrameRuntime": true,
22 | "useApiHook": true,
23 | "useApiHostProcess": true,
24 | "babelSetting": {
25 | "ignore": [],
26 | "disablePlugins": [],
27 | "outputPath": ""
28 | },
29 | "enableEngineNative": false,
30 | "useIsolateContext": true,
31 | "useCompilerModule": true,
32 | "userConfirmedUseCompilerModuleSwitch": false,
33 | "userConfirmedBundleSwitch": false,
34 | "packNpmManually": false,
35 | "packNpmRelationList": [],
36 | "minifyWXSS": true,
37 | "disableUseStrict": false,
38 | "showES6CompileOption": false,
39 | "useCompilerPlugins": false,
40 | "ignoreUploadUnusedFiles": false,
41 | "useStaticServer": true,
42 | "condition": false
43 | },
44 | "compileType": "miniprogram",
45 | "libVersion": "2.24.0",
46 | "appid": "wx0000000000000000",
47 | "projectname": "demo",
48 | "simulatorType": "wechat",
49 | "simulatorPluginLibVersion": {},
50 | "condition": {},
51 | "packOptions": {
52 | "ignore": [],
53 | "include": []
54 | },
55 | "editorSetting": {
56 | "tabIndent": "insertSpaces",
57 | "tabSize": 2
58 | }
59 | }
--------------------------------------------------------------------------------
/demo-album/pages/preview/preview.js:
--------------------------------------------------------------------------------
1 | Page({
2 | data: {
3 | type: 'image',
4 | url: '',
5 | showPreview: false,
6 | },
7 | onShareAppMessage: function (res) {
8 | var typeMap = {
9 | image: '图片',
10 | video: '视频',
11 | };
12 | var data = {
13 | title: 'COS 上传示例 - 预览' + (typeMap[this.data.type] || '文件'),
14 | path: this.route + '?type=' + this.data.type + '&url=' + encodeURIComponent(this.data.url),
15 | };
16 | if (this.data.type === 'image') {
17 | data.imageUrl = this.data.url;
18 | }
19 | return data;
20 | },
21 | onShow() {
22 | this.setData({showPreview: false});
23 | this.setData({
24 | type: this.options.type || 'image',
25 | url: decodeURIComponent(this.options.url) || '',
26 | });
27 | },
28 | showPreviewBox() {
29 | this.setData({showPreview: true});
30 | },
31 | copyLink() {
32 | wx.setClipboardData({
33 | data: this.data.url || '',
34 | success: function () {
35 | wx.showToast({title: '复制成功', icon: 'success', duration: 2000});
36 | },
37 | fail: function () {
38 | wx.showToast({title: '复制失败', icon: 'error', duration: 2000});
39 | },
40 | });
41 | },
42 | saveImage() {
43 | wx.downloadFile({
44 | url: this.data.url,
45 | success: function(res) {
46 | // 只要服务器有响应数据,就会把响应内容写入文件并进入 success 回调,业务需要自行判断是否下载到了想要的内容
47 | if (res.statusCode === 200) {
48 | wx.saveImageToPhotosAlbum({
49 | filePath: res.tempFilePath,
50 | });
51 | }
52 | }
53 | });
54 | },
55 | });
--------------------------------------------------------------------------------
/demo-album/pages/album/album.wxml:
--------------------------------------------------------------------------------
1 |
2 | 上传图片到 COS 共享给他人查看,长按图片可删除
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | 上传图片
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 | 保存到本地
35 | 删除图片
36 | 取消
37 |
38 |
39 | {{loadingMessage}}
40 | {{toastMessage}}
--------------------------------------------------------------------------------
/demo/ciDemo/mediaProcess.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | var wxfs = wx.getFileSystemManager();
3 | const config = require('../config');
4 | const { cos, requestCallback } = require('../tools');
5 |
6 | const mediaProcessDao = {
7 | 'describeMediaBuckets 查询已经开通数据万象功能的存储桶': describeMediaBuckets,
8 | 'getMediaInfo 获取媒体文件信息': getMediaInfo,
9 | 'getSnapshot 获取媒体文件某个时间的截图': getSnapshot,
10 | };
11 |
12 | function describeMediaBuckets() {
13 | var host = 'ci.' + config.Region + '.myqcloud.com';
14 | var url = 'https://' + host + '/mediabucket';
15 | cos.request(
16 | {
17 | Bucket: config.Bucket,
18 | Region: config.Region,
19 | Method: 'GET',
20 | Key: 'mediabucket' /** 固定值,必须 */,
21 | Url: url,
22 | Query: {
23 | pageNumber: '1' /** 第几页,非必须 */,
24 | pageSize: '10' /** 每页个数,非必须 */,
25 | // regions: 'ap-chengdu', /** 地域信息,例如'ap-beijing',支持多个值用逗号分隔如'ap-shanghai,ap-beijing',非必须 */
26 | // bucketNames: 'test-1250000000', /** 存储桶名称,精确搜索,例如'test-1250000000',支持多个值用逗号分隔如'test1-1250000000,test2-1250000000',非必须 */
27 | // bucketName: 'test', /** 存储桶名称前缀,前缀搜索,例如'test',支持多个值用逗号分隔如'test1,test2',非必须 */
28 | },
29 | },
30 | function (err, data) {
31 | console.log(err || data);
32 | }
33 | );
34 | }
35 |
36 | function getMediaInfo() {
37 | cos.request(
38 | {
39 | Bucket: config.Bucket,
40 | Region: config.Region,
41 | Method: 'GET',
42 | Key: 'test.mp4',
43 | Query: {
44 | 'ci-process': 'videoinfo' /** 固定值,必须 */,
45 | },
46 | },
47 | function (err, data) {
48 | console.log(err || data);
49 | }
50 | );
51 | }
52 |
53 | function getSnapshot() {
54 | cos.request(
55 | {
56 | Bucket: config.Bucket,
57 | Region: config.Region,
58 | Method: 'GET',
59 | Key: 'test.mp4',
60 | Query: {
61 | 'ci-process': 'snapshot' /** 固定值,必须 */,
62 | time: 1 /** 截图的时间点,单位为秒,必须 */,
63 | // width: 0, /** 截图的宽,非必须 */
64 | // height: 0, /** 截图的高,非必须 */
65 | // format: 'jpg', /** 截图的格式,支持 jpg 和 png,默认 jpg,非必须 */
66 | // rotate: 'auto', /** 图片旋转方式,默认为'auto',非必须 */
67 | // mode: 'exactframe', /** 截帧方式,默认为'exactframe',非必须 */
68 | },
69 | RawBody: true,
70 | DataType: 'arraybuffer',
71 | },
72 | function (err, data) {
73 | if (err) {
74 | console.log(err);
75 | } else {
76 | const imgBase64 = wx.arrayBufferToBase64(data.Body);
77 | console.log(imgBase64);
78 | }
79 | }
80 | );
81 | }
82 |
83 | module.exports = mediaProcessDao;
84 |
--------------------------------------------------------------------------------
/demo-album/pages/album/album.wxss:
--------------------------------------------------------------------------------
1 | .container {
2 | width: 100%;
3 | height: 100%;
4 | min-height: 242rpx;
5 | }
6 |
7 | .album-container {
8 | margin: 3rpx;
9 | min-height: 242rpx;
10 | marring-top: 83rpx;
11 | }
12 |
13 | .item-group {
14 | display: flex;
15 | }
16 |
17 | .album-item {
18 | flex: 1;
19 | text-align: center;
20 | margin: 3rpx;
21 | width: 242rpx;
22 | height: 242rpx;
23 | line-height: 242rpx;
24 | }
25 |
26 | .album-item.empty {
27 | background: transparent;
28 | }
29 |
30 | .upload-tips {
31 | color: #888;
32 | display: block;
33 | font-size: 30rpx;
34 | height: 80rpx;
35 | line-height: 80rpx;
36 | margin: 0 20rpx;
37 | }
38 | .upload-wrap {
39 | color: #ccc;
40 | background-color: #fff;
41 | border: 1px solid #888;
42 | position: absolute;
43 | left: 6rpx;
44 | top: 86rpx;
45 | width: 240rpx;
46 | height: 239rpx;
47 | text-align: center;
48 | line-height: 240rpx;
49 | overflow: hidden;
50 | }
51 | .upload-add-outer {
52 | display: inline-block;
53 | flex: 1;
54 | text-align: center;
55 | margin: 3rpx;
56 | width: 242rpx;
57 | height: 242rpx;
58 | line-height: 242rpx;
59 | }
60 | .upload-add {
61 | display: block;
62 | position: relative;
63 | text-align: center;
64 | margin: 3rpx;
65 | width: 242rpx;
66 | height: 240rpx;
67 | border: 1rpx solid #ccc;
68 | line-height: 240rpx;
69 | color: #ccc;
70 | overflow: hidden;
71 | box-sizing:border-box;
72 | }
73 |
74 | .upload-add image {
75 | position: absolute;
76 | left: 0;
77 | width: 240rpx;
78 | height: 2.6rem;
79 | top: 1.2rem;
80 | }
81 |
82 | .upload-add text {
83 | position: absolute;
84 | width: 240rpx;
85 | height: 2rem;
86 | left: 0;
87 | top: 1.6rem;
88 | color: #888;
89 | }
90 |
91 | .upload-image {
92 | color: #ccc;
93 | background-color: #fff;
94 | border: 1px solid #888;
95 | position: absolute;
96 | left: 6rpx;
97 | top: 86rpx;
98 | width: 240rpx;
99 | height: 239rpx;
100 | text-align: center;
101 | line-height: 240rpx;
102 | overflow: hidden;
103 | }
104 |
105 | .upload-image image {
106 | position: absolute;
107 | left: 0;
108 | width: 242rpx;
109 | height: 2.6rem;
110 | top: 1.2rem;
111 | }
112 |
113 | .upload-image text {
114 | position: absolute;
115 | width: 242rpx;
116 | height: 2rem;
117 | left: 0;
118 | top: 1.6rem;
119 | color: #888;
120 | }
121 |
122 | .swiper-container {
123 | position: fixed;
124 | left: 0;
125 | top: 0;
126 | width: 100%;
127 | height: 100%;
128 | background: #000;
129 | }
130 |
131 | .swiper-container image {
132 | width: 100%;
133 | height: 100%;
134 | }
135 |
136 | action-sheet-item.warn {
137 | color: #e64340;
138 | }
--------------------------------------------------------------------------------
/demo-album/pages/index/index.js:
--------------------------------------------------------------------------------
1 | var util = require('../../lib/util');
2 | var config = require('../../config');
3 | var cos = require('../../lib/cos');
4 |
5 | Page({
6 | onLoad: function () {
7 | },
8 | onShareAppMessage: function (res) {
9 | return {
10 | title: 'COS 上传示例',
11 | path: this.route,
12 | }
13 | },
14 | // 前往相册页
15 | uploadImage() {
16 | wx.chooseImage({
17 | count: 1,
18 | camera: 'back',
19 | sizeType: ['compressed'],
20 | sourceType: ['album', 'camera'],
21 | success: function (res) {
22 | var filePath = res.tempFilePaths[0];
23 | if (filePath) {
24 | var Key = util.getRandFileName(filePath);
25 | wx.showLoading({title: '正在上传...'});
26 | cos.postObject({
27 | Bucket: config.Bucket,
28 | Region: config.Region,
29 | Key: Key,
30 | FilePath: filePath,
31 | }, function (err, data) {
32 | wx.hideLoading();
33 | if (data && data.Location) {
34 | wx.navigateTo({url: '../preview/preview?type=image&url=' + encodeURIComponent('https://' + data.Location)});
35 | } else {
36 | wx.showToast({title: '上传失败', icon: 'error', duration: 2000});
37 | }
38 | });
39 | }
40 | }
41 | })
42 | },
43 | // 前往相册页
44 | uploadVideo() {
45 | wx.chooseVideo({
46 | count: 1,
47 | sizeType: ['compressed'],
48 | sourceType: ['album', 'camera'],
49 | maxDuration: 60,
50 | camera: 'back',
51 | success: function (res) {
52 | var filePath = res.tempFilePath;
53 | if (filePath) {
54 | var Key = util.getRandFileName(filePath);
55 | wx.showLoading({title: '正在上传...'});
56 | cos.postObject({
57 | Bucket: config.Bucket,
58 | Region: config.Region,
59 | Key: Key,
60 | FilePath: filePath,
61 | }, function (err, data) {
62 | wx.hideLoading();
63 | if (data && data.Location) {
64 | wx.navigateTo({url: '../preview/preview?type=video&url=' + encodeURIComponent('https://' + data.Location)});
65 | } else {
66 | wx.showToast({title: '上传失败', icon: 'error', duration: 2000});
67 | }
68 | });
69 | }
70 | }
71 | });
72 | },
73 | // 前往相册页
74 | gotoAlbum() {
75 | wx.navigateTo({url: '../album/album'});
76 | },
77 | });
78 |
--------------------------------------------------------------------------------
/demo/ciDemo/other.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | var wxfs = wx.getFileSystemManager();
3 | const config = require('../config');
4 | const { cos, requestCallback } = require('../tools');
5 |
6 | const otherDao = {
7 | '提交病毒检测任务 postVirusDetect': postVirusDetect,
8 | '查询病毒检测任务结果 getVirusDetectResult': getVirusDetectResult,
9 | '查询防盗链 describeRefer': describeRefer,
10 | '设置防盗链 setRefer': setRefer,
11 | };
12 |
13 | function postVirusDetect() {
14 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/virus/detect';
15 | var url = 'https://' + host;
16 | var body = COS.util.json2xml({
17 | Request: {
18 | Input: {
19 | Object: 'test/1.png', // 文件名,取值为文件在当前存储桶中的完整名称,与Url参数二选一
20 | // Url: 'http://examplebucket-1250000000.cos.ap-shanghai.myqcloud.com/virus.doc', // 病毒文件的链接地址,与Object参数二选一
21 | },
22 | Conf: {
23 | DetectType: 'Virus', // 检测的病毒类型,当前固定为:Virus
24 | // CallBack: 'http://callback.demo.com', // 任务回调的地址
25 | },
26 | },
27 | });
28 | cos.request(
29 | {
30 | Method: 'POST',
31 | Key: 'virus/detect',
32 | Url: url,
33 | Body: body,
34 | ContentType: 'application/xml',
35 | },
36 | function (err, data) {
37 | console.log(err || data);
38 | }
39 | );
40 | }
41 |
42 | function getVirusDetectResult() {
43 | var jobId = 'ss5a8d3065bd9011eda1445254009dadxx'; // 提交病毒检测任务后会返回当前任务的jobId
44 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/virus/detect/' + jobId;
45 | var url = 'https://' + host;
46 | cos.request(
47 | {
48 | Method: 'GET',
49 | Key: 'virus/detect/' + jobId,
50 | Url: url,
51 | },
52 | function (err, data) {
53 | console.log(err || data);
54 | }
55 | );
56 | }
57 |
58 | function describeRefer() {
59 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
60 | const host = config.Bucket + '.pic.' + config.Region + '.myqcloud.com/?hotlink';
61 | const url = 'https://' + host;
62 | cos.request(
63 | {
64 | Method: 'GET', // 固定值,必须
65 | Url: url, // 请求的url,必须
66 | },
67 | function (err, data) {
68 | if (err) {
69 | // 处理请求失败
70 | console.log(err);
71 | } else {
72 | // 处理请求成功
73 | console.log(data.Response);
74 | }
75 | }
76 | );
77 | }
78 |
79 | function setRefer() {
80 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
81 | const host = config.Bucket + '.pic.' + config.Region + '.myqcloud.com/?hotlink';
82 | const url = 'https://' + host;
83 | const body = COS.util.json2xml({
84 | Hotlink: {
85 | Url: 'https://www.example.com', // 必须,域名地址
86 | Type: 'white', // 必须,防盗链类型,white 为白名单,black 为黑名单,off 为关闭。
87 | },
88 | });
89 | cos.request(
90 | {
91 | Method: 'PUT',
92 | Url: url,
93 | Body: body,
94 | },
95 | function (err, data) {
96 | console.log(err || data);
97 | }
98 | );
99 | }
100 |
101 | module.exports = otherDao;
102 |
--------------------------------------------------------------------------------
/demo/demo-post-policy.js:
--------------------------------------------------------------------------------
1 | var config = require('./config');
2 |
3 | var uploadFile = function () {
4 | // 对更多字符编码的 url encode 格式
5 | var camSafeUrlEncode = function (str) {
6 | return encodeURIComponent(str)
7 | .replace(/!/g, '%21')
8 | .replace(/'/g, '%27')
9 | .replace(/\(/g, '%28')
10 | .replace(/\)/g, '%29')
11 | .replace(/\*/g, '%2A');
12 | };
13 |
14 | // 获取临时密钥
15 | var getCredentials = function (options, callback) {
16 | wx.request({
17 | method: 'GET',
18 | url: 'http://127.0.0.1:3000/post-policy?ext=' + options.ext, // 服务端签名,参考 server 目录下的两个签名例子
19 | dataType: 'json',
20 | success: function (result) {
21 | var data = result.data;
22 | if (data) {
23 | callback(data);
24 | } else {
25 | wx.showModal({ title: '临时密钥获取失败', content: JSON.stringify(data), showCancel: false });
26 | }
27 | },
28 | error: function (err) {
29 | wx.showModal({ title: '临时密钥获取失败', content: JSON.stringify(err), showCancel: false });
30 | },
31 | });
32 | };
33 |
34 | // 上传文件
35 | var uploadFile = function (filePath) {
36 | var extIndex = filePath.lastIndexOf('.');
37 | var fileExt = extIndex >= -1 ? filePath.substr(extIndex + 1) : '';
38 | getCredentials({ ext: fileExt }, function (credentials) {
39 | // 请求用到的参数
40 | // var prefix = 'https://cos.' + config.Region + '.myqcloud.com/' + config.Bucket + '/'; // 这个是后缀式,签名也要指定 Pathname: '/' + config.Bucket + '/'
41 | var prefix = 'https://' + credentials.bucket + '.cos.' + credentials.region + '.myqcloud.com/';
42 | var key = credentials.key; // 让服务端来决定文件名更安全
43 | var formData = {
44 | key: key,
45 | success_action_status: 200,
46 | 'Content-Type': '',
47 | 'q-sign-algorithm': credentials.qSignAlgorithm,
48 | 'q-ak': credentials.qAk,
49 | 'q-key-time': credentials.qKeyTime,
50 | 'q-signature': credentials.qSignature,
51 | policy: credentials.policy,
52 | };
53 | if (credentials.securityToken) formData['x-cos-security-token'] = credentials.securityToken;
54 | var requestTask = wx.uploadFile({
55 | url: prefix,
56 | name: 'file',
57 | filePath: filePath,
58 | formData: formData,
59 | success: function (res) {
60 | var url = prefix + camSafeUrlEncode(key).replace(/%2F/g, '/');
61 | if (res.statusCode === 200) {
62 | wx.showModal({ title: '上传成功', content: url, showCancel: false });
63 | } else {
64 | wx.showModal({ title: '上传失败', content: JSON.stringify(res), showCancel: false });
65 | }
66 | console.log(res.header['x-cos-request-id']);
67 | console.log(res.statusCode);
68 | console.log(url);
69 | },
70 | fail: function (res) {
71 | wx.showModal({ title: '上传失败', content: JSON.stringify(res), showCancel: false });
72 | },
73 | });
74 | requestTask.onProgressUpdate(function (res) {
75 | console.log('正在进度:', res);
76 | });
77 | });
78 | };
79 |
80 | // 选择文件
81 | wx.chooseMedia({
82 | count: 1, // 默认9
83 | mediaType: ['image', 'video'],
84 | sizeType: ['original'],
85 | sourceType: ['album', 'camera'],
86 | success: function (res) {
87 | uploadFile(res.tempFiles[0].tempFilePath);
88 | },
89 | });
90 | };
91 |
92 | module.exports = uploadFile;
93 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | ## 快速入门
2 |
3 | 微信小程序 SDK for [腾讯云对象存储服务](https://cloud.tencent.com/product/cos)
4 |
5 | ### 一、前期准备
6 |
7 | 1. 到 (COS对象存储控制台)[https://console.cloud.tencent.com/cos4] 创建存储桶,得到 Bucket(存储桶名称) 和 Region(地域名称)
8 | 2. 到 (控制台密钥管理)[https://console.cloud.tencent.com/capi] 获取您的项目 SecretId 和 SecretKey
9 |
10 | ### 二、计算签名
11 |
12 | 由于签名计算放在前端会暴露 SecretId 和 SecretKey,我们把签名计算过程放在后端实现,前端通过 ajax 向后端获取签名结果,正式部署时请再后端加一层自己网站本身的权限检验。
13 |
14 | 这里提供 [NodeJS 的签名示例](https://github.com/tencentyun/cos-js-sdk-v5/blob/master/server/),其他语言,请参照对应的 [XML SDK](https://cloud.tencent.com/document/product/436/6474)
15 |
16 | ### 三、上传例子
17 |
18 | 1. 把 demo/lib/cos-wx-sdk-v5.js 复制到自己小程序项目代码里,在需要上传文件的地方贴以下代码
19 |
20 | ```javascript
21 | // 存储桶名称,由bucketname-appid 组成,appid必须填入,可以在COS控制台查看存储桶名称。 https://console.cloud.tencent.com/cos5/bucket
22 | const Bucket = 'test-1250000000';
23 | // 存储桶Region可以在COS控制台指定存储桶的概览页查看 https://console.cloud.tencent.com/cos5/bucket/
24 | // 关于地域的详情见 https://cloud.tencent.com/document/product/436/6224
25 | const Region = 'ap-guangzhou';
26 |
27 | // 初始化实例
28 | const cos = new COS({
29 | SimpleUploadMethod: 'putObject', // 强烈建议,高级上传、批量上传内部对小文件做简单上传时使用putObject,sdk版本至少需要v1.3.0
30 | getAuthorization: function (options, callback) {
31 | // 初始化时不会调用,只有调用 cos 方法(例如 cos.putObject)时才会进入
32 | // 异步获取临时密钥
33 | // 服务端 JS 示例:https://github.com/tencentyun/cos-js-sdk-v5/blob/master/server/
34 | // 服务端其他语言参考 COS STS SDK :https://github.com/tencentyun/qcloud-cos-sts-sdk
35 | // STS 详细文档指引看:https://cloud.tencent.com/document/product/436/14048
36 | wx.request({
37 | url: 'https://example.com/server/sts', // url 替换成您自己的后端服务
38 | data: {
39 | bucket: options.Bucket,
40 | region: options.Region,
41 | },
42 | dataType: 'json',
43 | success: function (result) {
44 | const data = result.data;
45 | const credentials = data && data.credentials;
46 | if (!data || !credentials) return console.error('credentials invalid');
47 | // 检查credentials格式
48 | console.log(credentials);
49 | callback({
50 | TmpSecretId: credentials.tmpSecretId,
51 | TmpSecretKey: credentials.tmpSecretKey,
52 | // v1.2.0之前版本的 SDK 使用 XCosSecurityToken 而不是 SecurityToken
53 | SecurityToken: credentials.sessionToken,
54 | // 建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
55 | StartTime: data.startTime, // 时间戳,单位秒,如:1580000000
56 | ExpiredTime: data.expiredTime, // 时间戳,单位秒,如:1580000900
57 | });
58 | },
59 | });
60 | },
61 | });
62 |
63 | // 选择文件
64 | wx.chooseMedia({
65 | count: 1, // 默认9
66 | mediaType: ['image', 'video'],
67 | sizeType: ['original'],
68 | sourceType: ['album', 'camera'],
69 | success: function (res) {
70 | const filePath = res.tempFiles[0].tempFilePath;
71 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
72 | cos.postObject(
73 | {
74 | Bucket: Bucket,
75 | Region: Region,
76 | Key: filename,
77 | FilePath: filePath,
78 | onProgress: function (info) {
79 | console.log(JSON.stringify(info));
80 | },
81 | },
82 | function (err, data) {
83 | console.log(err || data);
84 | }
85 | );
86 | },
87 | });
88 | ```
89 |
90 | ## 说明文档
91 |
92 | [使用例子](demo/demo-sdk.js)
93 |
94 | [快速入门](https://cloud.tencent.com/document/product/436/31953)
95 |
96 | [接口文档](https://cloud.tencent.com/document/product/436/31953)
97 |
--------------------------------------------------------------------------------
/src/session.js:
--------------------------------------------------------------------------------
1 | var util = require('./util');
2 |
3 | // 按照文件特征值,缓存 UploadId
4 | var cacheKey = 'cos_sdk_upload_cache';
5 | var expires = 30 * 24 * 3600;
6 | var cache;
7 | var timer;
8 |
9 | var getCache = function () {
10 | try {
11 | var val = JSON.parse(wx.getStorageSync(cacheKey));
12 | } catch (e) {}
13 | if (!val) val = [];
14 | return val;
15 | };
16 | var setCache = function () {
17 | try {
18 | if (cache.length) wx.setStorageSync(cacheKey, JSON.stringify(cache));
19 | else wx.removeStorageSync(cacheKey);
20 | } catch (e) {}
21 | };
22 |
23 | var init = function () {
24 | if (cache) return;
25 | cache = getCache();
26 | // 清理太老旧的数据
27 | var changed = false;
28 | var now = Math.round(Date.now() / 1000);
29 | for (var i = cache.length - 1; i >= 0; i--) {
30 | var mtime = cache[i][2];
31 | if (!mtime || mtime + expires < now) {
32 | cache.splice(i, 1);
33 | changed = true;
34 | }
35 | }
36 | changed && setCache();
37 | };
38 |
39 | // 把缓存存到本地
40 | var save = function () {
41 | if (timer) return;
42 | timer = setTimeout(function () {
43 | setCache();
44 | timer = null;
45 | }, 400);
46 | };
47 |
48 | var mod = {
49 | using: {},
50 | // 标记 UploadId 正在使用
51 | setUsing: function (uuid) {
52 | mod.using[uuid] = true;
53 | },
54 | // 标记 UploadId 已经没在使用
55 | removeUsing: function (uuid) {
56 | delete mod.using[uuid];
57 | },
58 | // 用上传参数生成哈希值
59 | getFileId: function (FileStat, ChunkSize, Bucket, Key) {
60 | if (FileStat.FilePath && FileStat.size && FileStat.lastModifiedTime && ChunkSize) {
61 | return (
62 | util.md5([FileStat.FilePath].join('::')) +
63 | '-' +
64 | util.md5(
65 | [
66 | FileStat.size,
67 | FileStat.mode,
68 | FileStat.lastAccessedTime,
69 | FileStat.lastModifiedTime,
70 | ChunkSize,
71 | Bucket,
72 | Key,
73 | ].join('::')
74 | )
75 | );
76 | } else {
77 | return null;
78 | }
79 | },
80 | // 用上传参数生成哈希值
81 | getCopyFileId: function (copySource, sourceHeaders, ChunkSize, Bucket, Key) {
82 | var size = sourceHeaders['content-length'];
83 | var etag = sourceHeaders.etag || '';
84 | var lastModified = sourceHeaders['last-modified'];
85 | if (copySource && ChunkSize) {
86 | return util.md5([copySource, size, etag, lastModified, ChunkSize, Bucket, Key].join('::'));
87 | } else {
88 | return null;
89 | }
90 | },
91 | // 获取文件对应的 UploadId 列表
92 | getUploadIdList: function (uuid) {
93 | if (!uuid) return null;
94 | init();
95 | var list = [];
96 | for (var i = 0; i < cache.length; i++) {
97 | if (cache[i][0] === uuid) list.push(cache[i][1]);
98 | }
99 | return list.length ? list : null;
100 | },
101 | // 缓存 UploadId
102 | saveUploadId: function (uuid, UploadId, limit) {
103 | init();
104 | if (!uuid) return;
105 | // 清理没用的 UploadId
106 | var part1 = uuid.substr(0, uuid.indexOf('-') + 1);
107 | for (var i = cache.length - 1; i >= 0; i--) {
108 | var item = cache[i];
109 | if (item[0] === uuid && item[1] === UploadId) {
110 | cache.splice(i, 1);
111 | } else if (uuid !== item[0] && item[0].indexOf(part1) === 0) {
112 | // 文件路径相同,但其他信息不同,说明文件改变了或上传参数(存储桶、路径、分片大小)变了,直接清理掉
113 | cache.splice(i, 1);
114 | }
115 | }
116 | cache.unshift([uuid, UploadId, Math.round(Date.now() / 1000)]);
117 | if (cache.length > limit) cache.splice(limit);
118 | save();
119 | },
120 | // UploadId 已用完,移除掉
121 | removeUploadId: function (UploadId) {
122 | init();
123 | delete mod.using[UploadId];
124 | for (var i = cache.length - 1; i >= 0; i--) {
125 | if (cache[i][1] === UploadId) cache.splice(i, 1);
126 | }
127 | save();
128 | },
129 | };
130 |
131 | module.exports = mod;
132 |
--------------------------------------------------------------------------------
/src/cos.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | var util = require('./util');
4 | var event = require('./event');
5 | var task = require('./task');
6 | var base = require('./base');
7 | var advance = require('./advance');
8 | var pkg = require('../package.json');
9 |
10 | var defaultOptions = {
11 | SecretId: '',
12 | SecretKey: '',
13 | SecurityToken: '', // 使用临时密钥需要注意自行刷新 Token
14 | StartTime: 0, // 临时密钥返回起始时间
15 | ExpiredTime: 0, // 临时密钥过期时间
16 | ChunkRetryTimes: 2,
17 | FileParallelLimit: 3,
18 | ChunkParallelLimit: 3,
19 | ChunkSize: 1024 * 1024,
20 | SliceSize: 1024 * 1024,
21 | CopyChunkParallelLimit: 20,
22 | CopyChunkSize: 1024 * 1024 * 10,
23 | CopySliceSize: 1024 * 1024 * 10,
24 | MaxPartNumber: 10000,
25 | ProgressInterval: 1000,
26 | UploadQueueSize: 10000,
27 | Domain: '',
28 | ServiceDomain: '',
29 | Protocol: '',
30 | CompatibilityMode: false,
31 | ForcePathStyle: false,
32 | Timeout: 0, // 单位毫秒,0 代表不设置超时时间
33 | CorrectClockSkew: true,
34 | SystemClockOffset: 0, // 单位毫秒,ms
35 | UploadCheckContentMd5: false,
36 | UploadAddMetaMd5: false,
37 | UploadIdCacheLimit: 50,
38 | UseAccelerate: false,
39 | ForceSignHost: true, // 默认将host加入签名计算,关闭后可能导致越权风险,建议保持为true
40 | HttpDNSServiceId: '', // HttpDNS 服务商 Id,填写后代表开启 HttpDNS 服务。HttpDNS 用法详见https://developers.weixin.qq.com/miniprogram/dev/framework/ability/HTTPDNS.html
41 | SimpleUploadMethod: 'postObject', // 高级上传内部判断需要走简单上传时,指定的上传方法,可选postObject或putObject
42 | AutoSwitchHost: false,
43 | CopySourceParser: null, // 自定义拷贝源解析器
44 | ObjectKeySimplifyCheck: true, // 开启合并校验 getObject Key
45 | /** 上报相关配置 **/
46 | DeepTracker: false, // 上报时是否对每个分块上传做单独上报
47 | TrackerDelay: 5000, // 周期性上报,单位毫秒。0代表实时上报
48 | CustomId: '', // 自定义上报id
49 | BeaconReporter: null, // 灯塔上报组件,如有需要请自行传入,传入即代表开启上报
50 | ClsReporter: null, // cls 上报组件,如有需要请自行传入,传入即代表开启上报
51 | };
52 |
53 | // 对外暴露的类
54 | var COS = function (options) {
55 | this.options = util.extend(util.clone(defaultOptions), options || {});
56 | this.options.FileParallelLimit = Math.max(1, this.options.FileParallelLimit);
57 | this.options.ChunkParallelLimit = Math.max(1, this.options.ChunkParallelLimit);
58 | this.options.ChunkRetryTimes = Math.max(0, this.options.ChunkRetryTimes);
59 | this.options.ChunkSize = Math.max(1024 * 1024, this.options.ChunkSize);
60 | this.options.CopyChunkParallelLimit = Math.max(1, this.options.CopyChunkParallelLimit);
61 | this.options.CopyChunkSize = Math.max(1024 * 1024, this.options.CopyChunkSize);
62 | this.options.CopySliceSize = Math.max(0, this.options.CopySliceSize);
63 | this.options.MaxPartNumber = Math.max(1024, Math.min(10000, this.options.MaxPartNumber));
64 | this.options.Timeout = Math.max(0, this.options.Timeout);
65 | this.options.EnableReporter = this.options.BeaconReporter || this.options.ClsReporter;
66 | if (this.options.AppId) {
67 | console.warn(
68 | 'warning: AppId has been deprecated, Please put it at the end of parameter Bucket(E.g: "test-1250000000").'
69 | );
70 | }
71 | if (this.options.SecretId && this.options.SecretId.indexOf(' ') > -1) {
72 | console.error('error: SecretId格式错误,请检查');
73 | console.error('error: SecretId format is incorrect. Please check');
74 | }
75 | if (this.options.SecretKey && this.options.SecretKey.indexOf(' ') > -1) {
76 | console.error('error: SecretKey格式错误,请检查');
77 | console.error('error: SecretKey format is incorrect. Please check');
78 | }
79 | if (this.options.ForcePathStyle) {
80 | console.warn(
81 | 'cos-wx-sdk-v5不再支持使用path-style,仅支持使用virtual-hosted-style,参考文档:https://cloud.tencent.com/document/product/436/96243'
82 | );
83 | throw new Error('ForcePathStyle is not supported');
84 | }
85 | event.init(this);
86 | task.init(this);
87 | };
88 |
89 | base.init(COS, task);
90 | advance.init(COS, task);
91 |
92 | COS.util = {
93 | md5: util.md5,
94 | xml2json: util.xml2json,
95 | json2xml: util.json2xml,
96 | encodeBase64: util.encodeBase64,
97 | };
98 | COS.getAuthorization = util.getAuth;
99 | COS.version = pkg.version;
100 |
101 | module.exports = COS;
102 |
--------------------------------------------------------------------------------
/lib/request.js:
--------------------------------------------------------------------------------
1 | function camSafeUrlEncode(str) {
2 | return encodeURIComponent(str)
3 | .replace(/!/g, '%21')
4 | .replace(/'/g, '%27')
5 | .replace(/\(/g, '%28')
6 | .replace(/\)/g, '%29')
7 | .replace(/\*/g, '%2A');
8 | }
9 |
10 | function getObjectKeys(obj, forKey) {
11 | var list = [];
12 | for (var key in obj) {
13 | if (obj.hasOwnProperty(key)) {
14 | list.push(forKey ? camSafeUrlEncode(key).toLowerCase() : key);
15 | }
16 | }
17 | return list.sort(function (a, b) {
18 | a = a.toLowerCase();
19 | b = b.toLowerCase();
20 | return a === b ? 0 : (a > b ? 1 : -1);
21 | });
22 | };
23 |
24 | var obj2str = function (obj, lowerCaseKey) {
25 | var i, key, val;
26 | var list = [];
27 | var keyList = getObjectKeys(obj);
28 | for (i = 0; i < keyList.length; i++) {
29 | key = keyList[i];
30 | val = (obj[key] === undefined || obj[key] === null) ? '' : ('' + obj[key]);
31 | key = lowerCaseKey ? camSafeUrlEncode(key).toLowerCase() : camSafeUrlEncode(key);
32 | val = camSafeUrlEncode(val) || '';
33 | list.push(key + '=' + val)
34 | }
35 | return list.join('&');
36 | };
37 |
38 | var request = function (params, callback) {
39 | var filePath = params.filePath;
40 | var headers = params.headers || {};
41 | var url = params.url || params.Url;
42 | var method = params.method;
43 | var onProgress = params.onProgress;
44 | var httpDNSServiceId = params.httpDNSServiceId;
45 | var requestTask;
46 |
47 | var cb = function (err, response) {
48 | var H = response.header
49 | var headers = {};
50 | if (H) for (var key in H) {
51 | if (H.hasOwnProperty(key)) headers[key.toLowerCase()] = H[key];
52 | }
53 | callback(err, { statusCode: response.statusCode, headers: headers }, response.data);
54 | };
55 |
56 | if (filePath) {
57 | var fileKey;
58 | var m = url.match(/^(https?:\/\/[^/]+\/)([^/]*\/?)(.*)$/);
59 | if (params.pathStyle) {
60 | fileKey = decodeURIComponent(m[3] || '');
61 | url = m[1] + m[2];
62 | } else {
63 | fileKey = decodeURIComponent(m[2] + m[3] || '');
64 | url = m[1];
65 | }
66 |
67 | // 整理 postObject 参数
68 | var formData = {
69 | 'key': fileKey,
70 | 'success_action_status': 200,
71 | 'Signature': headers.Authorization,
72 | };
73 | var headerKeys = [
74 | 'Cache-Control',
75 | 'Content-Type',
76 | 'Content-Disposition',
77 | 'Content-Encoding',
78 | 'Expires',
79 | 'x-cos-storage-class',
80 | 'x-cos-security-token',
81 | 'x-ci-security-token',
82 | ];
83 | for (var i in params.headers) {
84 | if (params.headers.hasOwnProperty(i) && (i.indexOf('x-cos-meta-') > -1 || headerKeys.indexOf(i) > -1)) {
85 | formData[i] = params.headers[i];
86 | }
87 | }
88 | headers['x-cos-acl'] && (formData.acl = headers['x-cos-acl']);
89 | !formData['Content-Type'] && (formData['Content-Type'] = '');
90 |
91 | requestTask = wx.uploadFile({
92 | url: url,
93 | method: method,
94 | name: 'file',
95 | header: headers,
96 | filePath: filePath,
97 | formData: formData,
98 | timeout: params.timeout,
99 | success: function (response) {
100 | cb(null, response);
101 | },
102 | fail: function (response) {
103 | cb(response.errMsg, response);
104 | }
105 | });
106 | requestTask.onProgressUpdate(function (res) {
107 | onProgress && onProgress({
108 | loaded: res.totalBytesSent,
109 | total: res.totalBytesExpectedToSend,
110 | progress: res.progress / 100
111 | });
112 | });
113 | } else {
114 | var qsStr = params.qs && obj2str(params.qs) || '';
115 | if (qsStr) {
116 | url += (url.indexOf('?') > -1 ? '&' : '?') + qsStr;
117 | }
118 | headers['Content-Length'] && (delete headers['Content-Length']);
119 | var requestParams = {
120 | url: url,
121 | method: method,
122 | header: headers,
123 | dataType: 'text',
124 | data: params.body,
125 | responseType: params.dataType || 'text',
126 | timeout: params.timeout,
127 | redirect: 'manual',
128 | success: function (response) {
129 | cb(null, response);
130 | },
131 | fail: function (response) {
132 | cb(response.errMsg, response);
133 | }
134 | };
135 | if (httpDNSServiceId) {
136 | Object.assign(requestParams, {
137 | enableHttpDNS: true,
138 | httpDNSServiceId: httpDNSServiceId,
139 | });
140 | }
141 | requestTask = wx.request(requestParams);
142 | }
143 |
144 | return requestTask;
145 | };
146 |
147 | module.exports = request;
148 |
--------------------------------------------------------------------------------
/demo/tools.js:
--------------------------------------------------------------------------------
1 | var COS = require('./lib/cos-wx-sdk-v5');
2 | var config = require('./config');
3 | const Beacon = require('./lib/beacon_mp.min');
4 | const ClsClient = require('./lib/cls.min');
5 | const clsClient = new ClsClient({
6 | topicId: 'xxxxxx-xxxx-xxxx-xxxx-xxxxxx', // 日志主题 id
7 | region: 'ap-guangzhou', // 日志主题所在地域,比如 ap-guangzhou,需在小程序平台设置域名白名单:https://ap-guangzhou.cls.tencentcs.com
8 | maxRetainDuration: 30, // 默认 30s
9 | maxRetainSize: 20, // 默认20条
10 | });
11 |
12 | // 签名回调
13 | var getAuthorization = function (options, callback) {
14 | // 格式一、(推荐)后端通过获取临时密钥给到前端,前端计算签名
15 | // 服务端 JS 和 PHP 例子:https://github.com/tencentyun/cos-js-sdk-v5/blob/master/server/
16 | // 服务端其他语言参考 COS STS SDK :https://github.com/tencentyun/qcloud-cos-sts-sdk
17 | wx.request({
18 | method: 'GET',
19 | url: config.stsUrl, // 服务端签名,参考 server 目录下的两个签名例子
20 | dataType: 'json',
21 | success: function (result) {
22 | var data = result.data;
23 | var credentials = data && data.credentials;
24 | if (!data || !credentials) return console.error('credentials invalid');
25 | callback({
26 | TmpSecretId: credentials.tmpSecretId,
27 | TmpSecretKey: credentials.tmpSecretKey,
28 | SecurityToken: credentials.sessionToken,
29 | StartTime: data.startTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
30 | ExpiredTime: data.expiredTime, // 时间戳,单位秒,如:1580000900
31 | });
32 | },
33 | });
34 |
35 | // // 格式二、(推荐)【细粒度控制权限】后端通过获取临时密钥给到前端,前端只有相同请求才重复使用临时密钥,后端可以通过 Scope 细粒度控制权限
36 | // // 服务端例子:https://github.com/tencentyun/qcloud-cos-sts-sdk/edit/master/scope.md
37 | // wx.request({
38 | // method: 'POST',
39 | // url: 'http://127.0.0.1:3000/sts-scope',
40 | // data: options.Scope,
41 | // dataType: 'json',
42 | // success: function(result) {
43 | // var data = result.data;
44 | // var credentials = data && data.credentials;
45 | // if (!data || !credentials) return console.error('credentials invalid');
46 | // callback({
47 | // TmpSecretId: credentials.tmpSecretId,
48 | // TmpSecretKey: credentials.tmpSecretKey,
49 | // XCosSecurityToken: credentials.sessionToken,
50 | // StartTime: data.startTime, // 时间戳,单位秒,如:1580000000,建议返回服务器时间作为签名的开始时间,避免用户浏览器本地时间偏差过大导致签名错误
51 | // ExpiredTime: data.expiredTime, // 时间戳,单位秒,如:1580000900
52 | // ScopeLimit: true, // 细粒度控制权限需要设为 true,会限制密钥只在相同请求时重复使用
53 | // });
54 | // }
55 | // });
56 |
57 | // // 格式三、(不推荐,分片上传权限不好控制)前端每次请求前都需要通过 getAuthorization 获取签名,后端使用固定密钥或临时密钥计算签名返回给前端
58 | // // 服务端获取签名,请参考对应语言的 COS SDK:https://cloud.tencent.com/document/product/436/6474
59 | // // 注意:这种有安全风险,后端需要通过 method、pathname 严格控制好权限,比如不允许 put / 等
60 | // wx.request({
61 | // method: 'POST',
62 | // url: 'https://example.com/sts-auth.php, // 服务端签名,参考 server 目录下的两个签名例子
63 | // data: {
64 | // method: options.Method,
65 | // pathname: options.Pathname,
66 | // query: options.Query,
67 | // headers: options.Headers,
68 | // },
69 | // dataType: 'json',
70 | // success: function(result) {
71 | // var data = result.data;
72 | // if (!data || !data.authorization) return console.error('authorization invalid');
73 | // callback({
74 | // Authorization: data.authorization,
75 | // // XCosSecurityToken: data.sessionToken, // 如果使用临时密钥,需要传 sessionToken
76 | // });
77 | // }
78 | // });
79 |
80 | // // 格式四、(不推荐,适用于前端调试,避免泄露密钥)前端使用固定密钥计算签名
81 | // var authorization = COS.getAuthorization({
82 | // SecretId: 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
83 | // SecretKey: 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
84 | // Method: options.Method,
85 | // Pathname: options.Pathname,
86 | // Query: options.Query,
87 | // Headers: options.Headers,
88 | // Expires: 60,
89 | // });
90 | // callback({
91 | // Authorization: authorization,
92 | // // XCosSecurityToken: credentials.sessionToken, // 如果使用临时密钥,需要传 XCosSecurityToken
93 | // });
94 | };
95 |
96 | var cos = new COS({
97 | getAuthorization: getAuthorization,
98 | // 是否使用全球加速域名。开启该配置后仅以下接口支持操作:putObject、getObject、headObject、optionsObject、multipartInit、multipartListPart、multipartUpload、multipartAbort、multipartComplete、multipartList、sliceUploadFile、uploadFiles
99 | // UseAccelerate: true,
100 | // BeaconReporter: Beacon, // 开启灯塔上报
101 | // ClsReporter: clsClient, // 开启 cls 上报
102 | });
103 |
104 | // 回调统一处理函数
105 | var requestCallback = function (err, data) {
106 | console.log(err || data);
107 | if (err && err.error) {
108 | wx.showModal({
109 | title: '返回错误',
110 | content: '请求失败:' + (err.error.Message || err.error) + ';状态码:' + err.statusCode,
111 | showCancel: false,
112 | });
113 | } else if (err) {
114 | wx.showModal({
115 | title: '请求出错',
116 | content: '请求出错:' + err + ';状态码:' + err.statusCode,
117 | showCancel: false,
118 | });
119 | } else {
120 | wx.showToast({
121 | title: '请求成功',
122 | icon: 'success',
123 | duration: 3000,
124 | });
125 | }
126 | };
127 |
128 | module.exports = {
129 | cos,
130 | requestCallback,
131 | };
--------------------------------------------------------------------------------
/lib/base64.js:
--------------------------------------------------------------------------------
1 | /*
2 | * $Id: base64.js,v 2.15 2014/04/05 12:58:57 dankogai Exp dankogai $
3 | *
4 | * Licensed under the BSD 3-Clause License.
5 | * http://opensource.org/licenses/BSD-3-Clause
6 | *
7 | * References:
8 | * http://en.wikipedia.org/wiki/Base64
9 | */
10 |
11 | var Base64 = (function(global) {
12 | global = global || {};
13 | 'use strict';
14 | // existing version for noConflict()
15 | var _Base64 = global.Base64;
16 | var version = "2.1.9";
17 | // if node.js, we use Buffer
18 | var buffer;
19 | // constants
20 | var b64chars
21 | = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
22 | var b64tab = function(bin) {
23 | var t = {};
24 | for (var i = 0, l = bin.length; i < l; i++) t[bin.charAt(i)] = i;
25 | return t;
26 | }(b64chars);
27 | var fromCharCode = String.fromCharCode;
28 | // encoder stuff
29 | var cb_utob = function(c) {
30 | if (c.length < 2) {
31 | var cc = c.charCodeAt(0);
32 | return cc < 0x80 ? c
33 | : cc < 0x800 ? (fromCharCode(0xc0 | (cc >>> 6))
34 | + fromCharCode(0x80 | (cc & 0x3f)))
35 | : (fromCharCode(0xe0 | ((cc >>> 12) & 0x0f))
36 | + fromCharCode(0x80 | ((cc >>> 6) & 0x3f))
37 | + fromCharCode(0x80 | ( cc & 0x3f)));
38 | } else {
39 | var cc = 0x10000
40 | + (c.charCodeAt(0) - 0xD800) * 0x400
41 | + (c.charCodeAt(1) - 0xDC00);
42 | return (fromCharCode(0xf0 | ((cc >>> 18) & 0x07))
43 | + fromCharCode(0x80 | ((cc >>> 12) & 0x3f))
44 | + fromCharCode(0x80 | ((cc >>> 6) & 0x3f))
45 | + fromCharCode(0x80 | ( cc & 0x3f)));
46 | }
47 | };
48 | var re_utob = /[\uD800-\uDBFF][\uDC00-\uDFFFF]|[^\x00-\x7F]/g;
49 | var utob = function(u) {
50 | return u.replace(re_utob, cb_utob);
51 | };
52 | var cb_encode = function(ccc) {
53 | var padlen = [0, 2, 1][ccc.length % 3],
54 | ord = ccc.charCodeAt(0) << 16
55 | | ((ccc.length > 1 ? ccc.charCodeAt(1) : 0) << 8)
56 | | ((ccc.length > 2 ? ccc.charCodeAt(2) : 0)),
57 | chars = [
58 | b64chars.charAt( ord >>> 18),
59 | b64chars.charAt((ord >>> 12) & 63),
60 | padlen >= 2 ? '=' : b64chars.charAt((ord >>> 6) & 63),
61 | padlen >= 1 ? '=' : b64chars.charAt(ord & 63)
62 | ];
63 | return chars.join('');
64 | };
65 | var btoa = global.btoa ? function(b) {
66 | return global.btoa(b);
67 | } : function(b) {
68 | return b.replace(/[\s\S]{1,3}/g, cb_encode);
69 | };
70 | var _encode = buffer ? function (u) {
71 | return (u.constructor === buffer.constructor ? u : new buffer(u))
72 | .toString('base64')
73 | }
74 | : function (u) { return btoa(utob(u)) }
75 | ;
76 | var encode = function(u, urisafe) {
77 | return !urisafe
78 | ? _encode(String(u))
79 | : _encode(String(u)).replace(/[+\/]/g, function(m0) {
80 | return m0 == '+' ? '-' : '_';
81 | }).replace(/=/g, '');
82 | };
83 | var encodeURI = function(u) { return encode(u, true) };
84 | // decoder stuff
85 | var re_btou = new RegExp([
86 | '[\xC0-\xDF][\x80-\xBF]',
87 | '[\xE0-\xEF][\x80-\xBF]{2}',
88 | '[\xF0-\xF7][\x80-\xBF]{3}'
89 | ].join('|'), 'g');
90 | var cb_btou = function(cccc) {
91 | switch(cccc.length) {
92 | case 4:
93 | var cp = ((0x07 & cccc.charCodeAt(0)) << 18)
94 | | ((0x3f & cccc.charCodeAt(1)) << 12)
95 | | ((0x3f & cccc.charCodeAt(2)) << 6)
96 | | (0x3f & cccc.charCodeAt(3)),
97 | offset = cp - 0x10000;
98 | return (fromCharCode((offset >>> 10) + 0xD800)
99 | + fromCharCode((offset & 0x3FF) + 0xDC00));
100 | case 3:
101 | return fromCharCode(
102 | ((0x0f & cccc.charCodeAt(0)) << 12)
103 | | ((0x3f & cccc.charCodeAt(1)) << 6)
104 | | (0x3f & cccc.charCodeAt(2))
105 | );
106 | default:
107 | return fromCharCode(
108 | ((0x1f & cccc.charCodeAt(0)) << 6)
109 | | (0x3f & cccc.charCodeAt(1))
110 | );
111 | }
112 | };
113 | var btou = function(b) {
114 | return b.replace(re_btou, cb_btou);
115 | };
116 | var cb_decode = function(cccc) {
117 | var len = cccc.length,
118 | padlen = len % 4,
119 | n = (len > 0 ? b64tab[cccc.charAt(0)] << 18 : 0)
120 | | (len > 1 ? b64tab[cccc.charAt(1)] << 12 : 0)
121 | | (len > 2 ? b64tab[cccc.charAt(2)] << 6 : 0)
122 | | (len > 3 ? b64tab[cccc.charAt(3)] : 0),
123 | chars = [
124 | fromCharCode( n >>> 16),
125 | fromCharCode((n >>> 8) & 0xff),
126 | fromCharCode( n & 0xff)
127 | ];
128 | chars.length -= [0, 0, 2, 1][padlen];
129 | return chars.join('');
130 | };
131 | var atob = global.atob ? function(a) {
132 | return global.atob(a);
133 | } : function(a){
134 | return a.replace(/[\s\S]{1,4}/g, cb_decode);
135 | };
136 | var _decode = buffer ? function(a) {
137 | return (a.constructor === buffer.constructor
138 | ? a : new buffer(a, 'base64')).toString();
139 | }
140 | : function(a) { return btou(atob(a)) };
141 | var decode = function(a){
142 | return _decode(
143 | String(a).replace(/[-_]/g, function(m0) { return m0 == '-' ? '+' : '/' })
144 | .replace(/[^A-Za-z0-9\+\/]/g, '')
145 | );
146 | };
147 | var noConflict = function() {
148 | var Base64 = global.Base64;
149 | global.Base64 = _Base64;
150 | return Base64;
151 | };
152 | // export Base64
153 | var Base64 = {
154 | VERSION: version,
155 | atob: atob,
156 | btoa: btoa,
157 | fromBase64: decode,
158 | toBase64: encode,
159 | utob: utob,
160 | encode: encode,
161 | encodeURI: encodeURI,
162 | btou: btou,
163 | decode: decode,
164 | noConflict: noConflict
165 | };
166 | return Base64;
167 | })();
168 |
169 | module.exports = Base64;
170 |
171 |
--------------------------------------------------------------------------------
/demo/lib/cos-auth.js:
--------------------------------------------------------------------------------
1 | !function(){"use strict";function t(t){return encodeURIComponent(t).replace(/!/g,"%21").replace(/'/g,"%27").replace(/\(/g,"%28").replace(/\)/g,"%29").replace(/\*/g,"%2A")}var e=e||function(t,e){var n={},r=n.lib={},i=function(){},s=r.Base={extend:function(t){i.prototype=this;var e=new i;return t&&e.mixIn(t),e.hasOwnProperty("init")||(e.init=function(){e.$super.init.apply(this,arguments)}),e.init.prototype=e,e.$super=this,e},create:function(){var t=this.extend();return t.init.apply(t,arguments),t},init:function(){},mixIn:function(t){for(var e in t)t.hasOwnProperty(e)&&(this[e]=t[e]);t.hasOwnProperty("toString")&&(this.toString=t.toString)},clone:function(){return this.init.prototype.extend(this)}},o=r.WordArray=s.extend({init:function(t,e){t=this.words=t||[],this.sigBytes=void 0!=e?e:4*t.length},toString:function(t){return(t||c).stringify(this)},concat:function(t){var e=this.words,n=t.words,r=this.sigBytes;if(t=t.sigBytes,this.clamp(),r%4)for(var i=0;i>>2]|=(n[i>>>2]>>>24-i%4*8&255)<<24-(r+i)%4*8;else if(65535>>2]=n[i>>>2];else e.push.apply(e,n);return this.sigBytes+=t,this},clamp:function(){var e=this.words,n=this.sigBytes;e[n>>>2]&=4294967295<<32-n%4*8,e.length=t.ceil(n/4)},clone:function(){var t=s.clone.call(this);return t.words=this.words.slice(0),t},random:function(e){for(var n=[],r=0;r>>2]>>>24-r%4*8&255;n.push((i>>>4).toString(16)),n.push((15&i).toString(16))}return n.join("")},parse:function(t){for(var e=t.length,n=[],r=0;r>>3]|=parseInt(t.substr(r,2),16)<<24-r%8*4;return new o.init(n,e/2)}},h=a.Latin1={stringify:function(t){var e=t.words;t=t.sigBytes;for(var n=[],r=0;r>>2]>>>24-r%4*8&255));return n.join("")},parse:function(t){for(var e=t.length,n=[],r=0;r>>2]|=(255&t.charCodeAt(r))<<24-r%4*8;return new o.init(n,e)}},u=a.Utf8={stringify:function(t){try{return decodeURIComponent(escape(h.stringify(t)))}catch(t){throw Error("Malformed UTF-8 data")}},parse:function(t){return h.parse(unescape(encodeURIComponent(t)))}},f=r.BufferedBlockAlgorithm=s.extend({reset:function(){this._data=new o.init,this._nDataBytes=0},_append:function(t){"string"==typeof t&&(t=u.parse(t)),this._data.concat(t),this._nDataBytes+=t.sigBytes},_process:function(e){var n=this._data,r=n.words,i=n.sigBytes,s=this.blockSize,a=i/(4*s),a=e?t.ceil(a):t.max((0|a)-this._minBufferSize,0);if(e=a*s,i=t.min(4*e,i),e){for(var c=0;ch;h++){if(16>h)s[h]=0|t[e+h];else{var u=s[h-3]^s[h-8]^s[h-14]^s[h-16];s[h]=u<<1|u>>>31}u=(r<<5|r>>>27)+c+s[h],u=20>h?u+(1518500249+(i&o|~i&a)):40>h?u+(1859775393+(i^o^a)):60>h?u+((i&o|i&a|o&a)-1894007588):u+((i^o^a)-899497514),c=a,a=o,o=i<<30|i>>>2,i=r,r=u}n[0]=n[0]+r|0,n[1]=n[1]+i|0,n[2]=n[2]+o|0,n[3]=n[3]+a|0,n[4]=n[4]+c|0},_doFinalize:function(){var t=this._data,e=t.words,n=8*this._nDataBytes,r=8*t.sigBytes;return e[r>>>5]|=128<<24-r%32,e[14+(r+64>>>9<<4)]=Math.floor(n/4294967296),e[15+(r+64>>>9<<4)]=n,t.sigBytes=4*e.length,this._process(),this._hash},clone:function(){var t=i.clone.call(this);return t._hash=this._hash.clone(),t}});t.SHA1=i._createHelper(n),t.HmacSHA1=i._createHmacHelper(n)}(),function(){var t=e,n=t.enc.Utf8;t.algo.HMAC=t.lib.Base.extend({init:function(t,e){t=this._hasher=new t.init,"string"==typeof e&&(e=n.parse(e));var r=t.blockSize,i=4*r;e.sigBytes>i&&(e=t.finalize(e)),e.clamp();for(var s=this._oKey=e.clone(),o=this._iKey=e.clone(),a=s.words,c=o.words,h=0;h>>2]>>>24-s%4*8&255,a=e[s+1>>>2]>>>24-(s+1)%4*8&255,c=e[s+2>>>2]>>>24-(s+2)%4*8&255,h=o<<16|a<<8|c,u=0;u<4&&s+.75*u>>6*(3-u)&63));var f=r.charAt(64);if(f)for(;i.length%4;)i.push(f);return i.join("")},parse:function(t){var e=t.length,n=this._map,i=n.charAt(64);if(i){var s=t.indexOf(i);-1!=s&&(e=s)}for(var o=[],a=0,c=0;c>>6-c%4*2;o[a>>>2]|=(h|u)<<24-a%4*8,a++}return r.create(o,a)},_map:"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="}}();var n=function(t){var n=t.Pathname||"/",r=t.Expires,i="",s="",o=t.Bucket.match(/^(.+)-(\d+)$/);o&&(i=o[1],s=o[2]);var a=parseInt(Math.random()*Math.pow(2,32)),c=parseInt(Date.now()/1e3),h=c+(void 0===r?900:1*r||0),u="/"+s+"/"+i+encodeURIComponent(n).replace(/%2F/g,"/"),f="a="+s+"&b="+i+"&k="+t.SecretId+"&e="+h+"&t="+c+"&r="+a+"&f="+u,l=e.HmacSHA1(f,t.SecretKey),p=e.enc.Utf8.parse(f);return l.concat(p).toString(e.enc.Base64)},r=function(r){if(!r.SecretId)return console.error("missing param SecretId");if(!r.SecretKey)return console.error("missing param SecretKey");if("4.0"===r.Version)return n(r);r=r||{};var i=r.SecretId,s=r.SecretKey,o=(r.Method||"get").toLowerCase(),a=r.Query||{},c=r.Headers||{},h=r.Pathname||"/",u=r.Expires,f=function(t){var e=[];for(var n in t)t.hasOwnProperty(n)&&e.push(n);return e.sort(function(t,e){return t=t.toLowerCase(),e=e.toLowerCase(),t===e?0:t>e?1:-1})},l=function(e){var n,r,i,s=[],o=f(e);for(n=0;n ('000000' + x).slice(-(n || 2));
43 | var d = new Date();
44 | var ymd = d.getFullYear() + N(d.getMonth() + 1) + N(d.getDate());
45 | var hms = N(d.getHours()) + N(d.getMinutes()) + N(d.getSeconds());
46 | var r = N(Math.round((Math.random() * 1000000)), 6);
47 | var fileName = ymd + '_' + hms + '_' + r;
48 | if (ext) fileName += '.' + ext;
49 | return fileName;
50 | };
51 |
52 | // 创建临时密钥服务
53 | var app = express();
54 | app.use(bodyParser.json());
55 |
56 | // 格式一:临时密钥接口
57 | app.all('/sts', function (req, res, next) {
58 |
59 | // TODO 这里根据自己业务需要做好放行判断
60 | if (config.allowPrefix === '_ALLOW_DIR_/*') {
61 | res.send({error: '请修改 allowPrefix 配置项,指定允许上传的路径前缀'});
62 | return;
63 | }
64 |
65 | // 获取临时密钥
66 | var LongBucketName = config.bucket;
67 | var ShortBucketName = LongBucketName.substr(0, LongBucketName.lastIndexOf('-'));
68 | var AppId = LongBucketName.substr(LongBucketName.lastIndexOf('-') + 1);
69 | var policy = {
70 | 'version': '2.0',
71 | 'statement': [{
72 | 'action': config.allowActions,
73 | 'effect': 'allow',
74 | 'resource': [
75 | 'qcs::cos:' + config.region + ':uid/' + AppId + ':prefix//' + AppId + '/' + ShortBucketName + '/' + config.allowPrefix,
76 | ],
77 | }],
78 | };
79 | var startTime = Math.round(Date.now() / 1000);
80 | STS.getCredential({
81 | secretId: config.secretId,
82 | secretKey: config.secretKey,
83 | proxy: config.proxy,
84 | region: config.region,
85 | durationSeconds: config.durationSeconds,
86 | policy: policy,
87 | }, function (err, tempKeys) {
88 | if (tempKeys) tempKeys.startTime = startTime;
89 | res.send(err || tempKeys);
90 | });
91 | });
92 |
93 |
94 | // // 格式二:临时密钥接口,支持细粒度权限控制
95 | // // 判断是否允许获取密钥
96 | // var allowScope = function (scope) {
97 | // var allow = (scope || []).every(function (item) {
98 | // return config.allowActions.includes(item.action) &&
99 | // item.bucket === config.bucket &&
100 | // item.region === config.region &&
101 | // (item.prefix || '').startsWith(config.allowPrefix);
102 | // });
103 | // return allow;
104 | // };
105 | // app.all('/sts-scope', function (req, res, next) {
106 | // var scope = req.body;
107 | //
108 | // // TODO 这里根据自己业务需要做好放行判断
109 | // if (config.allowPrefix === '_ALLOW_DIR_/*') {
110 | // res.send({error: '请修改 allowPrefix 配置项,指定允许上传的路径前缀'});
111 | // return;
112 | // }
113 | // // TODO 这里可以判断 scope 细粒度控制权限
114 | // if (!scope || !scope.length || !allowScope(scope)) return res.send({error: 'deny'});
115 | //
116 | // // 获取临时密钥
117 | // var policy = STS.getPolicy(scope);
118 | // var startTime = Math.round(Date.now() / 1000);
119 | // STS.getCredential({
120 | // secretId: config.secretId,
121 | // secretKey: config.secretKey,
122 | // proxy: config.proxy,
123 | // durationSeconds: config.durationSeconds,
124 | // policy: policy,
125 | // }, function (err, tempKeys) {
126 | // if (tempKeys) tempKeys.startTime = startTime;
127 | // res.send(err || tempKeys);
128 | // });
129 | // });
130 | //
131 | // // 用于 PostObject 签名保护
132 | // app.all('/post-policy', function (req, res, next) {
133 | // var query = req.query;
134 | // var ext = query.ext;
135 | // var key = 'images/' + createFileName(ext);
136 | // var now = Math.round(Date.now() / 1000);
137 | // var exp = now + 900;
138 | // var qKeyTime = now + ';' + exp;
139 | // var qSignAlgorithm = 'sha1';
140 | // var policy = JSON.stringify({
141 | // 'expiration': new Date(exp * 1000).toISOString(),
142 | // 'conditions': [
143 | // // {'acl': query.ACL},
144 | // // ['starts-with', '$Content-Type', 'image/'],
145 | // // ['starts-with', '$success_action_redirect', redirectUrl],
146 | // // ['eq', '$x-cos-server-side-encryption', 'AES256'],
147 | // {'q-sign-algorithm': qSignAlgorithm},
148 | // {'q-ak': config.secretId},
149 | // {'q-sign-time': qKeyTime},
150 | // {'bucket': config.bucket},
151 | // {'key': key},
152 | // ]
153 | // });
154 | //
155 | // // 签名算法说明文档:https://www.qcloud.com/document/product/436/7778
156 | // // 步骤一:生成 SignKey
157 | // var signKey = crypto.createHmac('sha1', config.secretKey).update(qKeyTime).digest('hex');
158 | //
159 | // // 步骤二:生成 StringToSign
160 | // var stringToSign = crypto.createHash('sha1').update(policy).digest('hex');
161 | //
162 | // // 步骤三:生成 Signature
163 | // var qSignature = crypto.createHmac('sha1', signKey).update(stringToSign).digest('hex');
164 | //
165 | // console.log(policy);
166 | // res.send({
167 | // bucket: config.bucket,
168 | // region: config.region,
169 | // key: key,
170 | // policyObj: JSON.parse(policy),
171 | // policy: Buffer.from(policy).toString('base64'),
172 | // qSignAlgorithm: qSignAlgorithm,
173 | // qAk: config.secretId,
174 | // qKeyTime: qKeyTime,
175 | // qSignature: qSignature,
176 | // // securityToken: securityToken, // 如果使用临时密钥,要返回在这个资源 sessionToken 的值
177 | // });
178 | // });
179 |
180 | app.all('*', function (req, res, next) {
181 | res.send({code: -1, message: '404 Not Found'});
182 | });
183 |
184 | // 启动签名服务
185 | app.listen(3000);
186 | console.log('app is listening at http://127.0.0.1:3000');
187 |
--------------------------------------------------------------------------------
/demo/ciDemo/audit.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | const config = require('../config');
3 | const { cos, requestCallback } = require('../tools');
4 |
5 | const auditDao = {
6 | '图片同步审核 getImageAuditing': getImageAuditing,
7 | '图片批量审核 postImagesAuditing': postImagesAuditing,
8 | '查询图片审核任务结果 getImageAuditingResult': getImageAuditingResult,
9 | '提交视频审核任务 postVideoAuditing': postVideoAuditing,
10 | '查询视频审核任务结果 getVideoAuditingResult': getVideoAuditingResult,
11 | '提交音频审核任务 postAudioAuditing': postAudioAuditing,
12 | '查询音频审核任务结果 getAudioAuditingResult': getAudioAuditingResult,
13 | '提交文本审核任务 postTextAuditing': postTextAuditing,
14 | '提交直播审核任务 postLiveAuditing': postLiveAuditing,
15 | '查询直播审核任务结果 getLiveAuditingResult': getLiveAuditingResult,
16 | };
17 |
18 | function getImageAuditing() {
19 | cos.request(
20 | {
21 | Bucket: config.Bucket,
22 | Region: config.Region,
23 | Method: 'GET',
24 | Key: '1.png',
25 | Query: {
26 | 'ci-process': 'sensitive-content-recognition' /** 固定值,必须 */,
27 | 'biz-type': '' /** 审核类型,非必须 */,
28 | 'detect-url': '' /** 审核任意公网可访问的图片链接,非必须,与Key二选一传递 */,
29 | interval: 5 /** 审核 GIF 动图时,每隔interval帧截取一帧,非必须 */,
30 | 'max-frames': 5 /** 审核 GIF 动图时,最大截帧数,非必须 */,
31 | 'large-image-detect': '0' /** 是否需要压缩图片后再审核,非必须 */,
32 | },
33 | },
34 | function (err, data) {
35 | console.log(err || data);
36 | }
37 | );
38 | }
39 |
40 | function postImagesAuditing() {
41 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
42 | var url = 'https://' + host + '/image/auditing';
43 | var body = COS.util.json2xml({
44 | Request: {
45 | Input: [
46 | {
47 | Object: '1.png',
48 | },
49 | {
50 | Object: '6.png',
51 | },
52 | ],
53 | Conf: {
54 | BizType: '',
55 | },
56 | },
57 | });
58 | cos.request(
59 | {
60 | Bucket: config.Bucket,
61 | Region: config.Region,
62 | Method: 'POST',
63 | Url: url,
64 | Key: '/image/auditing',
65 | ContentType: 'application/xml',
66 | Body: body,
67 | },
68 | function (err, data) {
69 | console.log(err || data);
70 | }
71 | );
72 | }
73 |
74 | function getImageAuditingResult() {
75 | var jobId = 'si8263213daf3711eca0d1525400d88xxx'; // jobId可以通过图片批量审核返回
76 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
77 | var url = 'https://' + host + '/image/auditing/' + jobId;
78 | cos.request(
79 | {
80 | Bucket: config.Bucket,
81 | Region: config.Region,
82 | Method: 'GET',
83 | Key: '/image/auditing/' + jobId,
84 | Url: url,
85 | },
86 | function (err, data) {
87 | console.log(err || data);
88 | }
89 | );
90 | }
91 |
92 | function postVideoAuditing() {
93 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
94 | var url = 'https://' + host + '/video/auditing';
95 | var body = COS.util.json2xml({
96 | Request: {
97 | Input: {
98 | Object: '1.mp4',
99 | },
100 | Conf: {
101 | BizType: '',
102 | Snapshot: {
103 | Count: 1000, // 视频截帧数量
104 | },
105 | DetectContent: 1, // 是否审核视频声音,0-只审核视频不审核声音;1-审核视频+声音
106 | },
107 | },
108 | });
109 | cos.request(
110 | {
111 | Bucket: config.Bucket,
112 | Region: config.Region,
113 | Method: 'POST',
114 | Url: url,
115 | Key: '/video/auditing',
116 | ContentType: 'application/xml',
117 | Body: body,
118 | },
119 | function (err, data) {
120 | console.log(err || data);
121 | }
122 | );
123 | }
124 |
125 | function getVideoAuditingResult() {
126 | var jobId = 'av5bd873d9f39011ecb6c95254009a49da'; // jobId可以通过提交视频审核任务返回
127 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
128 | var url = 'https://' + host + '/video/auditing/' + jobId;
129 | cos.request(
130 | {
131 | Bucket: config.Bucket,
132 | Region: config.Region,
133 | Method: 'GET',
134 | Key: '/video/auditing/' + jobId,
135 | Url: url,
136 | },
137 | function (err, data) {
138 | console.log(err || data);
139 | }
140 | );
141 | }
142 |
143 | function postAudioAuditing() {
144 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
145 | var url = 'https://' + host + '/audio/auditing';
146 | var body = COS.util.json2xml({
147 | Request: {
148 | Input: {
149 | Object: '1.mp3',
150 | },
151 | Conf: {
152 | BizType: '',
153 | },
154 | },
155 | });
156 | cos.request(
157 | {
158 | Bucket: config.Bucket,
159 | Region: config.Region,
160 | Method: 'POST',
161 | Url: url,
162 | Key: '/audio/auditing',
163 | ContentType: 'application/xml',
164 | Body: body,
165 | },
166 | function (err, data) {
167 | console.log(err || data);
168 | }
169 | );
170 | }
171 |
172 | function getAudioAuditingResult() {
173 | var jobId = 'sa0c28d41daff411ecb23352540078cxxx'; // jobId可以通过提交音频审核任务返回
174 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
175 | var url = 'https://' + host + '/audio/auditing/' + jobId;
176 | cos.request(
177 | {
178 | Bucket: config.Bucket,
179 | Region: config.Region,
180 | Method: 'GET',
181 | Key: '/audio/auditing/' + jobId,
182 | Url: url,
183 | },
184 | function (err, data) {
185 | console.log(err || data);
186 | }
187 | );
188 | }
189 |
190 | function postTextAuditing() {
191 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
192 | var url = 'https://' + host + '/text/auditing';
193 | var body = COS.util.json2xml({
194 | Request: {
195 | Input: {
196 | Content: COS.util.encodeBase64('hello') /* 需要审核的文本内容 */,
197 | },
198 | Conf: {
199 | BizType: '',
200 | },
201 | },
202 | });
203 | cos.request(
204 | {
205 | Bucket: config.Bucket,
206 | Region: config.Region,
207 | Method: 'POST',
208 | Url: url,
209 | Key: '/text/auditing' /** 固定值,必须 */,
210 | ContentType: 'application/xml' /** 固定值,必须 */,
211 | Body: body,
212 | },
213 | function (err, data) {
214 | console.log(err || data);
215 | }
216 | );
217 | }
218 |
219 | function postLiveAuditing() {
220 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
221 | var url = 'https://' + host + '/video/auditing';
222 | var body = COS.util.json2xml({
223 | Request: {
224 | Type: 'live_video',
225 | Input: {
226 | Url: 'rtmp://example.com/live/123', // 需要审核的直播流播放地址
227 | // DataId: '',
228 | // UserInfo: {},
229 | },
230 | Conf: {
231 | BizType: '',
232 | // Callback: 'https://callback.com', // 回调地址,非必须
233 | // CallbackType: 1, // 回调片段类型,非必须
234 | },
235 | },
236 | });
237 | cos.request(
238 | {
239 | Bucket: config.Bucket,
240 | Region: config.Region,
241 | Method: 'POST',
242 | Url: url,
243 | Key: '/video/auditing',
244 | ContentType: 'application/xml',
245 | Body: body,
246 | },
247 | function (err, data) {
248 | console.log(err || data);
249 | }
250 | );
251 | }
252 |
253 | function getLiveAuditingResult() {
254 | var jobId = 'av2b14a74dbd9011edb05a52540084c0xx'; // jobId可以通过提交直播审核任务返回
255 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com';
256 | var url = 'https://' + host + '/video/auditing/' + jobId;
257 | cos.request(
258 | {
259 | Bucket: config.Bucket,
260 | Region: config.Region,
261 | Method: 'GET',
262 | Key: '/video/auditing/' + jobId,
263 | Url: url,
264 | },
265 | function (err, data) {
266 | console.log(err || data);
267 | }
268 | );
269 | }
270 |
271 | module.exports = auditDao;
272 |
--------------------------------------------------------------------------------
/lib/crypto.js:
--------------------------------------------------------------------------------
1 | /*
2 | CryptoJS v3.1.2
3 | code.google.com/p/crypto-js
4 | (c) 2009-2013 by Jeff Mott. All rights reserved.
5 | code.google.com/p/crypto-js/wiki/License
6 | */
7 | var CryptoJS=CryptoJS||function(g,l){var e={},d=e.lib={},m=function(){},k=d.Base={extend:function(a){m.prototype=this;var c=new m;a&&c.mixIn(a);c.hasOwnProperty("init")||(c.init=function(){c.$super.init.apply(this,arguments)});c.init.prototype=c;c.$super=this;return c},create:function(){var a=this.extend();a.init.apply(a,arguments);return a},init:function(){},mixIn:function(a){for(var c in a)a.hasOwnProperty(c)&&(this[c]=a[c]);a.hasOwnProperty("toString")&&(this.toString=a.toString)},clone:function(){return this.init.prototype.extend(this)}},
8 | p=d.WordArray=k.extend({init:function(a,c){a=this.words=a||[];this.sigBytes=c!=l?c:4*a.length},toString:function(a){return(a||n).stringify(this)},concat:function(a){var c=this.words,q=a.words,f=this.sigBytes;a=a.sigBytes;this.clamp();if(f%4)for(var b=0;b>>2]|=(q[b>>>2]>>>24-8*(b%4)&255)<<24-8*((f+b)%4);else if(65535>>2]=q[b>>>2];else c.push.apply(c,q);this.sigBytes+=a;return this},clamp:function(){var a=this.words,c=this.sigBytes;a[c>>>2]&=4294967295<<
9 | 32-8*(c%4);a.length=g.ceil(c/4)},clone:function(){var a=k.clone.call(this);a.words=this.words.slice(0);return a},random:function(a){for(var c=[],b=0;b>>2]>>>24-8*(f%4)&255;b.push((d>>>4).toString(16));b.push((d&15).toString(16))}return b.join("")},parse:function(a){for(var c=a.length,b=[],f=0;f>>3]|=parseInt(a.substr(f,
10 | 2),16)<<24-4*(f%8);return new p.init(b,c/2)}},j=b.Latin1={stringify:function(a){var c=a.words;a=a.sigBytes;for(var b=[],f=0;f>>2]>>>24-8*(f%4)&255));return b.join("")},parse:function(a){for(var c=a.length,b=[],f=0;f>>2]|=(a.charCodeAt(f)&255)<<24-8*(f%4);return new p.init(b,c)}},h=b.Utf8={stringify:function(a){try{return decodeURIComponent(escape(j.stringify(a)))}catch(c){throw Error("Malformed UTF-8 data");}},parse:function(a){return j.parse(unescape(encodeURIComponent(a)))}},
11 | r=d.BufferedBlockAlgorithm=k.extend({reset:function(){this._data=new p.init;this._nDataBytes=0},_append:function(a){"string"==typeof a&&(a=h.parse(a));this._data.concat(a);this._nDataBytes+=a.sigBytes},_process:function(a){var c=this._data,b=c.words,f=c.sigBytes,d=this.blockSize,e=f/(4*d),e=a?g.ceil(e):g.max((e|0)-this._minBufferSize,0);a=e*d;f=g.min(4*a,f);if(a){for(var k=0;ka;a++){if(16>a)m[a]=d[e+a]|0;else{var c=m[a-3]^m[a-8]^m[a-14]^m[a-16];m[a]=c<<1|c>>>31}c=(n<<5|n>>>27)+l+m[a];c=20>a?c+((j&h|~j&g)+1518500249):40>a?c+((j^h^g)+1859775393):60>a?c+((j&h|j&g|h&g)-1894007588):c+((j^h^
15 | g)-899497514);l=g;g=h;h=j<<30|j>>>2;j=n;n=c}b[0]=b[0]+n|0;b[1]=b[1]+j|0;b[2]=b[2]+h|0;b[3]=b[3]+g|0;b[4]=b[4]+l|0},_doFinalize:function(){var d=this._data,e=d.words,b=8*this._nDataBytes,g=8*d.sigBytes;e[g>>>5]|=128<<24-g%32;e[(g+64>>>9<<4)+14]=Math.floor(b/4294967296);e[(g+64>>>9<<4)+15]=b;d.sigBytes=4*e.length;this._process();return this._hash},clone:function(){var e=d.clone.call(this);e._hash=this._hash.clone();return e}});g.SHA1=d._createHelper(l);g.HmacSHA1=d._createHmacHelper(l)})();
16 | (function(){var g=CryptoJS,l=g.enc.Utf8;g.algo.HMAC=g.lib.Base.extend({init:function(e,d){e=this._hasher=new e.init;"string"==typeof d&&(d=l.parse(d));var g=e.blockSize,k=4*g;d.sigBytes>k&&(d=e.finalize(d));d.clamp();for(var p=this._oKey=d.clone(),b=this._iKey=d.clone(),n=p.words,j=b.words,h=0;h>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
57 | var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;
58 | var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;
59 |
60 | var triplet = (byte1 << 16) | (byte2 << 8) | byte3;
61 |
62 | for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {
63 | base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));
64 | }
65 | }
66 |
67 | // Add padding
68 | var paddingChar = map.charAt(64);
69 | if (paddingChar) {
70 | while (base64Chars.length % 4) {
71 | base64Chars.push(paddingChar);
72 | }
73 | }
74 |
75 | return base64Chars.join('');
76 | },
77 |
78 | /**
79 | * Converts a Base64 string to a word array.
80 | *
81 | * @param {string} base64Str The Base64 string.
82 | *
83 | * @return {WordArray} The word array.
84 | *
85 | * @static
86 | *
87 | * @example
88 | *
89 | * var wordArray = CryptoJS.enc.Base64.parse(base64String);
90 | */
91 | parse: function (base64Str) {
92 | // Shortcuts
93 | var base64StrLength = base64Str.length;
94 | var map = this._map;
95 |
96 | // Ignore padding
97 | var paddingChar = map.charAt(64);
98 | if (paddingChar) {
99 | var paddingIndex = base64Str.indexOf(paddingChar);
100 | if (paddingIndex != -1) {
101 | base64StrLength = paddingIndex;
102 | }
103 | }
104 |
105 | // Convert
106 | var words = [];
107 | var nBytes = 0;
108 | for (var i = 0; i < base64StrLength; i++) {
109 | if (i % 4) {
110 | var bits1 = map.indexOf(base64Str.charAt(i - 1)) << ((i % 4) * 2);
111 | var bits2 = map.indexOf(base64Str.charAt(i)) >>> (6 - (i % 4) * 2);
112 | words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);
113 | nBytes++;
114 | }
115 | }
116 |
117 | return WordArray.create(words, nBytes);
118 | },
119 |
120 | _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
121 | };
122 | }());
123 |
124 | module.exports = CryptoJS;
125 |
--------------------------------------------------------------------------------
/demo/ciDemo/asr.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | const config = require('../config');
3 | const { cos, requestCallback } = require('../tools');
4 |
5 | const asrDao = {
6 | '提交音频降噪任务 postNoiseReduction': postNoiseReduction,
7 | '提交人声分离任务 postVoiceSeparate': postVoiceSeparate,
8 | '提交语音合成任务 postTts': postTts,
9 | '提交语音识别任务 postSpeechRecognition': postSpeechRecognition,
10 | '查询语音识别队列 getAsrQueue': getAsrQueue,
11 | '更新语音识别队列 putAsrQueue': putAsrQueue,
12 | '查询语音识别开通状态 getAsrBucket': getAsrBucket,
13 | };
14 |
15 | function postNoiseReduction() {
16 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/jobs';
17 | var url = 'https://' + host;
18 | var body = COS.util.json2xml({
19 | Request: {
20 | Tag: 'NoiseReduction',
21 | Input: {
22 | Object: 'ci/music.mp3', // 文件名,取值为文件在当前存储桶中的完整名称
23 | },
24 | Operation: {
25 | Output: {
26 | Bucket: config.Bucket, // 输出的存储桶
27 | Region: config.Region, // 输出的存储桶的地域
28 | Object: 'ci/out.mp3', // 输出的文件Key
29 | },
30 | },
31 | // QueueId: '', // 任务所在的队列 ID,非必须
32 | // CallBackFormat: '', // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式,非必须
33 | // CallBackType: '', // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型,非必须
34 | // CallBack: '', // 任务回调地址,优先级高于队列的回调地址。设置为 no 时,表示队列的回调地址不产生回调,非必须
35 | // CallBackMqConfig: '', // 任务回调 TDMQ 配置,当 CallBackType 为 TDMQ 时必填,非必须
36 | },
37 | });
38 | cos.request(
39 | {
40 | Method: 'POST',
41 | Key: 'jobs',
42 | Url: url,
43 | Body: body,
44 | ContentType: 'application/xml',
45 | },
46 | function (err, data) {
47 | console.log(err || data);
48 | }
49 | );
50 | }
51 |
52 | function postVoiceSeparate() {
53 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/jobs';
54 | var url = 'https://' + host;
55 | var body = COS.util.json2xml({
56 | Request: {
57 | Tag: 'VoiceSeparate',
58 | Input: {
59 | Object: 'ci/music.mp3', // 文件名,取值为文件在当前存储桶中的完整名称
60 | },
61 | Operation: {
62 | // VoiceSeparate: {}, // 指定转码模板参数,非必须
63 | TemplateId: 't13fca82ad97e84878a22cd81bd2e5652c', // 指定的模板 ID,必须
64 | // JobLevel: 0, // 任务优先级,级别限制:0 、1 、2。级别越大任务优先级越高,默认为0,非必须
65 | Output: {
66 | Bucket: config.Bucket, // 输出的存储桶
67 | Region: config.Region, // 输出的存储桶的地域
68 | Object: 'ci/out/background.mp3', // 输出的文件Key,背景音结果文件名,不能与 AuObject 同时为空
69 | AuObject: 'ci/out/audio.mp3',
70 | },
71 | },
72 | // QueueId: '', // 任务所在的队列 ID,非必须
73 | // CallBackFormat: '', // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式,非必须
74 | // CallBackType: '', // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型,非必须
75 | // CallBack: '', // 任务回调地址,优先级高于队列的回调地址。设置为 no 时,表示队列的回调地址不产生回调,非必须
76 | // CallBackMqConfig: '', // 任务回调 TDMQ 配置,当 CallBackType 为 TDMQ 时必填,非必须
77 | },
78 | });
79 | cos.request(
80 | {
81 | Method: 'POST',
82 | Key: 'jobs',
83 | Url: url,
84 | Body: body,
85 | ContentType: 'application/xml',
86 | },
87 | function (err, data) {
88 | console.log(err || data);
89 | }
90 | );
91 | }
92 |
93 | function postTts() {
94 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/jobs';
95 | var url = 'https://' + host;
96 | var body = COS.util.json2xml({
97 | Request: {
98 | Tag: 'Tts',
99 | Operation: {
100 | // VoiceSeparate: {}, // 指定转码模板参数,非必须
101 | TemplateId: 't192931b3564084168a3f50ebfea59acb3', // 指定的模板 ID,必须
102 | // JobLevel: 0, // 任务优先级,级别限制:0 、1 、2。级别越大任务优先级越高,默认为0,非必须
103 | TtsConfig: {
104 | InputType: 'Text',
105 | Input: '床前明月光,疑是地上霜',
106 | },
107 | Output: {
108 | Bucket: config.Bucket, // 输出的存储桶
109 | Region: config.Region, // 输出的存储桶的地域
110 | Object: 'ci/out/tts.mp3', // 输出的文件Key
111 | },
112 | },
113 | // QueueId: '', // 任务所在的队列 ID,非必须
114 | // CallBackFormat: '', // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式,非必须
115 | // CallBackType: '', // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型,非必须
116 | // CallBack: '', // 任务回调地址,优先级高于队列的回调地址。设置为 no 时,表示队列的回调地址不产生回调,非必须
117 | // CallBackMqConfig: '', // 任务回调 TDMQ 配置,当 CallBackType 为 TDMQ 时必填,非必须
118 | },
119 | });
120 | cos.request(
121 | {
122 | Method: 'POST',
123 | Key: 'jobs',
124 | Url: url,
125 | Body: body,
126 | ContentType: 'application/xml',
127 | },
128 | function (err, data) {
129 | console.log(err || data);
130 | }
131 | );
132 | }
133 |
134 | function postSpeechRecognition() {
135 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/asr_jobs';
136 | var url = 'https://' + host;
137 | var body = COS.util.json2xml({
138 | Request: {
139 | Tag: 'SpeechRecognition',
140 | Input: {
141 | Object: 'ci/music.mp3', // 文件名,取值为文件在当前存储桶中的完整名称,与Url参数二选一
142 | // Url: 'http://examplebucket-1250000000.cos.ap-shanghai.myqcloud.com/music.mp3', // 病毒文件的链接地址,与Object参数二选一
143 | },
144 | Operation: {
145 | SpeechRecognition: {
146 | EngineModelType: '16k_zh_video', // 引擎模型类型
147 | ChannelNum: 1, // 语音声道数
148 | ResTextFormat: 0, // 识别结果返回形式
149 | FilterDirty: 1, // 是否过滤脏词(目前支持中文普通话引擎)
150 | FilterModal: 1, // 是否过语气词(目前支持中文普通话引擎)
151 | ConvertNumMode: 0, // 是否进行阿拉伯数字智能转换(目前支持中文普通话引擎)
152 | },
153 | Output: {
154 | Bucket: config.Bucket, // 输出的存储桶
155 | Region: config.Region, // 输出的存储桶的地域
156 | Object: 'ci/out/SpeechRecognition.mp3', // 输出的文件Key
157 | },
158 | },
159 | // QueueId: '', // 任务所在的队列 ID,非必须
160 | // CallBackFormat: '', // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式,非必须
161 | // CallBackType: '', // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型,非必须
162 | // CallBack: '', // 任务回调地址,优先级高于队列的回调地址。设置为 no 时,表示队列的回调地址不产生回调,非必须
163 | // CallBackMqConfig: '', // 任务回调 TDMQ 配置,当 CallBackType 为 TDMQ 时必填,非必须
164 | },
165 | });
166 | cos.request(
167 | {
168 | Method: 'POST',
169 | Key: 'asr_jobs',
170 | Url: url,
171 | Body: body,
172 | ContentType: 'application/xml',
173 | },
174 | function (err, data) {
175 | console.log(err || data);
176 | }
177 | );
178 | }
179 |
180 | function getAsrQueue() {
181 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/asrqueue';
182 | var url = 'https://' + host;
183 | cos.request(
184 | {
185 | Method: 'GET',
186 | Key: 'asrqueue',
187 | Url: url,
188 | Query: {
189 | // queueIds: '', /* 非必须,队列 ID,以“,”符号分割字符串 */
190 | // state: '', /* 非必须,1=Active,2=Paused */
191 | // pageNumber: 1, /* 非必须,第几页 */
192 | // pageSize: 2, /* 非必须,每页个数 */
193 | },
194 | },
195 | function (err, data) {
196 | console.log(err || data);
197 | }
198 | );
199 | }
200 |
201 | function putAsrQueue() {
202 | // 任务所在的队列 ID,请使用查询队列(https://cloud.tencent.com/document/product/460/46234)获取或前往万象控制台(https://cloud.tencent.com/document/product/460/46487)在存储桶中查询
203 | var queueId = 'pcc77499e85c311edb9865254008618d9';
204 | var host = config.Bucket + '.ci.' + config.Region + '.myqcloud.com/asrqueue/' + queueId;
205 | var url = 'https://' + host;
206 | var body = COS.util.json2xml({
207 | Request: {
208 | Name: 'queue-doc-process-1',
209 | QueueID: queueId,
210 | State: 'Paused',
211 | NotifyConfig: {
212 | // Url: '',
213 | // Type: 'Url',
214 | // Event: '',
215 | State: 'Off',
216 | },
217 | },
218 | });
219 | cos.request(
220 | {
221 | Method: 'PUT',
222 | Key: 'asrqueue/' + queueId,
223 | Url: url,
224 | Body: body,
225 | ContentType: 'application/xml',
226 | },
227 | function (err, data) {
228 | console.log(err || data);
229 | }
230 | );
231 | }
232 |
233 | function getAsrBucket() {
234 | var host = 'ci.' + config.Region + '.myqcloud.com/asrbucket';
235 | var url = 'https://' + host;
236 | cos.request(
237 | {
238 | Method: 'GET',
239 | Key: 'asrbucket',
240 | Url: url,
241 | Query: {
242 | // regions: '', /* 非必须,地域信息,以“,”分隔字符串,支持 All、ap-shanghai、ap-beijing */
243 | // bucketNames: '', /* 非必须,存储桶名称,以“,”分隔,支持多个存储桶,精确搜索 */
244 | // bucketName: '', /* 非必须,存储桶名称前缀,前缀搜索 */
245 | // pageNumber: 1, /* 非必须,第几页 */
246 | // pageSize: 10, /* 非必须,每页个数 */
247 | },
248 | },
249 | function (err, data) {
250 | console.log(err || data);
251 | }
252 | );
253 | }
254 |
255 | module.exports = asrDao;
256 |
--------------------------------------------------------------------------------
/src/task.js:
--------------------------------------------------------------------------------
1 | var session = require('./session');
2 | var util = require('./util');
3 |
4 | var originApiMap = {};
5 | var transferToTaskMethod = function (apiMap, apiName) {
6 | originApiMap[apiName] = apiMap[apiName];
7 | apiMap[apiName] = function (params, callback) {
8 | if (params.SkipTask) {
9 | originApiMap[apiName].call(this, params, callback);
10 | } else {
11 | this._addTask(apiName, params, callback);
12 | }
13 | };
14 | };
15 |
16 | var initTask = function (cos) {
17 | var queue = [];
18 | var tasks = {};
19 | var uploadingFileCount = 0;
20 | var nextUploadIndex = 0;
21 |
22 | // 接口返回简略的任务信息
23 | var formatTask = function (task) {
24 | var t = {
25 | id: task.id,
26 | Bucket: task.Bucket,
27 | Region: task.Region,
28 | Key: task.Key,
29 | FilePath: task.FilePath,
30 | state: task.state,
31 | loaded: task.loaded,
32 | size: task.size,
33 | speed: task.speed,
34 | percent: task.percent,
35 | hashPercent: task.hashPercent,
36 | error: task.error,
37 | };
38 | if (task.FilePath) t.FilePath = task.FilePath;
39 | return t;
40 | };
41 |
42 | var emitListUpdate = (function () {
43 | var timer;
44 | var emit = function () {
45 | timer = 0;
46 | cos.emit('task-list-update', { list: util.map(queue, formatTask) });
47 | cos.emit('list-update', { list: util.map(queue, formatTask) });
48 | };
49 | return function () {
50 | if (!timer) timer = setTimeout(emit);
51 | };
52 | })();
53 |
54 | var clearQueue = function () {
55 | if (queue.length <= cos.options.UploadQueueSize) return;
56 | for (
57 | var i = 0;
58 | i < nextUploadIndex && // 小于当前操作的 index 才清理
59 | i < queue.length && // 大于队列才清理
60 | queue.length > cos.options.UploadQueueSize; // 如果还太多,才继续清理
61 |
62 | ) {
63 | var isActive = queue[i].state === 'waiting' || queue[i].state === 'checking' || queue[i].state === 'uploading';
64 | if (!queue[i] || !isActive) {
65 | tasks[queue[i].id] && delete tasks[queue[i].id];
66 | queue.splice(i, 1);
67 | nextUploadIndex--;
68 | } else {
69 | i++;
70 | }
71 | }
72 | emitListUpdate();
73 | };
74 |
75 | var startNextTask = function () {
76 | // 检查是否允许增加执行进程
77 | if (uploadingFileCount >= cos.options.FileParallelLimit) return;
78 | // 跳过不可执行的任务
79 | while (queue[nextUploadIndex] && queue[nextUploadIndex].state !== 'waiting') nextUploadIndex++;
80 | // 检查是否已遍历结束
81 | if (nextUploadIndex >= queue.length) return;
82 | // 上传该遍历到的任务
83 | var task = queue[nextUploadIndex];
84 | nextUploadIndex++;
85 | uploadingFileCount++;
86 | task.state = 'checking';
87 | task.params.onTaskStart && task.params.onTaskStart(formatTask(task));
88 | !task.params.UploadData && (task.params.UploadData = {});
89 | var apiParams = util.formatParams(task.api, task.params);
90 | originApiMap[task.api].call(cos, apiParams, function (err, data) {
91 | if (!cos._isRunningTask(task.id)) return;
92 | if (task.state === 'checking' || task.state === 'uploading') {
93 | task.state = err ? 'error' : 'success';
94 | err && (task.error = err);
95 | uploadingFileCount--;
96 | emitListUpdate();
97 | startNextTask();
98 | task.callback && task.callback(err, data);
99 | if (task.state === 'success') {
100 | if (task.params) {
101 | delete task.params.UploadData;
102 | delete task.params.Body;
103 | delete task.params;
104 | }
105 | delete task.callback;
106 | }
107 | }
108 | clearQueue();
109 | });
110 | emitListUpdate();
111 | // 异步执行下一个任务
112 | setTimeout(startNextTask);
113 | };
114 |
115 | var killTask = function (id, switchToState) {
116 | var task = tasks[id];
117 | if (!task) return;
118 | var waiting = task && task.state === 'waiting';
119 | var running = task && (task.state === 'checking' || task.state === 'uploading');
120 | if (
121 | (switchToState === 'canceled' && task.state !== 'canceled') ||
122 | (switchToState === 'paused' && waiting) ||
123 | (switchToState === 'paused' && running)
124 | ) {
125 | if (switchToState === 'paused' && task.params.Body && typeof task.params.Body.pipe === 'function') {
126 | console.error('stream not support pause');
127 | return;
128 | }
129 | task.state = switchToState;
130 | cos.emit('inner-kill-task', { TaskId: id, toState: switchToState });
131 | try {
132 | var UploadId = task && task.params && task.params.UploadData.UploadId;
133 | } catch (e) {}
134 | if (switchToState === 'canceled' && UploadId) session.removeUsing(UploadId);
135 | emitListUpdate();
136 | if (running) {
137 | uploadingFileCount--;
138 | startNextTask();
139 | }
140 | if (switchToState === 'canceled') {
141 | if (task.params) {
142 | delete task.params.UploadData;
143 | delete task.params.Body;
144 | delete task.params;
145 | }
146 | delete task.callback;
147 | }
148 | }
149 | clearQueue();
150 | };
151 |
152 | cos._addTasks = function (taskList) {
153 | util.each(taskList, function (task) {
154 | cos._addTask(task.api, task.params, task.callback, true);
155 | });
156 | emitListUpdate();
157 | };
158 |
159 | cos._addTask = function (api, params, callback, ignoreAddEvent) {
160 | // 如果小程序版本不支持获取文件分片内容,统一转到 简单上传 接口上传
161 | var simpleUploadMethod = cos.options.SimpleUploadMethod === 'postObject' ? 'postObject' : 'putObject';
162 | if (api === 'sliceUploadFile' && !util.canFileSlice()) api = simpleUploadMethod;
163 |
164 | // 复制参数对象
165 | params = util.formatParams(api, params);
166 |
167 | // 生成 id
168 | var id = util.uuid();
169 | params.TaskId = id;
170 | params.onTaskReady && params.onTaskReady(id);
171 |
172 | var task = {
173 | // env
174 | params: params,
175 | callback: callback,
176 | api: api,
177 | index: queue.length,
178 | // task
179 | id: id,
180 | Bucket: params.Bucket,
181 | Region: params.Region,
182 | Key: params.Key,
183 | FilePath: params.FilePath || '',
184 | state: 'waiting',
185 | loaded: 0,
186 | size: 0,
187 | speed: 0,
188 | percent: 0,
189 | hashPercent: 0,
190 | error: null,
191 | };
192 | var onHashProgress = params.onHashProgress;
193 | params.onHashProgress = function (info) {
194 | if (!cos._isRunningTask(task.id)) return;
195 | task.hashPercent = info.percent;
196 | onHashProgress && onHashProgress(info);
197 | emitListUpdate();
198 | };
199 | var onProgress = params.onProgress;
200 | params.onProgress = function (info) {
201 | if (!cos._isRunningTask(task.id)) return;
202 | task.state === 'checking' && (task.state = 'uploading');
203 | task.loaded = info.loaded;
204 | task.size = info.total;
205 | task.speed = info.speed;
206 | task.percent = info.percent;
207 | onProgress && onProgress(info);
208 | emitListUpdate();
209 | };
210 |
211 | // 异步获取 filesize
212 | util.getFileSize(api, params, function (err, size) {
213 | // 开始处理上传
214 | if (err) {
215 | // 如果获取大小出错,不加入队列
216 | callback(err);
217 | return;
218 | }
219 | // 获取完文件大小再把任务加入队列
220 | tasks[id] = task;
221 | queue.push(task);
222 | task.size = size;
223 | !ignoreAddEvent && emitListUpdate();
224 | startNextTask();
225 | clearQueue();
226 | });
227 | return id;
228 | };
229 | cos._isRunningTask = function (id) {
230 | var task = tasks[id];
231 | return !!(task && (task.state === 'checking' || task.state === 'uploading'));
232 | };
233 | cos.getTaskList = function () {
234 | return util.map(queue, formatTask);
235 | };
236 | cos.cancelTask = function (id) {
237 | killTask(id, 'canceled');
238 | };
239 | cos.pauseTask = function (id) {
240 | killTask(id, 'paused');
241 | };
242 | cos.restartTask = function (id) {
243 | var task = tasks[id];
244 | if (task && (task.state === 'paused' || task.state === 'error')) {
245 | task.state = 'waiting';
246 | emitListUpdate();
247 | nextUploadIndex = Math.min(nextUploadIndex, task.index);
248 | startNextTask();
249 | }
250 | };
251 | cos.isUploadRunning = function () {
252 | return uploadingFileCount || nextUploadIndex < queue.length;
253 | };
254 | };
255 |
256 | module.exports.transferToTaskMethod = transferToTaskMethod;
257 | module.exports.init = initTask;
258 |
--------------------------------------------------------------------------------
/demo-album/pages/album/album.js:
--------------------------------------------------------------------------------
1 | const { listToMatrix } = require('../../lib/util.js');
2 | var COS = require('../../lib/cos-wx-sdk-v5.js');
3 | const config = require('../../config.js');
4 | const util = require('../../lib/util.js');
5 |
6 |
7 | var cos = new COS({
8 | getAuthorization: function (options, callback) {
9 | wx.request({
10 | method: 'GET',
11 | url: config.stsUrl, // 服务端签名,参考 server 目录下的两个签名例子
12 | dataType: 'json',
13 | success: function (result) {
14 | var data = result.data;
15 | callback({
16 | TmpSecretId: data.credentials && data.credentials.tmpSecretId,
17 | TmpSecretKey: data.credentials && data.credentials.tmpSecretKey,
18 | XCosSecurityToken: data.credentials && data.credentials.sessionToken,
19 | ExpiredTime: data.expiredTime,
20 | });
21 | }
22 | });
23 | },
24 | });
25 |
26 | Page({
27 | data: {
28 | // 相册列表数据
29 | albumList: [],
30 |
31 | // 图片布局列表(二维数组,由`albumList`计算而得)
32 | layoutList: [],
33 |
34 | // 布局列数
35 | layoutColumnSize: 3,
36 |
37 | // 是否显示loading
38 | showLoading: false,
39 |
40 | // loading提示语
41 | loadingMessage: '',
42 |
43 | // 是否显示toast
44 | showToast: false,
45 |
46 | // 提示消息
47 | toastMessage: '',
48 |
49 | // 是否显示动作命令
50 | showActionsSheet: false,
51 |
52 | // 当前操作的图片
53 | imageInAction: '',
54 |
55 | // 图片预览模式
56 | previewMode: false,
57 |
58 | // 当前预览索引
59 | previewIndex: 0,
60 |
61 | // 切换动画的时间
62 | slideDuration: 400,
63 | },
64 | onShareAppMessage: function (res) {
65 | return {
66 | title: 'COS 上传示例 - 共享相册',
67 | path: this.route,
68 | }
69 | },
70 |
71 | // 显示loading提示
72 | showLoading(loadingMessage) {
73 | this.setData({ showLoading: true, loadingMessage });
74 | },
75 |
76 | // 隐藏loading提示
77 | hideLoading() {
78 | this.setData({ showLoading: false, loadingMessage: '' });
79 | },
80 |
81 | // 显示toast消息
82 | showToast(toastMessage) {
83 | this.setData({ showToast: true, toastMessage });
84 | },
85 |
86 | // 隐藏toast消息
87 | hideToast() {
88 | this.setData({ showToast: false, toastMessage: '' });
89 | },
90 |
91 | // 隐藏动作列表
92 | hideActionSheet() {
93 | this.setData({ showActionsSheet: false, imageInAction: '' });
94 | },
95 |
96 | onLoad() {
97 | var self = this;
98 | this.renderAlbumList();
99 | this.getAlbumList(function (list) {
100 | list = self.data.albumList.concat(list || {});
101 | if (!list.length) {
102 | list = [];
103 | }
104 | list = list.reverse();
105 | self.setData({ 'albumList': list });
106 | self.renderAlbumList();
107 | });
108 | },
109 |
110 | // 获取相册列表
111 | getAlbumList(callback) {
112 | this.showLoading('加载列表中…');
113 | setTimeout(() => this.hideLoading(), 1000);
114 | cos.getBucket({
115 | Bucket: config.Bucket,
116 | Region: config.Region,
117 | Prefix: config.albumDir
118 | }, function (err, data) {
119 | if (data) {
120 | var list = (data && data.Contents || [])
121 | .map((v) => 'https://' + config.Bucket + '.cos.' + config.Region + '.myqcloud.com/' + util.camSafeUrlEncode(v.Key).replace(/%2F/g, '/'))
122 | .filter((v) => /\.(jpg|png|gif)$/.test(v));
123 | callback(list);
124 | } else {
125 | callback([]);
126 | }
127 | });
128 | },
129 |
130 | // 渲染相册列表
131 | renderAlbumList() {
132 | let layoutColumnSize = this.data.layoutColumnSize;
133 | let layoutList = [];
134 | var imageList = [].concat(this.data.albumList);
135 | imageList.unshift({type:'add'});
136 | layoutList = listToMatrix(imageList, layoutColumnSize);
137 | this.setData({ layoutList });
138 | },
139 |
140 | // 从相册选择照片或拍摄照片
141 | chooseImage() {
142 | var self = this;
143 | wx.chooseImage({
144 | count: 9,
145 | sizeType: ['original', 'compressed'],
146 | sourceType: ['album', 'camera'],
147 | success: (res) => {
148 | this.showLoading('正在上传图片…');
149 | res.tempFilePaths.forEach(function (filePath) {
150 | var Key = config.albumDir + util.getRandFileName(filePath);
151 | filePath && cos.postObject({
152 | Bucket: config.Bucket,
153 | Region: config.Region,
154 | Key: Key,
155 | FilePath: filePath
156 | }, function (err, data) {
157 | if (data) {
158 | let albumList = self.data.albumList;
159 | albumList.unshift('https://' + data.Location);
160 | self.setData({ albumList });
161 | self.renderAlbumList();
162 | }
163 | self.hideLoading();
164 | });
165 | });
166 | },
167 | });
168 | },
169 |
170 | // 进入预览模式
171 | enterPreviewMode(event) {
172 | var self = this;
173 | if (this.data.showActionsSheet) {
174 | return;
175 | }
176 | let imageUrl = event.target.dataset.src;
177 | let previewIndex = this.data.albumList.indexOf(imageUrl);
178 |
179 | this.setData({ slideDuration: 0 });
180 | setTimeout(function () {
181 | self.setData({ previewMode: true, previewIndex: previewIndex});
182 | setTimeout(function () {
183 | self.setData({ slideDuration: 400 });
184 | }, 400);
185 | });
186 | },
187 |
188 | // 退出预览模式
189 | leavePreviewMode() {
190 | this.setData({ previewMode: false, previewIndex: 0 });
191 | },
192 |
193 | // 显示可操作命令
194 | showActions(event) {
195 | this.setData({ showActionsSheet: true, imageInAction: event.target.dataset.src });
196 | },
197 |
198 | // 下载图片
199 | downloadImage() {
200 | this.showLoading('正在保存图片…');
201 | console.log('download_image_url', this.data.imageInAction);
202 |
203 | wx.downloadFile({
204 | url: this.data.imageInAction,
205 | type: 'image',
206 | success: (resp) => {
207 | wx.saveFile({
208 | tempFilePath: resp.tempFilePath,
209 | success: (resp) => {
210 | this.showToast('图片保存成功');
211 | },
212 | fail: (resp) => {
213 | console.log('fail', resp);
214 | },
215 | complete: (resp) => {
216 | console.log('complete', resp);
217 | this.hideLoading();
218 | },
219 | });
220 | },
221 |
222 | fail: (resp) => {
223 | console.log('fail', resp);
224 | },
225 | });
226 |
227 | this.setData({ showActionsSheet: false, imageInAction: '' });
228 | },
229 |
230 | // 删除图片
231 | deleteImage() {
232 | let imageUrl = this.data.imageInAction;
233 | var m = imageUrl.match(/^https:\/\/[^\/]+\/([^#?]+)/);
234 | var Key = m && m[1] || '';
235 | if (Key) {
236 | this.showLoading('正在删除图片…');
237 | this.setData({ showActionsSheet: false, imageInAction: '' });
238 | cos.deleteObject({
239 | Bucket: config.Bucket,
240 | Region: config.Region,
241 | Key: Key,
242 | }, (err, data)=>{
243 | if (data) {
244 | let index = this.data.albumList.indexOf(imageUrl);
245 | if (~index) {
246 | let albumList = this.data.albumList;
247 | albumList.splice(index, 1);
248 | this.setData({ albumList });
249 | this.renderAlbumList();
250 | }
251 | this.showToast('图片删除成功');
252 | } else {
253 | this.showToast('图片删除失败');
254 | }
255 | this.hideLoading();
256 | });
257 | }
258 | },
259 | });
260 |
--------------------------------------------------------------------------------
/src/tracker.js:
--------------------------------------------------------------------------------
1 | const pkg = require('../package.json');
2 | let beacon = null;
3 |
4 | const getBeacon = (Beacon, delay) => {
5 | if (!beacon) {
6 | // 生成 beacon
7 | if (typeof Beacon !== 'function') {
8 | throw new Error('Beacon not found');
9 | }
10 | beacon = new Beacon({
11 | appkey: '0WEB05PY6MHRGK0U',
12 | versionCode: pkg.version,
13 | channelID: 'mp_sdk', //渠道,选填
14 | openid: 'openid', // 用户id, 选填
15 | unionid: 'unid', //用户unionid , 类似idfv,选填
16 | strictMode: false, //严苛模式开关, 打开严苛模式会主动抛出异常, 上线请务必关闭!!!
17 | delay, // 普通事件延迟上报时间(单位毫秒), 默认1000(1秒),选填
18 | sessionDuration: 60 * 1000, // session变更的时间间隔, 一个用户持续30分钟(默认值)没有任何上报则算另一次 session,每变更一次session上报一次启动事件(rqd_applaunched),使用毫秒(ms),最小值30秒,选填
19 | });
20 | }
21 | return beacon;
22 | };
23 |
24 | // 毫秒转秒
25 | const ms2s = function (ms) {
26 | if (!ms || ms < 0) return 0;
27 | return (ms / 1000).toFixed(3);
28 | };
29 |
30 | const utils = {
31 | // 生成uid 每个链路对应唯一一条uid
32 | getUid() {
33 | var S4 = function () {
34 | return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
35 | };
36 | return S4() + S4() + '-' + S4() + '-' + S4() + '-' + S4() + '-' + S4() + S4() + S4();
37 | },
38 | // 获取网络类型 4g | wifi
39 | getNetType() {
40 | return new Promise((resolve) => {
41 | if (wx.canIUse('getNetworkType')) {
42 | try {
43 | wx.getNetworkType({
44 | success(res) {
45 | resolve(res.networkType);
46 | },
47 | });
48 | } catch (e) {
49 | resolve('can_not_get_network_type');
50 | }
51 | } else {
52 | resolve('can_not_get_network_type');
53 | }
54 | });
55 | },
56 | // 获取系统信息
57 | getSystemInfo() {
58 | const baseInfo = {
59 | // ↓上报项
60 | devicePlatform: 'can_not_get_system_info', // ios/anroid/windows/mac/devtools
61 | wxVersion: 'can_not_get_system_info',
62 | wxSystem: 'can_not_get_system_info',
63 | wxSdkVersion: 'can_not_get_system_info',
64 | };
65 | let appBaseInfo = {};
66 | let deviceInfo = {};
67 | if (wx.canIUse('getAppBaseInfo')) {
68 | appBaseInfo = wx.getAppBaseInfo() || {};
69 | }
70 | if (wx.canIUse('getDeviceInfo')) {
71 | deviceInfo = wx.getDeviceInfo() || {};
72 | }
73 | const sdkVersion = appBaseInfo.SDKVersion || 'can_not_get_system_info';
74 | const version = appBaseInfo.version || 'can_not_get_system_info';
75 | const platform = deviceInfo.platform || 'can_not_get_system_info';
76 | const system = deviceInfo.system || 'can_not_get_system_info';
77 |
78 | Object.assign(baseInfo, {
79 | devicePlatform: platform,
80 | wxVersion: version,
81 | wxSystem: system,
82 | wxSdkVersion: sdkVersion,
83 | });
84 | return baseInfo;
85 | },
86 | };
87 |
88 | // 设备信息,只取一次值
89 | const deviceInfo = utils.getSystemInfo();
90 |
91 | const transApiName = (api) => {
92 | if (['putObject', 'sliceUploadFile', 'uploadFile', 'uploadFiles'].includes(api)) {
93 | return 'UploadTask';
94 | } else if (api === 'getObject') {
95 | return 'DownloadTask';
96 | } else if (['putObjectCopy', 'sliceCopyFile'].includes(api)) {
97 | return 'CopyTask';
98 | }
99 | return api;
100 | };
101 |
102 | // 上报参数驼峰改下划线
103 | function camel2underline(key) {
104 | return key.replace(/([A-Z])/g, '_$1').toLowerCase();
105 | }
106 | function formatParams(params) {
107 | const formattedParams = {};
108 | const successKeys = [
109 | 'sdkVersionName',
110 | 'sdkVersionCode',
111 | 'osName',
112 | 'networkType',
113 | 'requestName',
114 | 'requestResult',
115 | 'bucket',
116 | 'region',
117 | 'appid',
118 | 'accelerate',
119 | 'url',
120 | 'host',
121 | 'requestPath',
122 | 'userAgent',
123 | 'httpMethod',
124 | 'httpSize',
125 | 'httpSpeed',
126 | 'httpTookTime',
127 | 'httpMd5',
128 | 'httpSign',
129 | 'httpFullTime',
130 | 'httpDomain',
131 | 'partNumber',
132 | 'httpRetryTimes',
133 | 'customId',
134 | 'traceId',
135 | 'realApi',
136 | ];
137 | const failureKeys = [
138 | ...successKeys,
139 | 'errorNode',
140 | 'errorCode',
141 | 'errorName',
142 | 'errorMessage',
143 | 'errorRequestId',
144 | 'errorHttpCode',
145 | 'errorServiceName',
146 | 'errorType',
147 | 'fullError',
148 | ];
149 | // 需要上报的参数字段
150 | const reporterKeys = params.requestResult === 'Success' ? successKeys : failureKeys;
151 | for (let key in params) {
152 | if (!reporterKeys.includes(key)) continue;
153 | const formattedKey = camel2underline(key);
154 | formattedParams[formattedKey] = params[key];
155 | }
156 | formattedParams['request_name'] = params.realApi ? transApiName(params.realApi) : params.requestName;
157 | return formattedParams;
158 | }
159 |
160 | // 链路追踪器
161 | class Tracker {
162 | constructor(opt) {
163 | const {
164 | parent,
165 | traceId,
166 | bucket,
167 | region,
168 | apiName,
169 | realApi,
170 | httpMethod,
171 | fileKey,
172 | fileSize,
173 | accelerate,
174 | customId,
175 | delay,
176 | deepTracker,
177 | Beacon,
178 | clsReporter,
179 | } = opt;
180 | const appid = (bucket && bucket.substr(bucket.lastIndexOf('-') + 1)) || '';
181 | this.parent = parent;
182 | this.deepTracker = deepTracker;
183 | this.delay = delay;
184 | if (clsReporter && !this.clsReporter) {
185 | this.clsReporter = clsReporter;
186 | }
187 | // 上报用到的字段
188 | this.params = {
189 | // 通用字段
190 | sdkVersionName: 'cos-wx-sdk-v5',
191 | sdkVersionCode: pkg.version,
192 | osName: deviceInfo.devicePlatform,
193 | networkType: '',
194 |
195 | requestName: apiName || '',
196 | requestResult: '', // sdk api调用结果Success、Failure
197 | realApi,
198 |
199 | bucket,
200 | region,
201 | accelerate,
202 | httpMethod,
203 | url: '',
204 | host: '',
205 | httpDomain: '',
206 | requestPath: fileKey || '',
207 |
208 | errorType: '',
209 | errorCode: '',
210 | errorName: '',
211 | errorMessage: '',
212 | errorRequestId: '',
213 | errorHttpCode: 0,
214 | errorServiceName: '',
215 | errorNode: '',
216 |
217 | httpTookTime: 0, // http整体耗时
218 | httpSize: fileSize || 0, // 主要是文件大小,大小 B
219 | httpMd5: 0, // MD5耗时
220 | httpSign: 0, // 计算签名耗时
221 | httpFullTime: 0, // 任务整体耗时(包括md5、签名等)
222 | httpSpeed: 0, // 主要关注上传速度,KB/s
223 |
224 | size: fileSize || 0,
225 | httpMd5: 0, // MD5耗时
226 | httpSign: 0, // 计算签名耗时
227 | httpFull: 0, // http请求耗时
228 | name: apiName || '',
229 | tookTime: 0, // 总耗时
230 | md5StartTime: 0, // md5计算开始时间
231 | md5EndTime: 0, // md5计算结束时间
232 | signStartTime: 0, // 计算签名开始时间
233 | signEndTime: 0, // 计算签名结束时间
234 | httpStartTime: 0, // 发起网络请求开始时间
235 | httpEndTime: 0, // 网路请求结束时间
236 | startTime: new Date().getTime(), // sdk api调用起始时间,不是纯网络耗时
237 | endTime: 0, // sdk api调用结束时间,不是纯网络耗时
238 |
239 | // 小程序补充字段
240 | traceId: traceId || utils.getUid(), // 每条上报唯一标识
241 | appid,
242 | partNumber: 0, // 分块上传编号
243 | httpRetryTimes: 0, // sdk内部发起的请求重试
244 | customId: customId || '', // 业务id
245 | partTime: 0,
246 | };
247 | if (Beacon) {
248 | this.beacon = getBeacon(Beacon, delay);
249 | }
250 | }
251 |
252 | // 格式化sdk回调
253 | async formatResult(err, data) {
254 | /**
255 | * 解析到err的格式为:
256 | * 1.服务端有返回时
257 | * {
258 | * err: 同下方error,
259 | * error: {
260 | * error: {
261 | * Code: '', Message: '', Resource: '', RequestId: '', TraceId: '',
262 | * },
263 | * statusCode: xxx,
264 | * headers: {},
265 | * RequestId: '',
266 | * },
267 | * }
268 | * 2.本地抛出或小程序直接报错
269 | * {error: 'message'}或{error: {error: 'message' }}
270 | */
271 | const now = new Date().getTime();
272 | const networkType = await utils.getNetType();
273 | const errorCode = err ? err?.error?.error?.Code || 'Error' : '';
274 | const errorMessage = err ? err?.error?.error?.Message || err?.error?.error || err?.error || '' : '';
275 | const errorName = errorMessage;
276 | const errorHttpCode = err ? err?.error?.statusCode : data.statusCode;
277 | const errorServiceName = err ? err?.error?.error?.Resource : '';
278 | const requestId = err ? err?.error?.RequestId || '' : data?.RequestId || '';
279 | const errorType = err ? (requestId ? 'Server' : 'Client') : '';
280 |
281 | if (this.params.requestName === 'getObject') {
282 | this.params.httpSize = data ? data.headers && data.headers['content-length'] : 0;
283 | }
284 |
285 | // 上报 sliceUploadFile || uploadFile || uploadFiles 命中分块上传时
286 | const isSliceUploadFile = this.params.realApi === 'sliceUploadFile';
287 | const isSliceCopyFile = this.params.realApi === 'sliceCopyFile';
288 |
289 | if (isSliceUploadFile || isSliceCopyFile) {
290 | const speed = this.params.httpSize / 1024 / this.params.partTime;
291 | Object.assign(this.params, { httpSpeed: speed < 0 ? 0 : speed.toFixed(3) });
292 | } else {
293 | const httpFullTime = now - this.params.startTime;
294 | const httpTookTime = this.params.httpEndTime - this.params.httpStartTime;
295 | const speed = this.params.httpSize / 1024 / (httpTookTime / 1000);
296 | const httpMd5 = this.params.md5EndTime - this.params.md5StartTime;
297 | const httpSign = this.params.signEndTime - this.params.signStartTime;
298 |
299 | if (this.parent) {
300 | this.parent.addParamValue('httpTookTime', ms2s(httpTookTime));
301 | this.parent.addParamValue('httpFullTime', ms2s(httpFullTime));
302 | this.parent.addParamValue('httpMd5', ms2s(httpMd5));
303 | this.parent.addParamValue('httpSign', ms2s(httpSign));
304 | if (['multipartUpload', 'uploadPartCopy', 'putObjectCopy'].includes(this.params.requestName)) {
305 | // 只有小分块上传|复制才累计纯请求耗时,计算速度时用到
306 | this.parent.addParamValue('partTime', ms2s(httpTookTime));
307 | }
308 | }
309 | Object.assign(this.params, {
310 | httpFullTime: ms2s(httpFullTime),
311 | httpMd5: ms2s(httpMd5),
312 | httpSign: ms2s(httpSign),
313 | httpTookTime: ms2s(httpTookTime),
314 | httpSpeed: speed < 0 ? 0 : speed.toFixed(3),
315 | });
316 | }
317 |
318 | Object.assign(this.params, {
319 | networkType,
320 | requestResult: err ? 'Failure' : 'Success',
321 | errorType,
322 | errorCode,
323 | errorHttpCode,
324 | errorName,
325 | errorMessage,
326 | errorServiceName,
327 | errorRequestId: requestId,
328 | });
329 | if (err && (!errorCode || !errorMessage)) {
330 | // 暂存全量err一段时间 观察是否所有err格式都可被解析
331 | this.params.fullError = err ? JSON.stringify(err) : '';
332 | }
333 | if (this.params.name === 'getObject') {
334 | this.params.size = data ? data.headers && data.headers['content-length'] : -1;
335 | }
336 | if (this.params.url) {
337 | try {
338 | const exec = /^http(s)?:\/\/(.*?)\//.exec(this.params.url);
339 | this.params.host = exec[2];
340 | } catch (e) {
341 | this.params.host = this.params.url;
342 | }
343 | this.params.httpDomain = this.params.host;
344 | }
345 | }
346 |
347 | // 上报
348 | async report(err, data) {
349 | if (!this.beacon && !this.clsReporter) return;
350 | await this.formatResult(err, data);
351 | const formattedParams = formatParams(this.params);
352 | if (this.beacon) {
353 | this.sendEventsToBeacon(formattedParams);
354 | }
355 | if (this.clsReporter) {
356 | this.sendEventsToCLS(formattedParams);
357 | }
358 | }
359 |
360 | // 设置当前链路的参数
361 | setParams(params) {
362 | Object.assign(this.params, params);
363 | }
364 |
365 | addParamValue(key, value) {
366 | this.params[key] = (+this.params[key] + +value).toFixed(3);
367 | }
368 |
369 | // 上报灯塔
370 | sendEventsToBeacon(formattedParams) {
371 | // DeepTracker模式下才会上报分块上传内部细节
372 | const isSliceUploadFile =
373 | this.params.requestName === 'sliceUploadFile' || this.params.realApi === 'sliceUploadFile';
374 | if (isSliceUploadFile && !this.deepTracker) {
375 | return;
376 | }
377 | const eventCode = 'qcloud_track_cos_sdk';
378 |
379 | if (this.delay === 0) {
380 | // 实时上报
381 | this.beacon && this.beacon.onDirectUserAction(eventCode, formattedParams);
382 | } else {
383 | // 周期性上报
384 | this.beacon && this.beacon.onUserAction(eventCode, formattedParams);
385 | }
386 | }
387 |
388 | // 上报 cls
389 | sendEventsToCLS(formattedParams) {
390 | // 是否实时上报
391 | const immediate = !!(this.delay === 0);
392 |
393 | this.clsReporter.log(formattedParams, immediate);
394 | }
395 |
396 | // 生成子实例,与父所属一个链路,可用于分块上传内部流程上报单个分块操作
397 | generateSubTracker(subParams) {
398 | Object.assign(subParams, {
399 | parent: this,
400 | deepTracker: this.deepTracker,
401 | traceId: this.params.traceId,
402 | bucket: this.params.bucket,
403 | region: this.params.region,
404 | accelerate: this.params.accelerate,
405 | fileKey: this.params.requestPath,
406 | customId: this.params.customId,
407 | delay: this.params.delay,
408 | clsReporter: this.clsReporter,
409 | });
410 | return new Tracker(subParams);
411 | }
412 | }
413 |
414 | module.exports = Tracker;
415 |
--------------------------------------------------------------------------------
/demo/ciDemo/docPreview.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | var wxfs = wx.getFileSystemManager();
3 | const config = require('../config');
4 | const { cos, requestCallback } = require('../tools');
5 |
6 | const docPreviewDao = {
7 | '查询文档预览开通状态 describeDocProcessService': describeDocProcessService,
8 | '文档转html同步请求 getDocHtmlPreviewUrl': getDocHtmlPreviewUrl,
9 | '文档转码同步请求 getDocPreview': getDocPreview,
10 | '提交文档转码任务 createDocProcessJob': createDocProcessJob,
11 | '查询指定文档转码任务 describeDocProcessJob': describeDocProcessJob,
12 | '拉取符合条件的文档转码任务 describeDocProcessJobList': describeDocProcessJobList,
13 | '查询文档转码队列 describeDocProcessQueue': describeDocProcessQueue,
14 | '更新文档转码队列 updateProcessQueue': updateProcessQueue,
15 | };
16 |
17 | function describeDocProcessService() {
18 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
19 | const key = `docbucket`; //
20 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
21 | const url = `https://${host}/${key}`;
22 | cos.request(
23 | {
24 | Method: 'GET', // 固定值,必须
25 | Key: key, // 必须
26 | Url: url, // 请求的url,必须
27 | Query: {
28 | // 地域信息,以“,”分隔字符串,支持 All、ap-shanghai、ap-beijing;是否必传:否
29 | regions: '',
30 | // 存储桶名称,以“,”分隔,支持多个存储桶,精确搜索;是否必传:否
31 | bucketNames: '',
32 | // 存储桶名称前缀,前缀搜索;是否必传:否
33 | bucketName: '',
34 | // 第几页;是否必传:否
35 | pageNumber: '',
36 | // 每页个数;是否必传:否
37 | pageSize: '',
38 | },
39 | },
40 | function (err, data) {
41 | if (err) {
42 | // 处理请求失败
43 | console.log(err);
44 | } else {
45 | // 处理请求成功
46 | console.log(data.Response);
47 | }
48 | }
49 | );
50 | }
51 |
52 | function getDocHtmlPreviewUrl() {
53 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
54 | const ObjectKey = 'test.docx';
55 | const key = `${ObjectKey}`; // ObjectKey:{ObjectKey};
56 | const host = `${config.Bucket}.cos.${config.Region}.myqcloud.com`;
57 | const url = `https://${host}/${key}`;
58 |
59 | cos.request(
60 | {
61 | Method: 'GET', // 固定值,必须
62 | Key: key, // 必须
63 | Url: url, // 请求的url,必须
64 | Query: {
65 | // 数据万象处理能力,文档 HTML 预览固定为 doc-preview;是否必传:是
66 | 'ci-process': 'doc-preview',
67 | // 转换输出目标文件类型,文档 HTML 预览固定为 html(需为小写字母);是否必传:是
68 | dstType: 'html',
69 | // 是否获取预览链接。填入值为1会返回预览链接和Token信息;填入值为2只返回Token信息;不传会直接预览;是否必传:否
70 | weboffice_url: 1,
71 | // 指定目标文件类型,支持的文件类型请见下方;是否必传:否
72 | srcType: '',
73 | // 对象下载签名,如果预览的对象为私有读时,需要传入签名,详情请参见 请求签名 文档注意:需要进行 urlencode;是否必传:否
74 | sign: '',
75 | // 是否可复制。默认为可复制,填入值为1;不可复制,填入值为0;是否必传:否
76 | copyable: '',
77 | // 自定义配置参数,json结构,需要经过 URL 安全 的 Base64 编码,默认配置为:{ commonOptions: { isShowTopArea: true, isShowHeader: true, language: "zh" }},支持的配置参考 自定义配置项说明。htmlParams支持的特殊配置:语言切换,通过 commonOptions 的 language 参数指定预览语言,支持"zh"、"en“,默认为"zh"。;是否必传:否
78 | htmlParams: '',
79 | // 水印文字,需要经过 URL 安全 的 Base64 编码,默认为空;是否必传:否
80 | htmlwaterword: '',
81 | // 水印 RGBA(颜色和透明度),需要经过 URL 安全 的 Base64 编码,默认为:rgba(192,192,192,0.6);是否必传:否
82 | htmlfillstyle: '',
83 | // 水印文字样式,需要经过 URL 安全 的 Base64 编码,默认为:bold 20px Serif;是否必传:否
84 | htmlfront: '',
85 | // 水印文字旋转角度,0 - 360,默认315度;是否必传:否
86 | htmlrotate: '',
87 | // 水印文字水平间距,单位 px,默认为50;是否必传:否
88 | htmlhorizontal: '',
89 | // 水印文字垂直间距,单位 px,默认为100;是否必传:否
90 | htmlvertical: '',
91 | },
92 | },
93 | function (err, data) {
94 | if (err) {
95 | // 处理请求失败
96 | console.log(err);
97 | } else {
98 | // 处理请求成功
99 | console.log(data.Response);
100 | }
101 | }
102 | );
103 | }
104 |
105 | function getDocPreview() {
106 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
107 | const ObjectKey = 'test.docx';
108 | const key = `${ObjectKey}`; // ObjectKey:{ObjectKey};
109 | const host = `${config.Bucket}.cos.${config.Region}.myqcloud.com`;
110 | const url = `https://${host}/${key}`;
111 |
112 | cos.request(
113 | {
114 | Method: 'GET', // 固定值,必须
115 | Key: key, // 必须
116 | Url: url, // 请求的url,必须
117 | Query: {
118 | // 数据万象处理能力,文档预览固定为 doc-preview;是否必传:是
119 | 'ci-process': 'doc-preview',
120 | // 源数据的后缀类型,当前文档转换根据 COS 对象的后缀名来确定源数据类型。当 COS 对象没有后缀名时,可以设置该值;是否必传:否
121 | srcType: '',
122 | // 需转换的文档页码,默认从1开始计数;表格文件中 page 表示转换的第 X 个 sheet 的第 X 张图;是否必传:否
123 | page: 0,
124 | // 转换输出目标文件类型:png,转成 png 格式的图片文件jpg,转成 jpg 格式的图片文件pdf,转成 pdf 格式文件。 无法选择页码,page 参数不生效如果传入的格式未能识别,默认使用 jpg 格式;是否必传:否
125 | dstType: 'jpg',
126 | // Office 文档的打开密码,如果需要转换有密码的文档,请设置该字段;是否必传:否
127 | password: '',
128 | // 是否隐藏批注和应用修订,默认为00:隐藏批注,应用修订1:显示批注和修订;是否必传:否
129 | comment: 0,
130 | // 表格文件参数,转换第 X 个表,默认为1;是否必传:否
131 | sheet: 0,
132 | // 表格文件转换纸张方向,0代表垂直方向,非0代表水平方向,默认为0;是否必传:否
133 | excelPaperDirection: 0,
134 | // 设置纸张(画布)大小,对应信息为: 0 → A4 、 1 → A2 、 2 → A0 ,默认 A4 纸张 (需配合 excelRow 或 excelCol 一起使用);是否必传:否
135 | excelPaperSize: 0,
136 | // 转换后的图片处理参数,支持 基础图片处理 所有处理参数,多个处理参数可通过 管道操作符 分隔,从而实现在一次访问中按顺序对图片进行不同处理;是否必传:否
137 | ImageParams: '',
138 | // 生成预览图的图片质量,取值范围为 [1, 100],默认值100。 例如取值为100,代表生成图片质量为100%;是否必传:否
139 | quality: 0,
140 | // 预览图片的缩放参数,取值范围为 [10, 200], 默认值100。 例如取值为200,代表图片缩放比例为200% 即放大两倍;是否必传:否
141 | scale: 0,
142 | // 按指定 dpi 渲染图片,该参数与 scale 共同作用,取值范围 96-600 ,默认值为 96 。转码后的图片单边宽度需小于65500像素;是否必传:否
143 | imageDpi: 0,
144 | },
145 | },
146 | function (err, data) {
147 | if (err) {
148 | // 处理请求失败
149 | console.log(err);
150 | } else {
151 | // 处理请求成功
152 | console.log(data.Response);
153 | }
154 | }
155 | );
156 | }
157 |
158 | function createDocProcessJob() {
159 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
160 | const key = `doc_jobs`; //
161 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
162 | const url = `https://${host}/${key}`;
163 | const body = COS.util.json2xml({
164 | Request: {
165 | // 创建任务的 Tag,目前仅支持:DocProcess;是否必传:是
166 | Tag: 'DocProcess',
167 | // 待操作的文件对象;是否必传:是
168 | Input: {
169 | // 文件在 COS 上的文件路径,Bucket 由 Host 指定;是否必传:是
170 | Object: 'test.docx',
171 | },
172 | // 操作规则;是否必传:是
173 | Operation: {
174 | // 当 Tag 为 DocProcess 时有效,指定该任务的参数;是否必传:否
175 | DocProcess: {
176 | // 源数据的后缀类型,当前文档转换根据 cos 对象的后缀名来确定源数据类型,当 cos 对象没有后缀名时,可以设置该值;是否必传:否
177 | SrcType: '',
178 | // 转换输出目标文件类型:jpg,转成 jpg 格式的图片文件;如果传入的格式未能识别,默认使用 jpg 格式png,转成 png 格式的图片文件pdf,转成 pdf 格式文件(暂不支持指定页数);是否必传:否
179 | TgtType: '',
180 | // 从第 X 页开始转换;在表格文件中,一张表可能分割为多页转换,生成多张图片。StartPage 表示从指定 SheetId 的第 X 页开始转换。默认为1;是否必传:否
181 | StartPage: 1,
182 | // 转换至第 X 页;在表格文件中,一张表可能分割为多页转换,生成多张图片。EndPage 表示转换至指定 SheetId 的第 X 页。默认为-1,即转换全部页;是否必传:否
183 | EndPage: 2,
184 | // 表格文件参数,转换第 X 个表,默认为0;设置 SheetId 为0,即转换文档中全部表;是否必传:否
185 | SheetId: 0,
186 | // 表格文件转换纸张方向,0代表垂直方向,非0代表水平方向,默认为0;是否必传:否
187 | PaperDirection: 0,
188 | // 设置纸张(画布)大小,对应信息为: 0 → A4 、 1 → A2 、 2 → A0 ,默认 A4 纸张;是否必传:否
189 | PaperSize: 0,
190 | // 转换后的图片处理参数,支持 基础图片处理 所有处理参数,多个处理参数可通过 管道操作符 分隔,从而实现在一次访问中按顺序对图片进行不同处理;是否必传:否
191 | ImageParams: '',
192 | // 生成预览图的图片质量,取值范围 [1-100],默认值100。 例:值为100,代表生成图片质量为100%;是否必传:否
193 | Quality: 0,
194 | // 预览图片的缩放参数,取值范围[10-200], 默认值100。 例:值为200,代表图片缩放比例为200% 即放大两倍;是否必传:否
195 | Zoom: 100,
196 | // 按指定 dpi 渲染图片,该参数与 Zoom 共同作用,取值范围 96-600 ,默认值为 96 。转码后的图片单边宽度需小于65500像素;是否必传:否
197 | ImageDpi: 96,
198 | // 是否转换成单张长图,设置为 1 时,最多仅支持将 20 标准页面合成单张长图,超过可能会报错,分页范围可以通过 StartPage、EndPage 控制。默认值为 0 ,按页导出图片,TgtType="png"/"jpg" 时生效;是否必传:否
199 | PicPagination: 0,
200 | },
201 | // 结果输出地址;是否必传:是
202 | Output: {
203 | // 存储桶的地域;是否必传:是
204 | Region: config.Region,
205 | // 存储结果的存储桶;是否必传:是
206 | Bucket: config.Bucket,
207 | // 输出文件路径。非表格文件输出文件名需包含 ${Number} 或 ${Page} 参数。多个输出文件,${Number} 表示序号从1开始,${Page} 表示序号与预览页码一致。${Number} 表示多个输出文件,序号从1开始,例如输入 abc_${Number}.jpg,预览某文件5 - 6页,则输出文件名为 abc_1.jpg,abc_2.jpg${Page} 表示多个输出文件,序号与预览页码一致,例如输入 abc_${Page}.jpg,预览某文件5-6页,则输出文件名为 abc_5.jpg,abc_6.jpg表格文件输出路径需包含 ${SheetID} 占位符,输出文件名必须包含 ${Number} 参数。例如 /${SheetID}/abc_${Number}.jpg,先根据 excel 转换的表格数,生成对应数量的文件夹,再在对应的文件夹下,生成对应数量的图片文件;是否必传:是
208 | Object: '${Number}',
209 | },
210 | },
211 | // 任务所在的队列 ID,开通预览服务后自动生成,请使用 查询队列 获取或前往 万象控制台 在存储桶中查询 ;是否必传:否
212 | QueueId: '',
213 | },
214 | });
215 | cos.request(
216 | {
217 | Method: 'POST', // 固定值,必须
218 | Key: key, // 必须
219 | Url: url, // 请求的url,必须
220 | Body: body, // 请求体参数,必须
221 | ContentType: 'application/xml', // 固定值,必须
222 | },
223 | function (err, data) {
224 | if (err) {
225 | // 处理请求失败
226 | console.log(err);
227 | } else {
228 | // 处理请求成功
229 | console.log(data.Response);
230 | }
231 | }
232 | );
233 | }
234 |
235 | function describeDocProcessJob() {
236 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
237 | const jobId = 'xxx';
238 | const key = `doc_jobs/${jobId}`; // jobId:{jobId};
239 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
240 | const url = `https://${host}/${key}`;
241 |
242 | cos.request(
243 | {
244 | Method: 'GET', // 固定值,必须
245 | Key: key, // 必须
246 | Url: url, // 请求的url,必须
247 | },
248 | function (err, data) {
249 | if (err) {
250 | // 处理请求失败
251 | console.log(err);
252 | } else {
253 | // 处理请求成功
254 | console.log(data.Response);
255 | }
256 | }
257 | );
258 | }
259 |
260 | function describeDocProcessJobList() {
261 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
262 | const key = `doc_jobs`; //
263 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
264 | const url = `https://${host}/${key}`;
265 | cos.request(
266 | {
267 | Method: 'GET', // 固定值,必须
268 | Key: key, // 必须
269 | Url: url, // 请求的url,必须
270 | Query: {
271 | // 任务的 Tag:DocProcess;是否必传:是
272 | tag: 'DocProcess',
273 | // 拉取该队列 ID 下的任务,可在任务响应内容或控制台中获取;是否必传:否
274 | queueId: '',
275 | // Desc 或者 Asc。默认为 Desc;是否必传:否
276 | orderByTime: 'Desc',
277 | // 请求的上下文,用于翻页。上次返回的值;是否必传:否
278 | nextToken: '',
279 | // 拉取该状态的任务,以,分割,支持多状态:All、Submitted、Running、Success、Failed、Pause、Cancel。默认为 All;是否必传:否
280 | states: 'Success',
281 | // 拉取创建时间大于等于该时间的任务。格式为:%Y-%m-%dT%H:%m:%S%z;是否必传:否
282 | // startCreationTime: "",
283 | // 拉取创建时间小于等于该时间的任务。格式为:%Y-%m-%dT%H:%m:%S%z;是否必传:否
284 | // endCreationTime: "",
285 | },
286 | },
287 | function (err, data) {
288 | if (err) {
289 | // 处理请求失败
290 | console.log(err);
291 | } else {
292 | // 处理请求成功
293 | console.log(data.Response);
294 | }
295 | }
296 | );
297 | }
298 |
299 | function describeDocProcessQueue() {
300 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
301 | const key = `docqueue`; //
302 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
303 | const url = `https://${host}/${key}`;
304 | cos.request(
305 | {
306 | Method: 'GET', // 固定值,必须
307 | Key: key, // 必须
308 | Url: url, // 请求的url,必须
309 | Query: {
310 | // 队列 ID,以“,”符号分割字符串;是否必传:否
311 | queueIds: '',
312 | // 1. Active 表示队列内的作业会被文档预览服务调度执行2. Paused 表示队列暂停,作业不再会被文档预览服务调度执行,队列内的所有作业状态维持在暂停状态,已经处于执行中的任务将继续执行,不受影响;是否必传:否
313 | state: 'Active',
314 | // 第几页,默认第一页;是否必传:否
315 | pageNumber: '',
316 | // 每页个数,默认10个;是否必传:否
317 | pageSize: '',
318 | },
319 | },
320 | function (err, data) {
321 | if (err) {
322 | // 处理请求失败
323 | console.log(err);
324 | } else {
325 | // 处理请求成功
326 | console.log(data.Response);
327 | }
328 | }
329 | );
330 | }
331 |
332 | function updateProcessQueue() {
333 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
334 | const queueId = 'xxx';
335 | const key = `docqueue/${queueId}`;
336 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
337 | const url = `https://${host}/${key}`;
338 | const body = COS.util.json2xml({
339 | Request: {
340 | // 队列名称;是否必传:是
341 | Name: 'xxx',
342 | // 队列 ID;是否必传:是
343 | QueueID: queueId,
344 | // 队列状态;是否必传:是
345 | State: 'Active',
346 | // 通知渠道;是否必传:是
347 | NotifyConfig: {
348 | // 回调配置;是否必传:否
349 | Url: '',
350 | // 回调类型,普通回调:Url;是否必传:否
351 | Type: 'Url',
352 | // 回调事件,文档预览任务完成;是否必传:否
353 | Event: 'TaskFinish',
354 | // 回调开关,Off,On;是否必传:否
355 | State: 'Off',
356 | },
357 | },
358 | });
359 | cos.request(
360 | {
361 | Method: 'PUT', // 固定值,必须
362 | Key: key, // 必须
363 | Url: url, // 请求的url,必须
364 | Body: body, // 请求体参数,必须
365 | ContentType: 'application/xml', // 固定值,必须
366 | },
367 | function (err, data) {
368 | if (err) {
369 | // 处理请求失败
370 | console.log(err);
371 | } else {
372 | // 处理请求成功
373 | console.log(data.Response);
374 | }
375 | }
376 | );
377 | }
378 |
379 | module.exports = docPreviewDao;
380 |
--------------------------------------------------------------------------------
/demo/ciDemo/fileProcess.js:
--------------------------------------------------------------------------------
1 | const COS = require('../lib/cos-wx-sdk-v5');
2 | var wxfs = wx.getFileSystemManager();
3 | const config = require('../config');
4 | const { cos, requestCallback } = require('../tools');
5 |
6 | const fileProcessDao = {
7 | '开通文件处理服务 openFileProcessService': openFileProcessService,
8 | '查询文件处理服务 describeFileProcessService': describeFileProcessService,
9 | '关闭文件处理服务 closeFileProcessService': closeFileProcessService,
10 | '查询文件处理队列 describeFileProcessQueue': describeFileProcessQueue,
11 | '更新文件处理队列 updateFileProcessQueue': updateFileProcessQueue,
12 | '哈希值计算同步请求 fileHash': fileHash,
13 | '提交哈希值计算任务 createFileHashJob': createFileHashJob,
14 | '查询哈希值计算结果 describeFileHashJob': describeFileHashJob,
15 | '提交多文件打包压缩任务 createFileCompressJob': createFileCompressJob,
16 | '查询多文件打包压缩结果 describeFileCompressJob': describeFileCompressJob,
17 | '压缩包预览 zipPreview': zipPreview,
18 | '提交文件解压任务 createFileUnCompressJob': createFileUnCompressJob,
19 | '查询文件解压结果 describeFileUnCompressJob': describeFileUnCompressJob,
20 | };
21 |
22 | function openFileProcessService() {
23 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
24 | const key = `file_bucket`; //
25 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
26 | const url = `https://${host}/${key}`;
27 |
28 | cos.request(
29 | {
30 | Method: 'POST', // 固定值,必须
31 | Key: key, // 必须
32 | Url: url, // 请求的url,必须
33 | },
34 | function (err, data) {
35 | if (err) {
36 | // 处理请求失败
37 | console.log(err);
38 | } else {
39 | // 处理请求成功
40 | console.log(data.Response);
41 | }
42 | }
43 | );
44 | }
45 |
46 | function describeFileProcessService() {
47 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
48 | const key = `file_bucket`; //
49 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
50 | const url = `https://${host}/${key}`;
51 |
52 | cos.request(
53 | {
54 | Method: 'Get', // 固定值,必须
55 | Key: key, // 必须
56 | Url: url, // 请求的url,必须
57 | Query: {
58 | // regions: "", // 地域信息,例如 ap-shanghai、ap-beijing,若查询多个地域以“,”分隔字符串,非必须
59 | // bucketNames: "", // 存储桶名称,以“,”分隔,支持多个存储桶,精确搜索,非必须
60 | // bucketName: "", // 存储桶名称前缀,前缀搜索,非必须
61 | // pageNumber: "", // 第几页,非必须
62 | // pageSize: "", // 每页个数,大于0且小于等于100的整数,非必须
63 | },
64 | },
65 | function (err, data) {
66 | if (err) {
67 | // 处理请求失败
68 | console.log(err);
69 | } else {
70 | // 处理请求成功
71 | console.log(data.Response);
72 | }
73 | }
74 | );
75 | }
76 |
77 | function closeFileProcessService() {
78 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
79 | const key = `file_bucket`; //
80 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
81 | const url = `https://${host}/${key}`;
82 |
83 | cos.request(
84 | {
85 | Method: 'DELETE', // 固定值,必须
86 | Key: key, // 必须
87 | Url: url, // 请求的url,必须
88 | },
89 | function (err, data) {
90 | if (err) {
91 | // 处理请求失败
92 | console.log(err);
93 | } else {
94 | // 处理请求成功
95 | console.log(data.Response);
96 | }
97 | }
98 | );
99 | }
100 |
101 | function describeFileProcessQueue() {
102 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
103 | const key = `file_queue`; //
104 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
105 | const url = `https://${host}/${key}`;
106 |
107 | cos.request(
108 | {
109 | Method: 'GET', // 固定值,必须
110 | Key: key, // 必须
111 | Url: url, // 请求的url,必须
112 | Query: {
113 | // 队列 ID,以“,”符号分割字符串;是否必传:否
114 | queueIds: '',
115 | // Active 表示队列内的作业会被调度执行Paused 表示队列暂停,作业不再会被调度执行,队列内的所有作业状态维持在暂停状态,已经执行中的任务不受影响;是否必传:否
116 | state: 'Active',
117 | // 第几页,默认值1;是否必传:否
118 | pageNumber: '',
119 | // 每页个数,默认值10;是否必传:否
120 | pageSize: '',
121 | },
122 | },
123 | function (err, data) {
124 | if (err) {
125 | // 处理请求失败
126 | console.log(err);
127 | } else {
128 | // 处理请求成功
129 | console.log(data.Response);
130 | }
131 | }
132 | );
133 | }
134 |
135 | function updateFileProcessQueue() {
136 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
137 | const queueId = 'xxx';
138 | const key = `file_queue/${queueId}`; // queueId:{queueId};
139 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
140 | const url = `https://${host}/${key}`;
141 | const body = COS.util.json2xml({
142 | Request: {
143 | // 队列名称,仅支持中文、英文、数字、_、-和*,长度不超过 128;是否必传:是
144 | Name: 'xxx',
145 | // Active 表示队列内的作业会被调度执行Paused 表示队列暂停,作业不再会被调度执行,队列内的所有作业状态维持在暂停状态,已经执行中的任务不受影响;是否必传:是
146 | State: 'Active',
147 | // 回调配置;是否必传:是
148 | NotifyConfig: {
149 | // 回调开关OffOn;是否必传:否
150 | State: 'Off',
151 | // 回调事件TaskFinish:任务完成WorkflowFinish:工作流完成;是否必传:否
152 | Event: 'TaskFinish',
153 | // 回调格式XMLJSON;是否必传:否
154 | ResultFormat: '',
155 | // 回调类型UrlTDMQ;是否必传:否
156 | Type: 'Url',
157 | // 回调地址,不能为内网地址。;是否必传:否
158 | Url: '',
159 | // TDMQ 使用模式Topic:主题订阅Queue: 队列服务;是否必传:否
160 | MqMode: '',
161 | // TDMQ 所属园区,目前支持园区 sh(上海)、bj(北京)、gz(广州)、cd(成都)、hk(中国香港);是否必传:否
162 | MqRegion: '',
163 | // TDMQ 主题名称;是否必传:否
164 | MqName: '',
165 | },
166 | },
167 | });
168 | cos.request(
169 | {
170 | Method: 'PUT', // 固定值,必须
171 | Key: key, // 必须
172 | Url: url, // 请求的url,必须
173 | Body: body, // 请求体参数,必须
174 | ContentType: 'application/xml', // 固定值,必须
175 | },
176 | function (err, data) {
177 | if (err) {
178 | // 处理请求失败
179 | console.log(err);
180 | } else {
181 | // 处理请求成功
182 | console.log(data.Response);
183 | }
184 | }
185 | );
186 | }
187 |
188 | function fileHash() {
189 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
190 | const ObjectKey = 'test.docx';
191 | const key = `${ObjectKey}`; // ObjectKey:{ObjectKey};
192 | const host = `${config.Bucket}.cos.${config.Region}.myqcloud.com`;
193 | const url = `https://${host}/${key}`;
194 |
195 | cos.request(
196 | {
197 | Method: 'GET', // 固定值,必须
198 | Key: key, // 必须
199 | Url: url, // 请求的url,必须
200 | Query: {
201 | // 操作类型,哈希值计算固定为:filehash;是否必传:是
202 | 'ci-process': 'filehash',
203 | // 支持的哈希算法类型,有效值:md5、sha1、sha256;是否必传:是
204 | type: 'md5',
205 | // 是否将计算得到的哈希值,自动添加至文件的自定义header,格式为:x-cos-meta-md5/sha1/sha256;有效值: true、false,不填则默认为false;是否必传:否
206 | addtoheader: '',
207 | },
208 | },
209 | function (err, data) {
210 | if (err) {
211 | // 处理请求失败
212 | console.log(err);
213 | } else {
214 | // 处理请求成功
215 | console.log(data.Response);
216 | }
217 | }
218 | );
219 | }
220 |
221 | function createFileHashJob() {
222 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
223 | const key = `jobs`; //
224 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
225 | const url = `https://${host}/${key}`;
226 | const body = COS.util.json2xml({
227 | Request: {
228 | // 表示任务的类型,哈希值计算默认为:FileHashCode。;是否必传:是
229 | Tag: 'FileHashCode',
230 | // 包含待操作的文件信息。;是否必传:是
231 | Input: {
232 | // 文件名,取值为文件在当前存储桶中的完整名称。;是否必传:是
233 | Object: 'test.docx',
234 | },
235 | // 包含哈希值计算的处理规则。;是否必传:是
236 | Operation: {
237 | // 指定哈希值计算的处理规则。;是否必传:是
238 | FileHashCodeConfig: {
239 | // 哈希值的算法类型,支持:MD5、SHA1、SHA256;是否必传:是
240 | Type: 'MD5',
241 | // 是否将计算得到的哈希值添加至文件自定义header,有效值:true、false,默认值为 false。自定义header根据Type的值变化,例如Type值为MD5时,自定义header为 x-cos-meta-md5。;是否必传:否
242 | AddToHeader: '',
243 | },
244 | // 透传用户信息, 可打印的 ASCII 码,长度不超过1024。;是否必传:否
245 | UserData: '',
246 | },
247 | // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式。;是否必传:否
248 | CallBackFormat: '',
249 | // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型。;是否必传:否
250 | CallBackType: 'Url',
251 | // 任务回调的地址,优先级高于队列的回调地址。;是否必传:否
252 | CallBack: '',
253 | },
254 | });
255 | cos.request(
256 | {
257 | Method: 'POST', // 固定值,必须
258 | Key: key, // 必须
259 | Url: url, // 请求的url,必须
260 | Body: body, // 请求体参数,必须
261 | ContentType: 'application/xml', // 固定值,必须
262 | },
263 | function (err, data) {
264 | if (err) {
265 | // 处理请求失败
266 | console.log(err);
267 | } else {
268 | // 处理请求成功
269 | console.log(data.Response);
270 | }
271 | }
272 | );
273 | }
274 |
275 | function describeFileHashJob() {
276 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
277 | const jobId = 'xxx';
278 | const key = `file_jobs/${jobId}`; // jobId:{jobId};
279 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
280 | const url = `https://${host}/${key}`;
281 | cos.request(
282 | {
283 | Method: 'GET', // 固定值,必须
284 | Key: key, // 必须
285 | Url: url, // 请求的url,必须
286 | },
287 | function (err, data) {
288 | if (err) {
289 | // 处理请求失败
290 | console.log(err);
291 | } else {
292 | // 处理请求成功
293 | console.log(data.Response);
294 | }
295 | }
296 | );
297 | }
298 |
299 | function createFileCompressJob() {
300 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
301 | const key = `jobs`; //
302 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
303 | const url = `https://${host}/${key}`;
304 | const body = COS.util.json2xml({
305 | Request: {
306 | // 表示任务的类型,多文件打包压缩默认为:FileCompress。;是否必传:是
307 | Tag: 'FileCompress',
308 | // 包含文件打包压缩的处理规则。;是否必传:是
309 | Operation: {
310 | // 指定文件打包压缩的处理规则。;是否必传:是
311 | FileCompressConfig: {
312 | // 文件打包时,是否需要去除源文件已有的目录结构,有效值:0:不需要去除目录结构,打包后压缩包中的文件会保留原有的目录结构;1:需要,打包后压缩包内的文件会去除原有的目录结构,所有文件都在同一层级。例如:源文�� URL 为 https://domain/source/test.mp4,则源文件路径为 source/test.mp4,如果为 1,则 ZIP 包中该文件路径为 test.mp4;如果为0, ZIP 包中该文件路径为 source/test.mp4。;是否必传:是
313 | Flatten: '0',
314 | // 打包压缩的类型,有效值:zip、tar、tar.gz。;是否必传:是
315 | Format: 'zip',
316 | // 压缩类型,仅在Format为tar.gz或zip时有效。faster:压缩速度较快better:压缩质量较高,体积较小default:适中的压缩方式默认值为default;是否必传:否
317 | Type: '',
318 | // 压缩包密钥,传入时需先经过 base64 编码,编码后长度不能超过128。当 Format 为 zip 时生效。;是否必传:否
319 | CompressKey: '',
320 | // 支持将需要打包的文件整理成索引文件,后台将根据索引文件内提供的文件 url,打包为一个压缩包文件。索引文件需要保存在当前存储桶中,本字段需要提供索引文件的对象地址,例如:/test/index.csv。索引文件格式:仅支持 CSV 文件,一行一条 URL(仅支持本存储桶文件),如有多列字段,默认取第一列作为URL。最多不超过10000个文件, 总大小不超过50G,否则会导致任务失败。;是否必传:否
321 | UrlList: '',
322 | // 支持对存储桶中的某个前缀进行打包,如果需要对某个目录进行打包,需要加/,例如test目录打包,则值为:test/。最多不超过10000个文件,总大小不超过50G,否则会导致任务失败。;是否必传:否
323 | Prefix: 'example/',
324 | // 支持对存储桶中的多个文件进行打包,个数不能超过 1000,总大小不超过50G,否则会导致任务失败。;是否必传:否
325 | Key: '',
326 | },
327 | // 透传用户信息,可打印的 ASCII 码,长度不超过1024。;是否必传:否
328 | UserData: '',
329 | // 指定打包压缩后的文件保存的地址信息。;是否必传:是
330 | Output: {
331 | // 存储桶的地域。;是否必传:是
332 | Region: config.Region,
333 | // 保存压缩后文件的存储桶。;是否必传:是
334 | Bucket: config.Bucket,
335 | // 压缩后文件的文件名;是否必传:是
336 | Object: 'test.zip',
337 | },
338 | },
339 | // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式。;是否必传:否
340 | CallBackFormat: '',
341 | // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型。;是否必传:否
342 | CallBackType: 'Url',
343 | // 任务回调的地址,优先级高于队列的回调地址。;是否必传:否
344 | CallBack: '',
345 | },
346 | });
347 |
348 | cos.request(
349 | {
350 | Method: 'POST', // 固定值,必须
351 | Key: key, // 必须
352 | Url: url, // 请求的url,必须
353 | Body: body, // 请求体参数,必须
354 | ContentType: 'application/xml', // 固定值,必须
355 | },
356 | function (err, data) {
357 | if (err) {
358 | // 处理请求失败
359 | console.log(err);
360 | } else {
361 | // 处理请求成功
362 | console.log(data.Response);
363 | }
364 | }
365 | );
366 | }
367 |
368 | function describeFileCompressJob() {
369 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
370 | const jobId = 'xxx';
371 | const key = `file_jobs/${jobId}`; // jobId:{jobId};
372 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
373 | const url = `https://${host}/${key}`;
374 | cos.request(
375 | {
376 | Method: 'GET', // 固定值,必须
377 | Key: key, // 必须
378 | Url: url, // 请求的url,必须
379 | },
380 | function (err, data) {
381 | if (err) {
382 | // 处理请求失败
383 | console.log(err);
384 | } else {
385 | // 处理请求成功
386 | console.log(data.Response);
387 | }
388 | }
389 | );
390 | }
391 |
392 | function zipPreview() {
393 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
394 | const key = `test.zip`; //
395 | const host = `${config.Bucket}.cos.${config.Region}.myqcloud.com`;
396 | const url = `https://${host}/${key}`;
397 |
398 | cos.request(
399 | {
400 | Method: 'GET', // 固定值,必须
401 | Key: key, // 必须
402 | Url: url, // 请求的url,必须
403 | Query: {
404 | 'ci-process': 'zippreview', // 操作类型,压缩包预览计算固定为:zippreview,必须
405 | },
406 | },
407 | function (err, data) {
408 | if (err) {
409 | // 处理请求失败
410 | console.log(err);
411 | } else {
412 | // 处理请求成功
413 | console.log(data.Response);
414 | }
415 | }
416 | );
417 | }
418 |
419 | function createFileUnCompressJob() {
420 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
421 | const key = `jobs`; //
422 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
423 | const url = `https://${host}/${key}`;
424 | const body = COS.util.json2xml({
425 | Request: {
426 | // 表示任务的类型,文件解压默认为:FileUncompress。;是否必传:是
427 | Tag: 'FileUncompress',
428 | // 包含待操作的文件信息。;是否必传:是
429 | Input: {
430 | // 文件名,取值为文件在当前存储桶中的完整名称。;是否必传:是
431 | Object: 'test.zip',
432 | },
433 | // 包含文件解压的处理规则。;是否必传:是
434 | Operation: {
435 | // 指定文件解压的处理规则。;是否必传:是
436 | FileUncompressConfig: {
437 | // 指定解压后输出文件的前缀,不填则默认保存在存储桶根路径。;是否必传:否
438 | Prefix: 'output/',
439 | // 解压密钥,传入时需先经过 base64 编码。;是否必传:否
440 | UnCompressKey: '',
441 | // 指定解压后的文件路径是否需要替换前缀,有效值:0:不添加额外的前缀,解压缩将保存在Prefix指定的路径下(不会保留压缩包的名称,仅将压缩包内的文件保存至指定的路径)1:以压缩包本身的名称作为前缀,解压缩将保存在Prefix指定的路径下2:以压缩包完整路径作为前缀,此时如果不指定Prefix,就是解压到压缩包所在的当前路径(包含压缩包本身名称)默认值为0。;是否必传:否
442 | PrefixReplaced: '',
443 | },
444 | // 透传用户信息,可打印的 ASCII 码,长度不超过1024。;是否必传:否
445 | UserData: '',
446 | // 指定解压后的文件保存的存储桶信息。;是否必传:是
447 | Output: {
448 | // 存储桶的地域。;是否必传:是
449 | Region: config.Region,
450 | // 保存解压后文件的存储桶。;是否必传:是
451 | Bucket: config.Bucket,
452 | },
453 | },
454 | // 任务回调格式,JSON 或 XML,默认 XML,优先级高于队列的回调格式。;是否必传:否
455 | CallBackFormat: '',
456 | // 任务回调类型,Url 或 TDMQ,默认 Url,优先级高于队列的回调类型。;是否必传:否
457 | CallBackType: 'Url',
458 | // 任务回调的地址,优先级高于队列的回调地址。;是否必传:否
459 | CallBack: '',
460 | },
461 | });
462 | cos.request(
463 | {
464 | Method: 'POST', // 固定值,必须
465 | Key: key, // 必须
466 | Url: url, // 请求的url,必须
467 | Body: body, // 请求体参数,必须
468 | ContentType: 'application/xml', // 固定值,必须
469 | },
470 | function (err, data) {
471 | if (err) {
472 | // 处理请求失败
473 | console.log(err);
474 | } else {
475 | // 处理请求成功
476 | console.log(data.Response);
477 | }
478 | }
479 | );
480 | }
481 |
482 | function describeFileUnCompressJob() {
483 | // sdk引入以及初始化请参考:https://cloud.tencent.com/document/product/436/31953
484 | const jobId = 'xxx';
485 | const key = `file_jobs/${jobId}`; // jobId:{jobId};
486 | const host = `${config.Bucket}.ci.${config.Region}.myqcloud.com`;
487 | const url = `https://${host}/${key}`;
488 | cos.request(
489 | {
490 | Method: 'GET', // 固定值,必须
491 | Key: key, // 必须
492 | Url: url, // 请求的url,必须
493 | },
494 | function (err, data) {
495 | if (err) {
496 | // 处理请求失败
497 | console.log(err);
498 | } else {
499 | // 处理请求成功
500 | console.log(data.Response);
501 | }
502 | }
503 | );
504 | }
505 |
506 | module.exports = fileProcessDao;
507 |
--------------------------------------------------------------------------------
/demo/lib/beacon_mp.min.js:
--------------------------------------------------------------------------------
1 | /*! For license information please see beacon_mp.min.js.LICENSE.txt */
2 | !function(t,e){"object"==typeof exports&&"object"==typeof module?module.exports=e():"function"==typeof define&&define.amd?define([],e):"object"==typeof exports?exports.BeaconAction=e():t.BeaconAction=e()}(window,(function(){return function(t){var e={};function n(r){if(e[r])return e[r].exports;var o=e[r]={i:r,l:!1,exports:{}};return t[r].call(o.exports,o,o.exports,n),o.l=!0,o.exports}return n.m=t,n.c=e,n.d=function(t,e,r){n.o(t,e)||Object.defineProperty(t,e,{enumerable:!0,get:r})},n.r=function(t){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})},n.t=function(t,e){if(1&e&&(t=n(t)),8&e)return t;if(4&e&&"object"==typeof t&&t&&t.__esModule)return t;var r=Object.create(null);if(n.r(r),Object.defineProperty(r,"default",{enumerable:!0,value:t}),2&e&&"string"!=typeof t)for(var o in t)n.d(r,o,function(e){return t[e]}.bind(null,o));return r},n.n=function(t){var e=t&&t.__esModule?function(){return t.default}:function(){return t};return n.d(e,"a",e),e},n.o=function(t,e){return Object.prototype.hasOwnProperty.call(t,e)},n.p="",n(n.s=6)}([function(t,e,n){"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.BEACON_ATTA_REQUEST_URL=e.BEACON_ATTA_TOKEN=e.BEACON_ATTA_ID=e.BEACON_CONFIG_HTTPS_URL=e.BEACON_CONFIG_REQUEST_TIME=e.BEACON_CONFIG=e.BEACON_SENDING_IDS_KEY=e.BEACON_NORMAL_LOG_ID_KEY=e.BEACON_DRIECT_LOG_ID_KEY=e.BEACON_LASE_REPORT_TIME_KEY=e.BEACON_DEVICE_ID_KEY=e.BEACON_STORE_PREFIX=e.BEACON_LOG_ID_KEY=e.BEACON_IS_REALTIME_KEY=e.BEACON_DELAY_DEFAULT=e.BEACON_HTTPS_URL=e.BEACON_HTTP_URL=void 0,e.BEACON_HTTP_URL="http://oth.eve.mdt.qq.com:8080/analytics/v2_upload",e.BEACON_HTTPS_URL="https://otheve.beacon.qq.com/analytics/v2_upload",e.BEACON_DELAY_DEFAULT=3e3,e.BEACON_IS_REALTIME_KEY="A99",e.BEACON_LOG_ID_KEY="A100",e.BEACON_STORE_PREFIX="__BEACON_",e.BEACON_DEVICE_ID_KEY="__BEACON_deviceId",e.BEACON_LASE_REPORT_TIME_KEY="last_report_time",e.BEACON_DRIECT_LOG_ID_KEY="direct_log_id",e.BEACON_NORMAL_LOG_ID_KEY="normal_log_id",e.BEACON_SENDING_IDS_KEY="sending_event_ids",e.BEACON_CONFIG="beacon_config",e.BEACON_CONFIG_REQUEST_TIME="beacon_config_request_time",e.BEACON_CONFIG_HTTPS_URL="https://oth.str.beacon.qq.com/trpc.beacon.configserver.BeaconConfigService/QueryConfig",e.BEACON_ATTA_ID="00400014144",e.BEACON_ATTA_TOKEN="6478159937",e.BEACON_ATTA_REQUEST_URL="https://h.trace.qq.com/kv"},function(t,e,n){"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.getEventId=e.replaceSymbol=e.replace=e.assert=void 0;var r=n(0),o=n(9);function i(t){if("string"!=typeof t)return t;try{return t.replace(new RegExp("\\|","g"),"%7C").replace(new RegExp("\\&","g"),"%26").replace(new RegExp("\\=","g"),"%3D").replace(new RegExp("\\+","g"),"%2B")}catch(t){return""}}Object.defineProperty(e,"EventEmiter",{enumerable:!0,get:function(){return o.EventEmiter}}),e.assert=function(t,e){if(!t)throw e instanceof Error?e:new Error(e)},e.replace=function(t,e){for(var n={},r=0,o=Object.keys(t);r60*this.strategy.requestInterval*1e3},t.prototype.getUploadUrl=function(){return this.strategy.httpsUploadUrl+"?appkey="+this.appkey},t.prototype.isBlackEvent=function(t){return-1!=this.strategy.blacklist.indexOf(t)},t.prototype.isEventUpOnOff=function(){return this.strategy.isEventUpOnOff},t.prototype.isSampleEvent=function(t){return!!Object.prototype.hasOwnProperty.call(this.realSample,t)&&this.realSample[t] {
32 | console.log(err || data);
33 | }
34 | );
35 | }
36 |
37 | function uploadFile() {
38 | var uploadFile = function (file) {
39 | cos.uploadFile(
40 | {
41 | Bucket: config.Bucket,
42 | Region: config.Region,
43 | Key: file.name,
44 | FilePath: file.path,
45 | FileSize: file.size,
46 | SliceSize: 1024 * 1024 * 5, // 文件大于5mb自动使用分块上传
47 | onTaskReady: function (taskId) {
48 | TaskId = taskId;
49 | },
50 | onProgress: function (info) {
51 | var percent = parseInt(info.percent * 10000) / 100;
52 | var speed = parseInt((info.speed / 1024 / 1024) * 100) / 100;
53 | console.log('进度:' + percent + '%; 速度:' + speed + 'Mb/s;');
54 | },
55 | onFileFinish: function (err, data, options) {
56 | console.log(options.Key + '上传' + (err ? '失败' : '完成'));
57 | },
58 | },
59 | function (err, data) {
60 | console.log(err || data);
61 | }
62 | );
63 | };
64 | wx.chooseMedia({
65 | count: 1,
66 | mediaType: ['image', 'video'],
67 | sizeType: ['original'],
68 | sourceType: ['album', 'camera'],
69 | success: function (res) {
70 | const file = res.tempFiles[0];
71 | const filePath = file.tempFilePath;
72 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
73 | uploadFile({
74 | name: filename,
75 | path: filePath,
76 | size: file.size,
77 | });
78 | },
79 | });
80 | }
81 |
82 | function uploadFiles() {
83 | const uploadFiles = function (files) {
84 | const fileList = files.map(function (file) {
85 | const filePath = file.tempFilePath;
86 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
87 | return Object.assign({
88 | Bucket: config.Bucket,
89 | Region: config.Region,
90 | Key: filename,
91 | FilePath: filePath,
92 | });
93 | });
94 | cos.uploadFiles(
95 | {
96 | files: fileList,
97 | SliceSize: 1024 * 1024 * 5, // 文件大于5mb自动使用分块上传
98 | onProgress: function (info) {
99 | var percent = parseInt(info.percent * 10000) / 100;
100 | var speed = parseInt((info.speed / 1024 / 1024) * 100) / 100;
101 | console.log('进度:' + percent + '%; 速度:' + speed + 'Mb/s;');
102 | },
103 | onFileFinish: function (err, data, options) {
104 | console.log(options.Key + '上传' + (err ? '失败' : '完成'));
105 | },
106 | },
107 | function (err, data) {
108 | console.log(err || data);
109 | }
110 | );
111 | };
112 | wx.chooseMedia({
113 | count: 9,
114 | mediaType: ['image', 'video'],
115 | sizeType: ['original'],
116 | sourceType: ['album', 'camera'],
117 | success: function (res) {
118 | uploadFiles(res.tempFiles);
119 | },
120 | });
121 | }
122 |
123 | function sliceUploadFile() {
124 | var sliceUploadFile = function (file) {
125 | const filePath = file.tempFilePath;
126 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
127 | var key = filename;
128 | cos.sliceUploadFile(
129 | {
130 | Bucket: config.Bucket,
131 | Region: config.Region,
132 | Key: key,
133 | FilePath: filePath,
134 | FileSize: file.size,
135 | CacheControl: 'max-age=7200',
136 | Headers: {
137 | aa: 123,
138 | },
139 | Query: {
140 | bb: 123,
141 | },
142 | onTaskReady: function (taskId) {
143 | TaskId = taskId;
144 | },
145 | onHashProgress: function (info) {
146 | console.log('check hash', JSON.stringify(info));
147 | },
148 | onProgress: function (info) {
149 | console.log(JSON.stringify(info));
150 | },
151 | },
152 | requestCallback
153 | );
154 | };
155 | wx.chooseMedia({
156 | count: 1,
157 | mediaType: ['image', 'video'],
158 | sizeType: ['original'],
159 | sourceType: ['album', 'camera'],
160 | success: function (res) {
161 | sliceUploadFile(res.tempFiles[0]);
162 | },
163 | });
164 | // wx.chooseVideo({
165 | // sourceType: ['album','camera'],
166 | // maxDuration: 60,
167 | // camera: 'back',
168 | // success(res) {
169 | // var name = res.tempFilePath.replace(/^.*?([^/]{32}\.\w+)$/, '$1');
170 | // sliceUploadFile({
171 | // name: name,
172 | // path: res.tempFilePath,
173 | // size: res.size,
174 | // });
175 | // },
176 | // fail(err) {
177 | // console.log(err);
178 | // }
179 | // })
180 | }
181 |
182 | function postObject() {
183 | wx.chooseMedia({
184 | count: 1, // 默认9
185 | mediaType: ['image', 'video'],
186 | sizeType: ['original'],
187 | sourceType: ['album', 'camera'],
188 | success: function (res) {
189 | var file = res.tempFiles[0];
190 | cos.postObject(
191 | {
192 | Bucket: config.Bucket,
193 | Region: config.Region,
194 | Key: '1.png',
195 | FilePath: file.tempFilePath,
196 | onTaskReady: function (taskId) {
197 | TaskId = taskId;
198 | },
199 | onProgress: function (info) {
200 | console.log(JSON.stringify(info));
201 | },
202 | },
203 | requestCallback
204 | );
205 | },
206 | });
207 | }
208 |
209 | function putObject() {
210 | wx.chooseMedia({
211 | count: 10,
212 | mediaType: ['image', 'video'],
213 | sizeType: ['original'],
214 | sourceType: ['album', 'camera'],
215 | success: function (res) {
216 | const file = res.tempFiles[0];
217 | const filePath = file.tempFilePath;
218 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
219 | wxfs.readFile({
220 | filePath: filePath,
221 | success: function (res) {
222 | cos.putObject(
223 | {
224 | Bucket: config.Bucket,
225 | Region: config.Region,
226 | Key: filename,
227 | Body: res.data, // 在小程序里,putObject 接口只允许传字符串的内容,不支持 TaskReady 和 onProgress,上传请使用 cos.postObject 接口
228 | Headers: {
229 | // 万象持久化接口,上传时持久化。例子:通过 imageMogr2 接口使用图片缩放功能:指定图片宽度为 200,宽度等比压缩
230 | // 'Pic-Operations': '{"is_pic_info": 1, "rules": [{"fileid": "desample_photo.jpg", "rule": "imageMogr2/thumbnail/200x/"}]}'
231 | },
232 | },
233 | requestCallback
234 | );
235 | },
236 | fail: (err) => console.error(err),
237 | });
238 | },
239 | fail: (err) => console.error(err),
240 | });
241 | }
242 |
243 | function putObjectStr() {
244 | cos.putObject(
245 | {
246 | Bucket: config.Bucket,
247 | Region: config.Region,
248 | Key: '1.txt',
249 | Body: 'hello world', // 在小程序里,putObject 接口只允许传字符串的内容,不支持 TaskReady 和 onProgress,上传请使用 cos.postObject 接口
250 | Headers: {
251 | aa: 123,
252 | },
253 | Query: {
254 | bb: 123,
255 | },
256 | },
257 | requestCallback
258 | );
259 | }
260 |
261 | function putObjectBase64() {
262 | var base64Url =
263 | '';
264 | var m = /data:image\/(\w+);base64,(.*)/.exec(base64Url) || [];
265 | var format = m[1];
266 | var bodyData = m[2];
267 | var fileBuf = wx.base64ToArrayBuffer(bodyData);
268 | cos.putObject(
269 | {
270 | Bucket: config.Bucket,
271 | Region: config.Region,
272 | Key: '1.' + format,
273 | Body: fileBuf,
274 | },
275 | requestCallback
276 | );
277 | }
278 |
279 | function getObjectUrl() {
280 | var url = cos.getObjectUrl(
281 | {
282 | Bucket: config.Bucket, // Bucket 格式:test-1250000000
283 | Region: config.Region,
284 | Key: '1mb.zip',
285 | Expires: 60,
286 | Sign: true,
287 | },
288 | function (err, data) {
289 | console.log(err || data);
290 | }
291 | );
292 | console.log(url);
293 | }
294 |
295 | var toolsDao = {
296 | request: request,
297 | 'uploadFile 高级上传': uploadFile,
298 | 'uploadFiles 批量上传': uploadFiles,
299 | 'sliceUploadFile 分片上传': sliceUploadFile,
300 | // 上传文件适用于单请求上传大文件
301 | 'postObject 简单上传': postObject,
302 | 'putObject 简单上传文件': putObject,
303 | 'putObject 上传字符串': putObjectStr,
304 | // 上传文件
305 | 'putObject base64 转 ArrayBuffer 上传': putObjectBase64,
306 | 'getObjectUrl 获取对象访问url': getObjectUrl,
307 | };
308 |
309 | var bucketDao = {
310 | // Service
311 | 'getService 获取存储桶列表': function () {
312 | cos.getService(requestCallback);
313 | },
314 | // 简单 Bucket 操作
315 | 'putBucket 创建存储桶': function () {
316 | cos.putBucket(
317 | {
318 | Bucket: config.Bucket,
319 | Region: config.Region,
320 | },
321 | requestCallback
322 | );
323 | },
324 | 'headBucket 检索存储桶及其权限': function () {
325 | cos.headBucket(
326 | {
327 | Bucket: config.Bucket,
328 | Region: config.Region,
329 | },
330 | requestCallback
331 | );
332 | },
333 | 'deleteBucket 删除存储桶': function () {
334 | cos.deleteBucket(
335 | {
336 | Bucket: config.Bucket,
337 | Region: config.Region,
338 | },
339 | requestCallback
340 | );
341 | },
342 | 'getBucketACL 查询存储桶 ACL': function () {
343 | cos.getBucketAcl(
344 | {
345 | Bucket: config.Bucket,
346 | Region: config.Region,
347 | },
348 | requestCallback
349 | );
350 | },
351 | 'putBucketACL 设置存储桶 ACL': function () {
352 | cos.putBucketAcl(
353 | {
354 | Bucket: config.Bucket,
355 | Region: config.Region,
356 | ACL: 'public-read',
357 | },
358 | requestCallback
359 | );
360 | },
361 | 'getBucketCors 查询跨域配置': function () {
362 | cos.getBucketCors(
363 | {
364 | Bucket: config.Bucket,
365 | Region: config.Region,
366 | },
367 | requestCallback
368 | );
369 | },
370 | 'putBucketCors 设置跨域配置': function () {
371 | cos.putBucketCors(
372 | {
373 | Bucket: config.Bucket,
374 | Region: config.Region,
375 | ResponseVary: 'true',
376 | CORSRules: [
377 | {
378 | AllowedOrigin: ['*'],
379 | AllowedMethod: ['GET', 'POST', 'PUT', 'DELETE', 'HEAD'],
380 | AllowedHeader: ['*'],
381 | ExposeHeader: ['ETag', 'Content-Length'],
382 | MaxAgeSeconds: '5',
383 | },
384 | ],
385 | },
386 | requestCallback
387 | );
388 | },
389 | 'deleteBucketCors 删除跨域配置': function () {
390 | cos.deleteBucketCors(
391 | {
392 | Bucket: config.Bucket,
393 | Region: config.Region,
394 | },
395 | requestCallback
396 | );
397 | },
398 | 'putBucketPolicy 设置存储桶策略': function () {
399 | var AppId = config.Bucket.substr(config.Bucket.lastIndexOf('-') + 1);
400 | cos.putBucketPolicy(
401 | {
402 | Bucket: config.Bucket,
403 | Region: config.Region,
404 | Policy: {
405 | version: '2.0',
406 | principal: {
407 | qcs: ['qcs::cam::uin/10001:uin/10001'],
408 | }, // 这里的 10001 是 QQ 号
409 | statement: [
410 | {
411 | effect: 'allow',
412 | action: [
413 | 'name/cos:GetBucket',
414 | 'name/cos:PutObject',
415 | 'name/cos:PostObject',
416 | 'name/cos:PutObjectCopy',
417 | 'name/cos:InitiateMultipartUpload',
418 | 'name/cos:UploadPart',
419 | 'name/cos:UploadPartCopy',
420 | 'name/cos:CompleteMultipartUpload',
421 | 'name/cos:AbortMultipartUpload',
422 | 'name/cos:AppendObject',
423 | ],
424 | // "resource": ["qcs::cos:ap-guangzhou:uid/1250000000:test-1250000000/*"] // 1250000000 是 appid
425 | resource: ['qcs::cos:' + config.Region + ':uid/' + AppId + ':' + config.Bucket + '/*'], // 1250000000 是 appid
426 | },
427 | ],
428 | },
429 | },
430 | requestCallback
431 | );
432 | },
433 | 'getBucketPolicy 查询存储桶策略': function () {
434 | cos.getBucketPolicy(
435 | {
436 | Bucket: config.Bucket,
437 | Region: config.Region,
438 | },
439 | requestCallback
440 | );
441 | },
442 | 'deleteBucketPolicy 删除存储桶策略': function () {
443 | cos.deleteBucketPolicy(
444 | {
445 | Bucket: config.Bucket,
446 | Region: config.Region,
447 | },
448 | requestCallback
449 | );
450 | },
451 | 'getBucketLocation 获取Bucket的地域信息': function () {
452 | cos.getBucketLocation(
453 | {
454 | Bucket: config.Bucket,
455 | Region: config.Region,
456 | },
457 | requestCallback
458 | );
459 | },
460 | 'getBucketTagging 获取Bucket标签': function () {
461 | cos.getBucketTagging(
462 | {
463 | Bucket: config.Bucket,
464 | Region: config.Region,
465 | },
466 | requestCallback
467 | );
468 | },
469 | 'putBucketTagging 设置Bucket标签': function () {
470 | cos.putBucketTagging(
471 | {
472 | Bucket: config.Bucket,
473 | Region: config.Region,
474 | Tagging: {
475 | Tags: [
476 | {
477 | Key: 'k1',
478 | Value: 'v1',
479 | },
480 | {
481 | Key: 'k2',
482 | Value: 'v2',
483 | },
484 | ],
485 | },
486 | },
487 | requestCallback
488 | );
489 | },
490 | 'deleteBucketTagging 删除存储桶标签': function () {
491 | cos.deleteBucketTagging(
492 | {
493 | Bucket: config.Bucket,
494 | Region: config.Region,
495 | },
496 | requestCallback
497 | );
498 | },
499 | };
500 |
501 | var objectDao = {
502 | 'getBucket 获取对象列表': function () {
503 | cos.getBucket(
504 | {
505 | Bucket: config.Bucket,
506 | Region: config.Region,
507 | },
508 | requestCallback
509 | );
510 | },
511 | // 上传文件适用于单请求上传大文件
512 | 'postObject 表单上传对象': postObject,
513 | 'putObject 简单上传文件': putObject,
514 | 'putObject 上传字符串': putObjectStr,
515 | // 上传文件
516 | 'putObject base64 转 ArrayBuffer 上传': putObjectBase64,
517 | 'getObject 下载对象': function () {
518 | cos.getObject(
519 | {
520 | Bucket: config.Bucket,
521 | Region: config.Region,
522 | Key: '1.png',
523 | },
524 | function (err, data) {
525 | console.log('getObject:', err || data);
526 | }
527 | );
528 | },
529 | 'abortUploadTask 抛弃分块上传任务': function () {
530 | cos.abortUploadTask(
531 | {
532 | Bucket: config.Bucket,
533 | /* 必须 */ // Bucket 格式:test-1250000000
534 | Region: config.Region,
535 | /* 必须 */
536 | // 格式1,删除单个上传任务
537 | // Level: 'task',
538 | // Key: '10mb.zip',
539 | // UploadId: '14985543913e4e2642e31db217b9a1a3d9b3cd6cf62abfda23372c8d36ffa38585492681e3',
540 | // 格式2,删除单个文件所有未完成上传任务
541 | Level: 'file',
542 | Key: '10mb.zip',
543 | // 格式3,删除 Bucket 下所有未完成上传任务
544 | // Level: 'bucket',
545 | },
546 | requestCallback
547 | );
548 | },
549 | 'deleteObject 删除文件': function () {
550 | cos.deleteObject(
551 | {
552 | Bucket: config.Bucket,
553 | Region: config.Region,
554 | Key: '1.zip',
555 | },
556 | requestCallback
557 | );
558 | },
559 | };
560 |
561 | var advanceObjectDao = {
562 | 'sliceUploadFile 分块上传': function () {
563 | var sliceUploadFile = function (file) {
564 | const filePath = file.tempFilePath;
565 | const filename = filePath.substr(filePath.lastIndexOf('/') + 1);
566 | var key = filename;
567 | cos.sliceUploadFile(
568 | {
569 | Bucket: config.Bucket,
570 | Region: config.Region,
571 | Key: key,
572 | FilePath: filePath,
573 | FileSize: file.size,
574 | CacheControl: 'max-age=7200',
575 | Headers: {
576 | aa: 123,
577 | },
578 | Query: {
579 | bb: 123,
580 | },
581 | onTaskReady: function (taskId) {
582 | TaskId = taskId;
583 | },
584 | onHashProgress: function (info) {
585 | console.log('check hash', JSON.stringify(info));
586 | },
587 | onProgress: function (info) {
588 | console.log(JSON.stringify(info));
589 | },
590 | },
591 | requestCallback
592 | );
593 | };
594 | wx.chooseMedia({
595 | count: 1,
596 | mediaType: ['image', 'video'],
597 | sizeType: ['original'],
598 | sourceType: ['album', 'camera'],
599 | success: function (res) {
600 | sliceUploadFile(res.tempFiles[0]);
601 | },
602 | });
603 | // wx.chooseVideo({
604 | // sourceType: ['album','camera'],
605 | // maxDuration: 60,
606 | // camera: 'back',
607 | // success(res) {
608 | // var name = res.tempFilePath.replace(/^.*?([^/]{32}\.\w+)$/, '$1');
609 | // sliceUploadFile({
610 | // name: name,
611 | // path: res.tempFilePath,
612 | // size: res.size,
613 | // });
614 | // },
615 | // fail(err) {
616 | // console.log(err);
617 | // }
618 | // })
619 | },
620 | 'sliceCopyFile 分块复制对象': function () {
621 | // 创建测试文件
622 | var sourceName = '1.txt';
623 | var Key = '1.slicecopy.exe';
624 |
625 | var sourcePath = config.Bucket + '.cos.' + config.Region + '.myqcloud.com/' + sourceName;
626 |
627 | cos.sliceCopyFile(
628 | {
629 | Bucket: config.Bucket, // Bucket 格式:test-1250000000
630 | Region: config.Region,
631 | Key: Key,
632 | CopySource: sourcePath,
633 | SliceSize: 20 * 1024 * 1024, // 大于20M的文件用分片复制,小于则用单片复制
634 | onProgress: function (info) {
635 | var percent = parseInt(info.percent * 10000) / 100;
636 | var speed = parseInt((info.speed / 1024 / 1024) * 100) / 100;
637 | console.log('进度:' + percent + '%; 速度:' + speed + 'Mb/s;');
638 | },
639 | },
640 | requestCallback
641 | );
642 | },
643 | cancelTask: function () {
644 | cos.cancelTask(TaskId);
645 | console.log('canceled');
646 | },
647 | pauseTask: function () {
648 | cos.pauseTask(TaskId);
649 | console.log('paused');
650 | },
651 | restartTask: function () {
652 | cos.restartTask(TaskId);
653 | console.log('restart');
654 | },
655 | };
656 |
657 | // require('./test');
658 |
659 | module.exports = {
660 | toolsDao: toolsDao,
661 | bucketDao: bucketDao,
662 | objectDao: objectDao,
663 | advanceObjectDao: advanceObjectDao,
664 | };
665 |
--------------------------------------------------------------------------------