Repository: alibaba/anyproxy Branch: master Commit: b93f948107b9 Files: 165 Total size: 1.5 MB Directory structure: gitextract_zmx3jyvr/ ├── .eslintignore ├── .eslintrc ├── .github/ │ └── ISSUE_TEMPLATE.md ├── .gitignore ├── .npmignore ├── .travis.yml ├── CHANGELOG ├── LICENSE ├── README.md ├── babel.config.js ├── bin/ │ ├── anyproxy │ ├── anyproxy-ca │ ├── rootCACheck.js │ └── startServer.js ├── build_scripts/ │ ├── build-doc-site.sh │ └── prebuild-doc.js ├── docs/ │ ├── CNAME │ ├── README.md │ ├── assets/ │ │ ├── auto-lang.js │ │ ├── main.js │ │ └── website.css │ ├── cn/ │ │ ├── index.html │ │ ├── search_index.json │ │ └── src_doc.md │ ├── en/ │ │ ├── index.html │ │ ├── search_index.json │ │ └── src_doc.md │ ├── gitbook/ │ │ ├── fonts/ │ │ │ └── fontawesome/ │ │ │ └── FontAwesome.otf │ │ ├── gitbook-plugin-fontsettings/ │ │ │ ├── fontsettings.js │ │ │ └── website.css │ │ ├── gitbook-plugin-highlight/ │ │ │ ├── ebook.css │ │ │ └── website.css │ │ ├── gitbook-plugin-lunr/ │ │ │ └── search-lunr.js │ │ ├── gitbook-plugin-search/ │ │ │ ├── search-engine.js │ │ │ ├── search.css │ │ │ └── search.js │ │ ├── gitbook-plugin-sharing/ │ │ │ └── buttons.js │ │ ├── gitbook.js │ │ ├── style.css │ │ └── theme.js │ ├── index.html │ └── search_index.json ├── docs-src/ │ ├── CNAME │ ├── LANGS.md │ ├── README.md │ ├── _layouts/ │ │ └── layout.html │ ├── assets/ │ │ ├── auto-lang.js │ │ ├── main.js │ │ └── website.css │ ├── book.json │ ├── cn/ │ │ ├── README.md │ │ ├── SUMMARY.md │ │ ├── _layouts/ │ │ │ ├── layout.html │ │ │ └── website/ │ │ │ └── summary.html │ │ └── src_doc.md │ └── en/ │ ├── README.md │ ├── SUMMARY.md │ ├── _layouts/ │ │ ├── layout.html │ │ └── website/ │ │ └── summary.html │ ├── book.json │ └── src_doc.md ├── jest.config.js ├── lib/ │ ├── certMgr.js │ ├── configUtil.js │ ├── httpsServerMgr.js │ ├── log.js │ ├── recorder.js │ ├── requestErrorHandler.js │ ├── requestHandler.js │ ├── ruleLoader.js │ ├── rule_default.js │ ├── systemProxyMgr.js │ ├── util.js │ ├── webInterface.js │ ├── wsServer.js │ └── wsServerMgr.js ├── module_sample/ │ ├── core_reload.js │ ├── https_config.js │ ├── normal_use.js │ └── simple_use.js ├── package.json ├── proxy.js ├── resource/ │ ├── 502.pug │ ├── cert_download.pug │ └── cert_error.pug ├── rule_sample/ │ ├── sample_modify_request_data.js │ ├── sample_modify_request_header.js │ ├── sample_modify_request_path.js │ ├── sample_modify_request_protocol.js │ ├── sample_modify_response_data.js │ ├── sample_modify_response_header.js │ ├── sample_modify_response_statuscode.js │ └── sample_use_local_response.js ├── test/ │ ├── __snapshots__/ │ │ └── basic.spec.js.snap │ ├── basic.spec.js │ ├── fixtures/ │ │ ├── someRule.js │ │ └── upload.txt │ ├── lib/ │ │ ├── httpsServerMgr.spec.js │ │ ├── ruleLoader.spec.js │ │ └── util.spec.js │ ├── rule/ │ │ ├── beforeDealHttpsRequest.spec.js │ │ ├── beforeSendRequest.spec.js │ │ ├── beforeSendResponse.js │ │ └── onError.spec.js │ ├── util.js │ └── web/ │ ├── curlUtil.spec.js │ └── webInterface.spec.js └── web/ ├── index.html ├── postcss.config.js ├── src/ │ ├── action/ │ │ ├── globalStatusAction.js │ │ └── recordAction.js │ ├── common/ │ │ ├── ApiUtil.js │ │ ├── Constant.js │ │ ├── WsUtil.js │ │ ├── apiUtil.js │ │ ├── commonUtil.js │ │ ├── constant.js │ │ ├── curlUtil.js │ │ ├── promiseUtil.js │ │ └── wsUtil.js │ ├── component/ │ │ ├── download-root-ca.jsx │ │ ├── download-root-ca.less │ │ ├── header-menu.jsx │ │ ├── header-menu.less │ │ ├── json-viewer.jsx │ │ ├── json-viewer.less │ │ ├── left-menu.jsx │ │ ├── left-menu.less │ │ ├── map-local.jsx │ │ ├── map-local.less │ │ ├── modal-panel.jsx │ │ ├── modal-panel.less │ │ ├── record-detail.jsx │ │ ├── record-detail.less │ │ ├── record-filter.jsx │ │ ├── record-filter.less │ │ ├── record-list-diff-worker.jsx │ │ ├── record-panel.jsx │ │ ├── record-panel.less │ │ ├── record-request-detail.jsx │ │ ├── record-response-detail.jsx │ │ ├── record-row.jsx │ │ ├── record-row.less │ │ ├── record-worker.jsx │ │ ├── record-ws-message-detail.jsx │ │ ├── record-ws-message-detail.less │ │ ├── resizable-panel.jsx │ │ ├── resizable-panel.less │ │ ├── table-panel.jsx │ │ ├── table-panel.less │ │ ├── title-bar.jsx │ │ ├── title-bar.less │ │ └── ws-listener.jsx │ ├── index.jsx │ ├── index.less │ ├── reducer/ │ │ ├── globalStatusReducer.js │ │ ├── requestRecordReducer.js │ │ └── rootReducer.js │ ├── saga/ │ │ └── rootSaga.js │ └── style/ │ ├── animate.less │ ├── antd-constant.less │ ├── antd-reset.global.less │ ├── common.less │ └── constant.less └── webpack.config.js ================================================ FILE CONTENTS ================================================ ================================================ FILE: .eslintignore ================================================ node_modules dist web web2 resource *.sh docs ================================================ FILE: .eslintrc ================================================ { "extends": "airbnb", "parser": "babel-eslint", "env": { "browser": true, "node": true, "es6": true, "jest": true }, "globals": { "React": true, "ReactDOM": true, "Zepto": true, "JsBridgeUtil": true }, "rules": { "semi": [0], "comma-dangle": [0], "global-require": [0], "no-alert": [0], "no-console": [0], "no-param-reassign": [0], "max-len": [0], "func-names": [0], "no-underscore-dangle": [0], "no-unused-vars": ["error", { "vars": "all", "args": "none", "ignoreRestSiblings": false }], "object-shorthand": [0], "arrow-body-style": [0], "no-new": [0], "strict": [0], "no-script-url": [0], "spaced-comment": [0], "no-empty": [0], "no-constant-condition": [0], "no-else-return": [0], "no-use-before-define": [0], "no-unused-expressions": [0], "no-class-assign": [0], "new-cap": [0], "array-callback-return": [0], "prefer-template": [0], "no-restricted-syntax": [0], "no-trailing-spaces": [0], "import/no-unresolved": [0], "jsx-a11y/img-has-alt": [0], "camelcase": [0], "consistent-return": [0], "guard-for-in": [0], "one-var": [0], "react/wrap-multilines": [0], "react/no-multi-comp": [0], "react/jsx-no-bind": [0], "react/prop-types": [0], "react/prefer-stateless-function": [0], "react/jsx-first-prop-new-line": [0], "react/sort-comp": [0], "import/no-extraneous-dependencies": [0], "import/extensions": [0], "react/forbid-prop-types": [0], "react/require-default-props": [0], "class-methods-use-this": [0], "jsx-a11y/no-static-element-interactions": [0], "react/no-did-mount-set-state": [0], "jsx-a11y/alt-text": [0], "import/no-dynamic-require": [0], "no-extra-boolean-cast": [0], "no-lonely-if": [0], "no-plusplus": [0], "generator-star-spacing": ["error", {"before": true, "after": false}], "require-yield": [0], "arrow-parens": [0], "no-template-curly-in-string": [0], "no-mixed-operators": [0] } } ================================================ FILE: .github/ISSUE_TEMPLATE.md ================================================ Plese fill the template when you reporting a new issue, thanks! #### Which platform are you running AnyProxy #### The version of the AnyProxy #### Your expected behavior of AnyProxy #### The actual behavior of AnyProxy #### The log of the error ================================================ FILE: .gitignore ================================================ cert/**/*.srl cert/**/*.key cert/**/*.crt cert/**/*.csr web/build/.module-cache/ test/report/*.txt tmp.txt .*.swp ._* .DS_Store .vscode .git .hg .lock-wscript .svn .wafpickle-* .vscode CVS npm-debug.log logs *.log pids *.pid *.seed lib-cov coverage .grunt build/Release node_modules .lock-wscript temp dist test/report *.tgz doc_compiled.md docs-md/cn/doc.md docs-md/en/doc.md docs/gitbook/gitbook-plugin-livereload/ node_modules ================================================ FILE: .npmignore ================================================ cert/**/*.srl cert/**/*.key cert/**/*.crt cert/**/*.csr web/build/.module-cache/ tmp.txt .*.swp ._* .DS_Store .git .hg .lock-wscript .svn .wafpickle-* CVS npm-debug.log logs *.log pids *.pid *.seed lib-cov coverage .grunt node_modules .lock-wscript temp releases rule_sample test *.tgz ================================================ FILE: .travis.yml ================================================ language: node_js node_js: - 12 before_script: - node ./bin/anyproxy-ca -g ================================================ FILE: CHANGELOG ================================================ 22 Dec 2016: AnyProxy 4.0.0-beta: * to AnyProxy rules: all the rule interfaces are asynchronous now, you can write them in a Promise way * to the UI, rewrite the code and enhance the user experience 26 Feb 2016: AnyProxy 3.10.4: * let users assign the port for web socket in AnyProxy cli 19 Sep 2016: AnyProxy 3.10.3: * fix the cert path issue with Windows * split out the cert management to an independent module * add unit tests to AnyProxy 29 Apr 2016: AnyProxy 3.10.0: * using node-forge to generate HTTPS certificates instead of openssl 29 Apr 2016: AnyProxy 3.9.1: * update SHA1 to SHA256 for openssl certificates 19 Nov 2015: AnyProxy 3.8.1: * bugfix for image content in web GUI 19 Nov 2015: AnyProxy 3.8.1: * bugfix for image content in web GUI 16 Nov 2015: AnyProxy 3.8.0: * optimize the memory strategy 2 Oct 2015: AnyProxy 3.7.7: * bugfix for proxy.close() ref #36 9 Sep 2015: AnyProxy 3.7.6: * optimize detail panel, ref #35 3 Sep 2015: AnyProxy 3.7.5: * bugfix for intercepting urls like http://a.com?url=http://b.com 19 Aug 2015: AnyProxy 3.7.4: * bugfix for intercepting urls like http://a.com?url=http://b.com 31 July 2015: AnyProxy 3.7.3: * show lastest 100 records when visit the web ui * save map-local config file to local file * show an indicator when filter or map-local is in use 31 July 2015: AnyProxy 3.7.2: * bugfix for issue #29 28 July 2015: AnyProxy 3.7.1: * fix a bug about deflate compression 20 July 2015: AnyProxy 3.7.0: * add a map-local panel on web ui, now you can easily map some request to your local files 1 July 2015: AnyProxy 3.6.0: * add a filter on web ui 1 July 2015: AnyProxy 3.5.2: * optimize the row height on web ui 18 June 2015: AnyProxy 3.5.1: * print a hint when using SNI features in node <0.12 * Ref : https://github.com/alibaba/anyproxy/issues/25 18 June 2015: AnyProxy 3.5.0: * it's a formal release of 3.4.0@beta. 27 Apr 2015: AnyProxy 3.4.0@beta: * optimize web server and web socket interface 20 Apr 2015: AnyProxy 3.3.1: * now you can assign your own port for web gui 31 Mar 2015: AnyProxy 3.3.0: * optimize https features in windows * add switch to mute the console 20 Mar 2015: AnyProxy 3.2.5: * bugfix for internal https server 19 Mar 2015: AnyProxy 3.2.4: * bugfix for absolute rule path 23 Feb 2015: AnyProxy 3.2.2: * [bugfix for relative rule path](https://github.com/alibaba/anyproxy/pull/18) 10 Feb 2015: AnyProxy 3.2.1: * bugfix for 3.2.0 10 Feb 2015: AnyProxy 3.2.0: * using SNI when intercepting https requests 28 Jan 2015: AnyProxy 3.1.2: * thanks to iconv-lite, almost webpage with any charset can be correctly decoded in web interface. 28 Jan 2015: AnyProxy 3.1.1: * convert GBK to UTF8 in web interface 22 Jan 2015: AnyProxy 3.1.0: * will NOT intercept https request by default. Use ``anyproxy --intercept`` to turn on this feature. 12 Jan 2015: AnyProxy 3.0.4: * show anyproxy version by --version 12 Jan 2015: AnyProxy 3.0.3: * Bugfix: https throttle 9 Jan 2015: AnyProxy 3.0.2: * UI improvement: add link and qr code to root CA file. ================================================ FILE: LICENSE ================================================ Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright [yyyy] [name of copyright owner] Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ================================================ FILE: README.md ================================================ AnyProxy ---------------- [![NPM version][npm-image]][npm-url] [![node version][node-image]][node-url] [![npm download][download-image]][download-url] [![Build Status](https://travis-ci.org/alibaba/anyproxy.svg?branch=master)](https://travis-ci.org/alibaba/anyproxy) [npm-image]: https://img.shields.io/npm/v/anyproxy.svg?style=flat-square [npm-url]: https://npmjs.org/package/anyproxy [node-image]: https://img.shields.io/badge/node.js-%3E=_6.0.0-green.svg?style=flat-square [node-url]: http://nodejs.org/download/ [download-image]: https://img.shields.io/npm/dm/anyproxy.svg?style=flat-square [download-url]: https://npmjs.org/package/anyproxy AnyProxy is A fully configurable HTTP/HTTPS proxy in NodeJS. Home page : [AnyProxy.io](http://anyproxy.io) Issue: https://github.com/alibaba/anyproxy/issues AnyProxy是一个基于NodeJS的,可供插件配置的HTTP/HTTPS代理服务器。 主页:[AnyProxy.io](http://anyproxy.io),访问可能需要稳定的国际网络环境 ![](https://gw.alipayobjects.com/zos/rmsportal/gUfcjGxLONndTfllxynC.jpg@_90q) ---------------- Legacy doc of version 3.x : https://github.com/alibaba/anyproxy/wiki/3.x-docs ================================================ FILE: babel.config.js ================================================ if (process.env.NODE_ENV === 'test') { module.exports = {}; } else { module.exports = { presets: [ 'es2015', 'stage-0' ] }; } ================================================ FILE: bin/anyproxy ================================================ #!/usr/bin/env node 'use strict'; const program = require('commander'), color = require('colorful'), co = require('co'), packageInfo = require('../package.json'), util = require('../lib/util'), rootCACheck = require('./rootCACheck'), startServer = require('./startServer'), logUtil = require('../lib/log'); program .version(packageInfo.version) .option('-p, --port [value]', 'proxy port, 8001 for default') .option('-w, --web [value]', 'web GUI port, 8002 for default') .option('-r, --rule [value]', 'path for rule file,') .option('-l, --throttle [value]', 'throttle speed in kb/s (kbyte / sec)') .option('-i, --intercept', 'intercept(decrypt) https requests when root CA exists') .option('-s, --silent', 'do not print anything into terminal') .option('-c, --clear', 'clear all the certificates and temp files') .option('--ws-intercept', 'intercept websocket') .option('--ignore-unauthorized-ssl', 'ignore all ssl error') .parse(process.argv); if (program.clear) { require('../lib/certMgr').clearCerts(() => { util.deleteFolderContentsRecursive(util.getAnyProxyTmpPath()); console.log(color.green('done !')); process.exit(0); }); } else if (program.root) { require('../lib/certMgr').generateRootCA(() => { process.exit(0); }); } else { co(function *() { if (program.silent) { logUtil.setPrintStatus(false); } if (program.intercept) { try { yield rootCACheck(); } catch (e) { console.error(e); } } return startServer(program); }) } ================================================ FILE: bin/anyproxy-ca ================================================ #!/usr/bin/env node 'use strict' // exist-false, trusted-false : create CA // exist-true, trusted-false : trust CA // exist-true, trusted-true : all things done const program = require('commander'); const color = require('colorful'); const certMgr = require('../lib/certMgr'); const AnyProxy = require('../proxy'); const exec = require('child_process').exec; const co = require('co'); const path = require('path'); const inquirer = require('inquirer'); program .option('-c, --clear', 'clear all the tmp certificates and root CA') .option('-g, --generate', 'generate a new rootCA') .parse(process.argv); function openFolderOfFile(filePath) { const platform = process.platform; if (/^win/.test(platform)) { exec('start .', { cwd: path.dirname(filePath) }); } else if (/darwin/.test(platform)) { exec(`open -R ${filePath}`); } } function guideToGenrateCA() { AnyProxy.utils.certMgr.generateRootCA((error, keyPath, crtPath) => { if (!error) { const certDir = path.dirname(keyPath); console.log(`The cert is generated at ${certDir}. Please trust the ${color.bold('rootCA.crt')}.`); openFolderOfFile(crtPath); } else { console.error('failed to generate rootCA', error); } }); } function guideToTrustCA() { const certPath = AnyProxy.utils.certMgr.getRootCAFilePath(); if (certPath) { openFolderOfFile(certPath); } else { console.error('failed to get cert path'); } } if (program.clear) { AnyProxy.utils.certMgr.clearCerts(() => { console.log(color.green('done !')); }); } else if (program.generate) { guideToGenrateCA(); } else { console.log('detecting CA status...'); co(certMgr.getCAStatus) .then(status => { if (!status.exist) { console.log('AnyProxy CA does not exist.'); const questions = [{ type: 'confirm', name: 'ifCreate', message: 'Would you like to generate one ?', default: true }]; inquirer.prompt(questions).then(answers => { if (answers.ifCreate) { guideToGenrateCA(); } }); } else if (!status.trusted) { if (/^win/.test(process.platform)) { console.log('AnyProxy CA exists, make sure it has been trusted'); } else { console.log('AnyProxy CA exists, but not be trusted'); const questions = [{ type: 'confirm', name: 'ifGotoTrust', message: 'Would you like to open the folder and trust it ?', default: true }]; inquirer.prompt(questions).then(answers => { if (answers.ifGotoTrust) { guideToTrustCA(); } }); } // AnyProxy.utils.certMgr.clearCerts() } else { console.log(color.green('AnyProxy CA has already been trusted')); } }) .catch(e => { console.log(e); }) } ================================================ FILE: bin/rootCACheck.js ================================================ /** * check if root CA exists and installed * will prompt to generate when needed */ const thunkify = require('thunkify'); const AnyProxy = require('../proxy'); const logUtil = require('../lib/log'); const certMgr = AnyProxy.utils.certMgr; function checkRootCAExists() { return certMgr.isRootCAFileExists(); } module.exports = function *() { try { if (!checkRootCAExists()) { logUtil.warn('Missing root CA, generating now'); yield thunkify(certMgr.generateRootCA)(); yield certMgr.trustRootCA(); } else { const isCATrusted = yield thunkify(certMgr.ifRootCATrusted)(); if (!isCATrusted) { logUtil.warn('ROOT CA NOT INSTALLED YET'); yield certMgr.trustRootCA(); } } } catch (e) { console.error(e); } }; ================================================ FILE: bin/startServer.js ================================================ /** * start the AnyProxy server */ const ruleLoader = require('../lib/ruleLoader'); const logUtil = require('../lib/log'); const AnyProxy = require('../proxy'); module.exports = function startServer(program) { let proxyServer; // load rule module new Promise((resolve, reject) => { if (program.rule) { resolve(ruleLoader.requireModule(program.rule)); } else { resolve(null); } }) .catch(e => { logUtil.printLog('Failed to load rule file', logUtil.T_ERR); logUtil.printLog(e, logUtil.T_ERR); process.exit(); }) //start proxy .then(ruleModule => { proxyServer = new AnyProxy.ProxyServer({ type: 'http', port: program.port || 8001, throttle: program.throttle, rule: ruleModule, webInterface: { enable: true, webPort: program.web, }, wsIntercept: program.wsIntercept, forceProxyHttps: program.intercept, dangerouslyIgnoreUnauthorized: !!program.ignoreUnauthorizedSsl, silent: program.silent }); // proxyServer.on('ready', () => {}); proxyServer.start(); }) .catch(e => { logUtil.printLog(e, logUtil.T_ERR); if (e && e.code) { logUtil.printLog('code ' + e.code, logUtil.T_ERR); } logUtil.printLog(e.stack, logUtil.T_ERR); }); process.on('exit', (code) => { if (code > 0) { logUtil.printLog('AnyProxy is about to exit with code: ' + code, logUtil.T_ERR); } process.exit(); }); //exit cause ctrl+c process.on('SIGINT', () => { try { proxyServer && proxyServer.close(); } catch (e) { console.error(e); } process.exit(); }); process.on('uncaughtException', (err) => { let errorTipText = 'got an uncaught exception, is there anything goes wrong in your rule file ?\n'; try { if (err && err.stack) { errorTipText += err.stack; } else { errorTipText += err; } } catch (e) { } logUtil.printLog(errorTipText, logUtil.T_ERR); try { proxyServer && proxyServer.close(); } catch (e) { } process.exit(); }); } ================================================ FILE: build_scripts/build-doc-site.sh ================================================ #!/bin/bash ## get into the "build_scripts" folder regardless of the excution directory parent_path=$( cd "$(dirname "${BASH_SOURCE[0]}")" ; pwd -P ) cd "$parent_path/.." ## compile the doc node ./build_scripts/prebuild-doc.js gitbook build ./docs-src ./docs ## push the doc into github # git add ./docs # git commit -m 'building docs' # git push origin ================================================ FILE: build_scripts/prebuild-doc.js ================================================ const fs = require('fs'); const path = require('path'); const textTpl = [ '```bash', 'anyproxy --rule {{url}}', '```', '```js', '{{content}}', '```' ].join('\n'); /** * * @param {*} config * @param {string} config.input input markdown path * @param {string} config.ouput output markdown path */ function mergeMdWithRuleFile(config) { const doc = fs.readFileSync(config.input, { encoding: 'utf8' }); const rules = doc.match(/\{\{sample-rule:([\S]+)\}\}/g).map((rawToReplace) => ({ raw: rawToReplace, url: rawToReplace.replace(/\{\{sample-rule:([\S]+)\}\}/g, ($0, $1) => { return $1; }) })); const tasks = rules.map((item) => ( new Promise((resolve, reject) => { fs.readFile(item.url, 'utf8', (err, data) => { if (!err) { const result = Object.assign({}, item); result.content = data; resolve(result); } else { reject(err); } }); }) )); // fetch all samples return Promise.all(tasks) .then((results) => { // merge to doc let resultDoc = doc; results.forEach((item) => { const contentToInsert = textTpl.replace('{{url}}', item.url).replace('{{content}}', item.content); resultDoc = resultDoc.replace(item.raw, contentToInsert); }); fs.writeFileSync(config.output, resultDoc); }, (fail) => { console.log('failed to load resource'); console.log(fail); process.exit(); }) .catch(e => { console.log(e); process.exit(); }); } Promise.all([ { input: path.join(__dirname, '../docs-src/cn/src_doc.md'), output: path.join(__dirname, '../docs-src/cn/README.md'), }, { input: path.join(__dirname, '../docs-src/en/src_doc.md'), output: path.join(__dirname, '../docs-src/en/README.md'), } ].map(mergeMdWithRuleFile)).then(result => { console.log('done'); }).catch(e => { console.log('err'); console.log(e); }); ================================================ FILE: docs/CNAME ================================================ anyproxy.io ================================================ FILE: docs/README.md ================================================ # THIS IS AUTO GENERATED FILE, DO NOT EDIT THE HTML DIRECTLY. # YOU CAN EDIT THE SOURCE IN docs-md FOLDER ================================================ FILE: docs/assets/auto-lang.js ================================================ /* eslint no-var: off */ /** * detect if the browser is in UTF-8 zone * @return boolean */ function isUTF8Zone() { return new Date().getTimezoneOffset() === -480; } /** * detect if the browser is already in a locale view */ function isInLocaleView() { return /(cn|en)/i.test(location.href); } function initDefaultLocaleAndStatic() { if (!isInLocaleView()) { location.href = isUTF8Zone() ? '/cn' : 'en'; } } initDefaultLocaleAndStatic(); ================================================ FILE: docs/assets/main.js ================================================ /* eslint no-var: off */ function injectBaiduStatic() { var _hmt = _hmt || []; var hm = document.createElement('script'); var s = document.getElementsByTagName('script')[0]; hm.src = '//hm.baidu.com/hm.js?4e51565b7d471fd6623c163a8fd79e07'; s.parentNode.insertBefore(hm, s); } injectBaiduStatic(); ================================================ FILE: docs/assets/website.css ================================================ .book .book-summary ul.summary li.active>a, .book .book-summary ul.summary li a:hover { color: #008cff; background: transparent; text-decoration: none !important; } h1 { color: #2674BA; } h2 { color: #0099CC; } h3 { color: #108ee9; } h4 { color: #662D91; } h5 { color: #444444; } .gitbook-link { display: none !important; } .summary-title-span { position: relative !important; padding: 0 !important; } .rule-title:after, .sample-title:after { font-size: 12px; padding: 0 3px; border-radius: 3px; color: #fff; } .rule-title:after{ content: 'rule'; background-color: #108ee9; } .sample-title:after { content: 'sample'; background-color: #00a854; } .page-inner { max-width: 1000px !important; } ================================================ FILE: docs/cn/index.html ================================================ 简介 · AnyProxy

AnyProxy

本文档的适用范围是AnyProxy 4.0,欢迎提供反馈

Ref: English Doc

AnyProxy是一个开放式的HTTP代理服务器。

Github主页:https://github.com/alibaba/anyproxy

主要特性包括:

  • 基于Node.js,开放二次开发能力,允许自定义请求处理逻辑
  • 支持Https的解析
  • 提供GUI界面,用以观察请求

相比3.x版本,AnyProxy 4.0的主要变化:

  • 规则文件(Rule)全面支持Promise和Generator
  • 简化了规则文件内的接口
  • Web版界面重构

快速开始

作为全局模块

安装

对于Debian或者Ubuntu系统,在安装AnyProxy之前,可能还需要安装 nodejs-legacy

sudo apt-get install nodejs-legacy

然后,安装AnyProxy

npm install -g anyproxy

启动

  • 命令行启动AnyProxy,默认端口号8001
anyproxy
  • 启动后将终端http代理服务器配置为127.0.0.1:8001即可
  • 访问http://127.0.0.1:8002 ,web界面上能看到所有的请求信息

其他命令

  • 配置启动端口,如1080端口启动
anyproxy --port 1080

作为npm模块使用

AnyProxy可以作为一个npm模块使用,整合进其他工具。

如要启用https解析,请在代理服务器启动前自行调用AnyProxy.utils.certMgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxy-ca方法。

  • 引入
npm i anyproxy --save
  • 使用举例
const AnyProxy = require('anyproxy');
const options = {
  port: 8001,
  rule: require('myRuleModule'),
  webInterface: {
    enable: true,
    webPort: 8002
  },
  throttle: 10000,
  forceProxyHttps: false,
  wsIntercept: false, // 不开启websocket代理
  silent: false
};
const proxyServer = new AnyProxy.ProxyServer(options);

proxyServer.on('ready', () => { /* */ });
proxyServer.on('error', (e) => { /* */ });
proxyServer.start();

//when finished
proxyServer.close();
  • Class: AnyProxy.proxyServer

    • 创建代理服务器

      const proxy = new AnyProxy.proxyServer(options)
      
    • options

      • port {number} 必选,代理服务器端口
      • rule {object} 自定义规则模块
      • throttle {number} 限速值,单位kb/s,默认不限速
      • forceProxyHttps {boolean} 是否强制拦截所有的https,忽略规则模块的返回,默认false
      • silent {boolean} 是否屏蔽所有console输出,默认false
      • dangerouslyIgnoreUnauthorized {boolean} 是否忽略请求中的证书错误,默认false
      • wsIntercept {boolean} 是否开启websocket代理,默认false
      • webInterface {object} web版界面配置
        • enable {boolean} 是否启用web版界面,默认false
        • webPort {number} web版界面端口号,默认8002
    • Event: ready

      • 代理服务器启动完成
      • 示例
      proxy.on('ready', function() { })
      
    • Event: error

      • 代理服务器发生错误
      • 示例
      proxy.on('error', function() { })
      
    • Method: start

      • 启动代理服务器
      • 示例
      proxy.start();
      
    • Method: close

      • 关闭代理服务器
      • 示例
      proxy.close();
      
  • AnyProxy.utils.systemProxyMgr

    • 管理系统的全局代理配置,方法调用时可能会弹出密码框
    • 使用示例
    // 配置127.0.0.1:8001为全局http代理服务器
    AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001');
    
    // 关闭全局代理服务器
    AnyProxy.utils.systemProxyMgr.disableGlobalProxy();
    
  • AnyProxy.utils.certMgr

    • 管理AnyProxy的证书
    • AnyProxy.utils.certMgr.ifRootCAFileExists()
      • 校验系统内是否存在AnyProxy的根证书
    • AnyProxy.utils.certMgr.generateRootCA(callback)
      • 生成AnyProxy的rootCA,完成后请引导用户信任.crt文件
    • 样例
      const AnyProxy = require('anyproxy');
      const exec = require('child_process').exec;
    
      if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) {
        AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => {
          // let users to trust this CA before using proxy
          if (!error) {
            const certDir = require('path').dirname(keyPath);
            console.log('The cert is generated at', certDir);
            const isWin = /^win/.test(process.platform);
            if (isWin) {
              exec('start .', { cwd: certDir });
            } else {
              exec('open .', { cwd: certDir });
            }
          } else {
            console.error('error when generating rootCA', error);
          }
        });
      }
    

代理HTTPS

  • AnyProxy默认不对https请求做处理,如需看到明文信息,需要配置CA证书

解析https请求的原理是中间人攻击(man-in-the-middle),用户必须信任AnyProxy生成的CA证书,才能进行后续流程

  • 生成证书并解析所有https请求
anyproxy-ca #生成rootCA证书,生成后需要手动信任
anyproxy --intercept #启动AnyProxy,并解析所有https请求

代理WebSocket

anyproxy --ws-intercept

当启用HTTPS代理时,wss也会被代理,但是不会被AnyProxy记录。需要开启--ws-intercept后才会从界面上看到相应内容。

rule模块

AnyProxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。

注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题

规则模块的能力范围包括:

  • 拦截并修改正在发送的请求
    • 可修改内容包括请求头(request header),请求体(request body),甚至是请求的目标地址等
  • 拦截并修改服务端响应
    • 可修改的内容包括http状态码(status code)、响应头(response header)、响应内容等
  • 拦截https请求,对内容做修改
    • 本质是中间人攻击(man-in-the-middle attack),需要客户端提前信任AnyProxy生成的CA

开发示例

  • 举例

  • Step 1,编写规则

    // file: sample.js
    module.exports = {
      summary: 'a rule to hack response',
      *beforeSendResponse(requestDetail, responseDetail) {
        if (requestDetail.url === 'http://httpbin.org/user-agent') {
          const newResponse = responseDetail.response;
          newResponse.body += '- AnyProxy Hacked!';
    
          return new Promise((resolve, reject) => {
            setTimeout(() => { // delay
              resolve({ response: newResponse });
            }, 5000);
          });
        }
      },
    };
    
  • Step 2, 启动AnyProxy,加载规则

    • 运行 anyproxy --rule sample.js
  • Step 3, 测试规则

    • 用curl测试

      curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
      
    • 用浏览器测试:配置浏览器http代理为 127.0.0.1:8001,访问 http://httpbin.org/user-agent

    • 经过代理服务器后,期望的返回如下

    {
      "user-agent": "curl/7.43.0"
    }
    - AnyProxy Hacked!
    
  • Step 4, 查看请求信息

处理流程

  • 处理流程图如下

  • 当http请求经过代理服务器时,具体处理过程是:

    • 收集请求所有请求参数,包括method, header, body等
    • AnyProxy调用规则模块beforeSendRequest方法,由模块做处理,返回新的请求参数,或返回响应内容
    • 如果beforeSendRequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。
    • 根据请求参数,向服务端发出请求,接收服务端响应。
    • 调用规则模块beforeSendResponse方法,由模块对响应内容进行处理
    • 把响应信息返回给客户端
  • 当代理服务器收到https请求时,AnyProxy可以替换证书,对请求做明文解析。

    • 调用规则模块beforeDealHttpsRequest方法,如果返回true,会明文解析这个请求,其他请求不处理
    • 被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。

如何引用

如下几种方案都可以用来引用规则模块:

  • 使用本地路径
    anyproxy --rule ./rule.js
    
  • 使用在线地址

    anyproxy --rule https://sample.com/rule.js
    
  • 使用npm包

    • AnyProxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包
    anyproxy --rule ./myRulePkg/ #本地包
    npm i -g myRulePkg && anyproxy --rule myRulePkg #全局包
    

rule接口文档

规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。

module.exports = {
  // 模块介绍
  summary: 'my customized rule for AnyProxy',
  // 发送请求前拦截处理
  *beforeSendRequest(requestDetail) { /* ... */ },
  // 发送响应前处理
  *beforeSendResponse(requestDetail, responseDetail) { /* ... */ },
  // 是否处理https请求
  *beforeDealHttpsRequest(requestDetail) { /* ... */ },
  // 请求出错的事件
  *onError(requestDetail, error) { /* ... */ },
  // https连接服务器出错
  *onConnectError(requestDetail, error) { /* ... */ }
};

规则文件中,除了summary,都是由 co 驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。

summary

summary(): string | summary:string

  • 规则模块的介绍文案,用于AnyProxy提示用户, 可以是一个函数,也可以是一个普通的字符串

beforeSendRequest

beforeSendRequest(requestDetail)

  • AnyProxy向服务端发送请求前,会调用beforeSendRequest,并带上参数requestDetail
  • requestDetail
  • 举例:请求 anyproxy.io 时,requestDetail参数内容大致如下

    {
      protocol: 'http',
      url: 'http://anyproxy.io/',
      requestOptions: {
        hostname: 'anyproxy.io',
        port: 80,
        path: '/',
        method: 'GET',
        headers: {
          Host: 'anyproxy.io',
          'Proxy-Connection': 'keep-alive',
          'User-Agent': '...'
        }
      },
      requestData: '...',
      _req: { /* ... */}
    }
    
  • 以下几种返回都是合法的

    • 不做任何处理,返回null
    return null;
    
    • 修改请求协议,如强制改用https发起请求
    return {
      protocol: 'https'
    };
    
    • 修改请求参数
    var newOption = Object.assign({}, requestDetail.requestOptions);
    newOption.path = '/redirect/to/another/path';
    return {
      requestOptions: newOption
    };
    
    • 修改请求body
    return {
      requestData: 'my new request data'
      //这里也可以同时加上requestOptions
    };
    
    • 直接返回客户端,不再发起请求,其中statusCode header 是必选字段
    return {
      response: {
        statusCode: 200,
        header: { 'content-type': 'text/html' },
        body: 'this could be a <string> or <buffer>'
      }
    };
    

beforeSendResponse

beforeSendResponse(requestDetail, responseDetail)

  • AnyProxy向客户端发送请求前,会调用beforeSendResponse,并带上参数requestDetail responseDetail
  • requestDetailbeforeSendRequest中的参数
  • responseDetail
    • response {object} 服务端的返回信息,包括statusCode header body三个字段
    • _res {object} 原始的服务端返回对象
  • 举例,请求 anyproxy.io 时,responseDetail参数内容大致如下

    {
      response: {
        statusCode: 200,
        header: {
          'Content-Type': 'image/gif',
          Connection: 'close',
          'Cache-Control': '...'
        },
        body: '...'
      },
      _res: { /* ... */ }
    }
    
  • 以下几种返回都是合法的

    • 不做任何处理,返回null
    return null;
    
    • 修改返回的状态码
    var newResponse = Object.assign({}, responseDetail.response);
    newResponse.statusCode = 404;
    return {
      response: newResponse
    };
    
    • 修改返回的内容
    var newResponse = Object.assign({}, responseDetail.response);
    newResponse.body += '--from anyproxy--';
    return {
      response: newResponse
    };
    

beforeDealHttpsRequest

beforeDealHttpsRequest(requestDetail)

  • AnyProxy收到https请求时,会调用beforeDealHttpsRequest,并带上参数requestDetail
  • 如果配置了全局解析https的参数,则AnyProxy会略过这个调用
  • 只有返回true时,AnyProxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。
  • 注意:https over http的代理模式中,这里的request是CONNECT请求
  • requestDetail
    • host {string} 请求目标的Host,受制于协议,这里无法获取完整url
    • _req {object} 请求的原始request
  • 返回值
    • true或者false,表示是否需要AnyProxy替换证书并解析https

onError

onError(requestDetail, error)

  • 在请求处理过程中发生错误时,AnyProxy会调用onError方法,并提供对应的错误信息
  • 多数场景下,错误会在请求目标服务器的时候发生,比如DNS解析失败、请求超时等
  • requestDetailbeforeSendRequest中的参数
  • 以下几种返回都是合法的

    • 不做任何处理。此时AnyProxy会返回一个默认的错误页。
    return null;
    
    • 返回自定义错误页
    return {
      response: {
        statusCode: 200,
        header: { 'content-type': 'text/html' },
        body: 'this could be a <string> or <buffer>'
      }
    };
    

onConnectError

onConnectError(requestDetail, error)

  • AnyProxy在与目标HTTPS服务器建立连接的过程中,如果发生错误,AnyProxy会调用这个方法
  • requestDetailbeforeDealHttpsRequest中的参数
  • 此处无法控制向客户端的返回信息,无需返回值。

rule样例

  • 这里提供一些样例,来讲解规则模块的常见用法
  • 你可以通过 anyproxy --rule http://....js 来加载模块并体验
  • 用curl发请求测试的方法如下
    • 直接请求服务器:curl http://httpbin.org/
    • 通过代理服务器请求:curl http://httpbin.org/ --proxy http://127.0.0.1:8001

使用本地数据

anyproxy --rule rule_sample/sample_use_local_response.js
/* 
  sample: 
    intercept all requests toward httpbin.org, use a local response
  test:
    curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
*/
module.exports = {
  *beforeSendRequest(requestDetail) {
    const localResponse = {
      statusCode: 200,
      header: { 'Content-Type': 'application/json' },
      body: '{"hello": "this is local response"}'
    };
    if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
      return {
        response: localResponse
      };
    }
  },
};

修改请求头

  • 修改发送到 httpbin.org 的user-agent
anyproxy --rule rule_sample/sample_modify_request_header.js
/* 
  sample: 
    modify the user-agent in requests toward httpbin.org
  test:
    curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
*/
module.exports = {
  *beforeSendRequest(requestDetail) {
    if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
      const newRequestOptions = requestDetail.requestOptions;
      newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0';
      return {
        requestOptions: newRequestOptions
      };
    }
  },
};

修改请求数据

anyproxy --rule rule_sample/sample_modify_request_data.js
/*
  sample:
    modify the post data towards http://httpbin.org/post
  test:
    curl -H "Content-Type: text/plain" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001
  expected response:
    { "data": "i-am-anyproxy-modified-post-data" }
*/
module.exports = {
  summary: 'Rule to modify request data',
  *beforeSendRequest(requestDetail) {
    if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) {
      return {
        requestData: 'i-am-anyproxy-modified-post-data'
      };
    }
  },
};

修改请求的目标地址

anyproxy --rule rule_sample/sample_modify_request_path.js
/*
  sample:
    redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html
  test:
    curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001
  expected response:
    'hello world' from 127.0.0.1:8001/index.html
*/
module.exports = {
  *beforeSendRequest(requestDetail) {
    if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) {
      const newRequestOptions = requestDetail.requestOptions;
      requestDetail.protocol = 'http';
      newRequestOptions.hostname = '127.0.0.1'
      newRequestOptions.port = '8008';
      newRequestOptions.path = '/index.html';
      newRequestOptions.method = 'GET';
      return requestDetail;
    }
  },
  *beforeDealHttpsRequest(requestDetail) {
    return true;
  }
};

修改请求协议

anyproxy --rule rule_sample/sample_modify_request_protocol.js
/* 
  sample: 
    redirect all http requests of httpbin.org to https
  test:
    curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001
  expected response:
    { "X-Forwarded-Protocol": "https" }
*/
module.exports = {
  *beforeSendRequest(requestDetail) {
    if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
      const newOption = requestDetail.requestOptions;
      newOption.port = 443;
      return {
        protocol: 'https',
        requestOptions: newOption
      };
    }
  }
};

修改返回状态码

anyproxy --rule rule_sample/sample_modify_response_statuscode.js
/* 
  sample: 
    modify all status code of http://httpbin.org/ to 404
  test:
    curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
  expected response:
    HTTP/1.1 404 Not Found
*/
module.exports = {
  *beforeSendResponse(requestDetail, responseDetail) {
    if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
      const newResponse = responseDetail.response;
      newResponse.statusCode = 404;
      return {
        response: newResponse
      };
    }
  }
};

修改返回头

anyproxy --rule rule_sample/sample_modify_response_header.js
/* 
  sample: 
    modify response header of http://httpbin.org/user-agent
  test:
    curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
  expected response:
    X-Proxy-By: AnyProxy
*/
module.exports = {
  *beforeSendResponse(requestDetail, responseDetail) {
    if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) {
      const newResponse = responseDetail.response;
      newResponse.header['X-Proxy-By'] = 'AnyProxy';
      return {
        response: newResponse
      };
    }
  }
};

修改返回内容并延迟

anyproxy --rule rule_sample/sample_modify_response_data.js
/* 
  sample: 
    modify response data of http://httpbin.org/user-agent
  test:
    curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
  expected response:
    { "user-agent": "curl/7.43.0" } -- AnyProxy Hacked! --
*/

module.exports = {
  *beforeSendResponse(requestDetail, responseDetail) {
    if (requestDetail.url === 'http://httpbin.org/user-agent') {
      const newResponse = responseDetail.response;
      newResponse.body += '-- AnyProxy Hacked! --';
      return new Promise((resolve, reject) => {
        setTimeout(() => { // delay the response for 5s
          resolve({ response: newResponse });
        }, 5000);
      });
    }
  },
};

证书配置

OSX系统信任CA证书

  • 类似这种报错都是因为系统没有信任AnyProxy生成的CA所造成的

警告:CA证书和系统安全息息相关,建议亲自生成,并妥善保管

安装CA:

  • 双击打开rootCA.crt

  • 确认将证书添加到login或system

  • 找到刚刚导入的AnyProxy证书,配置为信任(Always Trust)

Windows系统信任CA证书

配置OSX系统代理

  • 在wifi高级设置中,配置http代理即可

配置浏览器HTTP代理

iOS系统信任CA证书

  • 点击web ui中的 Root CA,按提示扫描二维码即可安装

iOS >= 10.3信任CA证书

  • 除了上述证书安装过程,还需要在 设置->通用->关于本机->证书信任设置 中把AnyProxy证书的开关打开,否则safari将报错。

安卓系统信任CA证书

首先和iOS类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。

  • 下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可
  • 证书下载到指定目录后,需要从其他入口进行安装,包括:
    • 设置 -> 安全性与位置信息 -> 加密与凭据 -> 从存储设备安装。找到你下载的证书文件,进行安装
    • 设置 -> 安全 -> 从SD卡安装证书。找到你下载的证书文件,进行安装

不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为 .crt 的证书文件,少部分仅支持 .cer 文件(已知如 OPPO R15),AnyProxy 提供了多种类型的证书文件,可在下载安装时选择。

配置iOS/Android系统代理

  • 代理服务器都在wifi设置中配置

  • iOS HTTP代理配置

  • Android HTTP代理配置

FAQ

Q: 为什么https请求不能进入处理函数?

A: 以下任意一项都能用来改变https的处理特性:

 1. 命令行启动AnyProxy时配置`--intercept`参数,按npm模块启动时配置`forceProxyHttps`参数,所有Https请求都会被替换证书并解析
 2. 规则文件内提供`beforeDealHttpsRequest`方法,返回 *true* 的https请求会被解析

Q: 提示 function is not yieldable

  • A: 规则模块是用 co 驱动的,函数需要满足yieldable。可以使用generator方法或是返回Promise。

Q: The connection is not private

当访问特定的HTTPS站点,AnyProxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:

  • 命令行直接启动

    通过启动参数 --ignore-unauthorized-ssl 来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。

    anyproxy -i --ignore-unauthorized-ssl
    
  • 在Nodejs代码中启动

    在构造AnyProxy实例的时候,传入参数dangerouslyIgnoreUnauthorized:true, 如下:

     const options = {
      ...,
      dangerouslyIgnoreUnauthorized: true
     };
    
     const anyproxyIns = new AnyProxy.ProxyCore(options);
     anyproxyIns.start();
    

    通过这种方式初始化的AnyProxy,其配置也是全局性的,所有网站的证书问题都会被忽略

  • 通过自定义的Rule来修改

    我们自然也可以借助自定义的Rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。

    module.exports = {
      *beforeSendRequest(requestDetail) {
        if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) {
          const newRequestOptions = requestDetail.requestOptions;
          // 设置属性 rejectUnauthorized 为 false
          newRequestOptions.rejectUnauthorized = false;
          return {
            requestOptions: newRequestOptions
          };
        }
      },
    };
    

results matching ""

    No results matching ""

    ================================================ FILE: docs/cn/search_index.json ================================================ {"index":{"version":"0.5.12","fields":[{"name":"title","boost":10},{"name":"keywords","boost":15},{"name":"body","boost":1}],"ref":"url","documentStore":{"store":{"./":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"user","\"x","#全局包","#启动anyproxy,并解析所有https请求","#本地包","#生成rootca证书,生成后需要手动信任","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","*true*","+=",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","//这里也可以同时加上requestopt","/^win/.test(process.platform);","0)","1.","10.3信任ca证书","10000,","1080","127.0.0.1:8001/index.html","127.0.0.1:8001,访问","1,编写规则","2,","2.","200,","3,","4,","4.0的主要变化:","4.0,欢迎提供反馈","404","404;","443;","5000);","5s","80,","8001,","8002","=","===","=>",">",">=",">关于本机",">证书信任设置",">通用","_re","_req","_req:","_res:","a:","agent","agent\":","agent'","agent')","agent':","agent']","alive',","android","anyproxi","anyproxy',","anyproxy.io","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","anyproxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包","anyproxy可以作为一个npm模块使用,整合进其他工具。","anyproxy向客户端发送请求前,会调用beforesendresponse,并带上参数requestdetail","anyproxy向服务端发送请求前,会调用beforesendrequest,并带上参数requestdetail","anyproxy在与目标https服务器建立连接的过程中,如果发生错误,anyproxy会调用这个方法","anyproxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。","anyproxy收到https请求时,会调用beforedealhttpsrequest,并带上参数requestdetail","anyproxy是一个开放式的http代理服务器。","anyproxy调用规则模块beforesendrequest方法,由模块做处理,返回新的请求参数,或返回响应内容","anyproxy默认不对https请求做处理,如需看到明文信息,需要配置ca证书","apt","at',","attack),需要客户端提前信任anyproxy生成的ca","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","body:","body三个字段","body等","body),甚至是请求的目标地址等","by']","by:","by:anyproxi","ca","ca方法。","ca,按提示扫描二维码即可安装","cert","certdir","certdir);","class:","close","co","code","code)、响应头(respons","connect","connection':","connection:","console.error('error","console.log('th","const","control':","curl","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","data","data\"","data'","data',","delay","doc","enabl","enable:","english","error","error)","error);","event:","exec","exec('open","exec('start","expect","fals","false,","false;","faq","file:","finish","forceproxyhttp","forceproxyhttps:","forward","found","function","function()","g","gener","github主页:https://github.com/alibaba/anyproxi","h","hack","hacked!","hacked!';","header","header),请求体(request","header,","header:","headers:","header)、响应内容等","host","host:","hostname:","http","http/1.1","http://....j","http://127.0.0.1:8001","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://httpbin.org/us","https://sample.com/rule.j","https连接服务器出错","http代理配置","http的代理模式中,这里的request是connect请求","ignor","instal","intercept","intercept`参数,按npm模块启动时配置`forceproxyhttps`参数,所有https请求都会被替换证书并解析","intercept后才会从界面上看到相应内容。","io","ios系统信任ca证书","iswin","keypath)","know.com')","legaci","local","localrespons","method:","middl","middle),用户必须信任anyproxy生成的ca证书,才能进行后续流程","modifi","module.export","myrulepkg","new","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","nodej","npm","null;","object.assign({},","onconnecterror","onconnecterror(requestdetail,","onerror","onerror(requestdetail,","oppo","option","osx系统信任ca证书","over","path:","port","port:","post","privat","promise((resolve,","protocol","protocol\":","protocol:","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","r15),anyproxi","readi","redirect","ref:","reject)","rejectunauthor","request","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requestopt","requestoptions:","require('anyproxy');","require('child_process').exec;","require('myrulemodule'),","require('path').dirname(keypath);","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","root","rootca',","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","rule接口文档","rule样例","rule模块","sample.j","sample:","save","settimeout(()","silent","silent:","site","ssl","start","statu","statuscode:","step","string","sudo","summari","summary():","summary:","summary:str","test:","text/plain\"","throttl","throttle:","toward","true","true,","true;","true或者false,表示是否需要anyproxy替换证书并解析http","trust","trust)","type':","type:","ui中的","unauthor","url","url:","us","user","var","webinterfac","webinterface:","webport","webport:","web版界面端口号,默认8002","web版界面配置","web版界面重构","windows系统信任ca证书","world'","ws","wsintercept","wsintercept:","x","yieldabl","{","{boolean}","{number}","{object}","{string}","|","}","})","});","},","};","下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可","不做任何处理。此时anyproxy会返回一个默认的错误页。","不做任何处理,返回null","不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为","不开启websocket代理","中把anyproxy证书的开关打开,否则safari将报错。","为","为什么https请求不能进入处理函数?","主要特性包括:","举例","举例,请求","举例:请求","从sd卡安装证书。找到你下载的证书文件,进行安装","从存储设备安装。找到你下载的证书文件,进行安装","代理http","代理websocket","代理服务器发生错误","代理服务器启动完成","代理服务器都在wifi设置中配置","以chrome的switchyomega插件为例","以下任意一项都能用来改变https的处理特性:","以下几种返回都是合法的","作为npm模块使用","作为全局模块","你可以通过","使用npm包","使用举例","使用在线地址","使用本地数据","使用本地路径anyproxi","使用示例","修改发送到","修改请求bodi","修改请求协议","修改请求协议,如强制改用https发起请求","修改请求参数","修改请求头","修改请求数据","修改请求的目标地址","修改返回内容并延迟","修改返回头","修改返回状态码","修改返回的内容","修改返回的状态码","关闭代理服务器","关闭全局代理服务器","其他命令","创建代理服务器","加密与凭据","即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback","原始的服务端返回对象","双击打开rootca.crt","发送响应前处理","发送请求前拦截处理","只有返回true时,anyproxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。","可以是一个函数,也可以是一个普通的字符串","可修改内容包括请求头(request","可修改的内容包括http状态码(statu","同beforedealhttpsrequest中的参数","同beforesendrequest中的参数","启动","启动anyproxy,加载规则","启动代理服务器","启动后将终端http代理服务器配置为127.0.0.1:8001即可","命令行启动anyproxy时配置`","命令行启动anyproxy,默认端口号8001","命令行直接启动","在","在nodejs代码中启动","在wifi高级设置中,配置http代理即可","在构造anyproxy实例的时候,传入参数dangerouslyignoreunauthorized:true,","在请求处理过程中发生错误时,anyproxy会调用onerror方法,并提供对应的错误信息","基于node.js,开放二次开发能力,允许自定义请求处理逻辑","处理流程","处理流程图如下","多数场景下,错误会在请求目标服务器的时候发生,比如dns解析失败、请求超时等","如下几种方案都可以用来引用规则模块:","如下:","如何引用","如果beforesendrequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。","如果配置了全局解析https的参数,则anyproxy会略过这个调用","如要启用https解析,请在代理服务器启动前自行调用anyproxy.utils.certmgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxi","安全","安全性与位置信息","安卓系统信任ca证书","安装","安装ca:","对于debian或者ubuntu系统,在安装anyproxy之前,可能还需要安装","开发示例","引入","当http请求经过代理服务器时,具体处理过程是:","当代理服务器收到https请求时,anyproxy可以替换证书,对请求做明文解析。","当启用https代理时,wss也会被代理,但是不会被anyproxy记录。需要开启","当访问特定的https站点,anyproxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:","必选,代理服务器端口","快速开始","我们自然也可以借助自定义的rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。","所有http://httpbin.org","找到刚刚导入的anyproxy证书,配置为信任(alway","把","把响应信息返回给客户端","把所有发送到","把用http协议请求的","拦截https请求,对内容做修改","拦截发送到","拦截并修改服务端响应","拦截并修改正在发送的请求","提供gui界面,用以观察请求","提供了多种类型的证书文件,可在下载安装时选择。","提示","支持https的解析","收集请求所有请求参数,包括method,","改成https并发送","文件(已知如","时,requestdetail参数内容大致如下","时,responsedetail参数内容大致如下","是否启用web版界面,默认fals","是否处理https请求","是否屏蔽所有console输出,默认fals","是否开启websocket代理,默认fals","是否强制拦截所有的https,忽略规则模块的返回,默认fals","是否忽略请求中的证书错误,默认fals","是必选字段","服务端的返回信息,包括statuscod","本文档的适用范围是anyproxi","本质是中间人攻击(man","来加载模块并体验","来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。","查看请求信息","校验系统内是否存在anyproxy的根证书","样例","根据请求参数,向服务端发出请求,接收服务端响应。","模块介绍","此处无法控制向客户端的返回信息,无需返回值。","注意:http","注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题","测试规则","浏览器访问http://127.0.0.1:8002","点击web","然后,安装anyproxi","生成anyproxy的rootca,完成后请引导用户信任.crt文件","生成证书并解析所有https请求","用curl发请求测试的方法如下","用curl测试","用浏览器测试:配置浏览器http代理为","的https请求会被解析","的post数据","的user","的证书文件,少部分仅支持","的请求全部改到","的请求,使用本地数据代替服务端返回","的返回值里加上测试信息,并延迟5秒返回","的返回头里加上","的返回最后追加anyproxy的签名,并延迟5秒","的返回状态码都改成404","直接请求服务器:curl","直接返回客户端,不再发起请求,其中statuscod","相比3.x版本,anyproxi","确认将证书添加到login或system","示例","简介","简化了规则文件内的接口","管理anyproxy的证书","管理系统的全局代理配置,方法调用时可能会弹出密码框","类似这种报错都是因为系统没有信任anyproxy生成的ca所造成的","经过代理服务器后,期望的返回如下","自定义规则模块","被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。","规则文件中,除了summary,都是由","规则文件内提供`beforedealhttpsrequest`方法,返回","规则文件(rule)全面支持promise和gener","规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。","规则模块是用","规则模块的介绍文案,用于anyproxy提示用户,","规则模块的能力范围包括:","解析https请求的原理是中间人攻击(man","警告:ca证书和系统安全息息相关,建议亲自生成,并妥善保管","设置","设置属性","访问http://127.0.0.1:8002","证书下载到指定目录后,需要从其他入口进行安装,包括:","证书配置","请求bodi","请求url","请求使用的协议,http或者http","请求出错的事件","请求的原始request","请求目标的host,受制于协议,这里无法获取完整url","调用规则模块beforedealhttpsrequest方法,如果返回true,会明文解析这个请求,其他请求不处理","调用规则模块beforesendresponse方法,由模块对响应内容进行处理","运行","返回值","返回自定义错误页","这里提供一些样例,来讲解规则模块的常见用法","通过代理服务器请求:curl","通过启动参数","通过自定义的rule来修改","通过这种方式初始化的anyproxy,其配置也是全局性的,所有网站的证书问题都会被忽略","配置127.0.0.1:8001为全局http代理服务器","配置ios/android系统代理","配置osx系统代理","配置启动端口,如1080端口启动","配置浏览器http代理","附录:如何信任ca证书","限速值,单位kb/s,默认不限速","除了上述证书安装过程,还需要在","需要编写一个规则模块,在","首先和ios类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。","驱动的,函数需要满足yieldable。可以使用generator方法或是返回promise。","驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。",",web界面上能看到所有的请求信息",",界面上能看到刚才的请求信息"]},"length":1},"tokenStore":{"root":{"0":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0058953574060427415}}}},"1":{"0":{"0":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}},"docs":{}},"8":{"0":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{},".":{"3":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"docs":{}}},"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},",":{"docs":{},"访":{"docs":{},"问":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},",":{"docs":{},"编":{"docs":{},"写":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"2":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"3":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},";":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"docs":{}},"4":{"3":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},".":{"0":{"docs":{},"的":{"docs":{},"主":{"docs":{},"要":{"docs":{},"变":{"docs":{},"化":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},",":{"docs":{},"欢":{"docs":{},"迎":{"docs":{},"提":{"docs":{},"供":{"docs":{},"反":{"docs":{},"馈":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"docs":{}}},"5":{"0":{"0":{"0":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"docs":{}},"docs":{}},"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"8":{"0":{"0":{"1":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}},"docs":{},"\"":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"/":{"7":{"docs":{},".":{"4":{"3":{"docs":{},".":{"0":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"docs":{}}},"docs":{}},"docs":{}}},"docs":{}}}}}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"t":{"docs":{},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"#":{"docs":{},"全":{"docs":{},"局":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"地":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"生":{"docs":{},"成":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"生":{"docs":{},"成":{"docs":{},"后":{"docs":{},"需":{"docs":{},"要":{"docs":{},"手":{"docs":{},"动":{"docs":{},"信":{"docs":{},"任":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"&":{"docs":{},"&":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"8":{"0":{"0":{"1":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"8":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{"./":{"ref":"./","tf":0.0036845983787767134}},".":{"docs":{},".":{"docs":{},".":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"/":{"docs":{},"t":{"docs":{},"o":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"o":{"docs":{},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{},"/":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}},"a":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}},"/":{"0":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}}},"docs":{}}},"docs":{}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"l":{"docs":{},"i":{"docs":{},"c":{"docs":{},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"/":{"docs":{},"j":{"docs":{},"s":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}}}}}}},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"?":{"docs":{},"s":{"docs":{},"h":{"docs":{},"o":{"docs":{},"w":{"docs":{},"_":{"docs":{},"e":{"docs":{},"n":{"docs":{},"v":{"docs":{},"=":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"m":{"docs":{},"a":{"docs":{},"g":{"docs":{},"e":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"f":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"k":{"docs":{},"e":{"docs":{},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"m":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"t":{"docs":{},"e":{"docs":{},"x":{"docs":{},"t":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"{":{"docs":{},"\"":{"docs":{},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"(":{"docs":{},"!":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},".":{"docs":{},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"e":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"i":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},".":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},".":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},"o":{"docs":{},"f":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"/":{"docs":{},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"*":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.011790714812085483}},"}":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.005158437730287398}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"+":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}},".":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},".":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.005158437730287398}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"/":{"docs":{},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"/":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.012527634487840826}}},"/":{"docs":{"./":{"ref":"./","tf":0.010316875460574797}},"w":{"docs":{},"h":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"这":{"docs":{},"里":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"同":{"docs":{},"时":{"docs":{},"加":{"docs":{},"上":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"^":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},"/":{"docs":{},".":{"docs":{},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},".":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"t":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"m":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"=":{"docs":{"./":{"ref":"./","tf":0.03316138540899042}},"=":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.007369196757553427}}}},">":{"docs":{"./":{"ref":"./","tf":0.005158437730287398}}}},">":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}},"=":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"关":{"docs":{},"于":{"docs":{},"本":{"docs":{},"机":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"信":{"docs":{},"任":{"docs":{},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"通":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"q":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.010316875460574797}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"'":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},")":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"]":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"l":{"docs":{},"i":{"docs":{},"v":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.023581429624170966}}},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},".":{"docs":{},"i":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"c":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},"r":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},".":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"(":{"docs":{},"(":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"s":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"s":{"docs":{},".":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"使":{"docs":{},"用":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},"加":{"docs":{},"载":{"docs":{},"本":{"docs":{},"地":{"docs":{},"规":{"docs":{},"则":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"在":{"docs":{},"参":{"docs":{},"数":{"docs":{},"里":{"docs":{},"传":{"docs":{},"入":{"docs":{},"一":{"docs":{},"个":{"docs":{},"本":{"docs":{},"地":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{},"路":{"docs":{},"径":{"docs":{},",":{"docs":{},"或":{"docs":{},"是":{"docs":{},"某":{"docs":{},"个":{"docs":{},"全":{"docs":{},"局":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"作":{"docs":{},"为":{"docs":{},"一":{"docs":{},"个":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{},",":{"docs":{},"整":{"docs":{},"合":{"docs":{},"进":{"docs":{},"其":{"docs":{},"他":{"docs":{},"工":{"docs":{},"具":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"在":{"docs":{},"与":{"docs":{},"目":{"docs":{},"标":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"建":{"docs":{},"立":{"docs":{},"连":{"docs":{},"接":{"docs":{},"的":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"这":{"docs":{},"个":{"docs":{},"方":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"了":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"的":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"j":{"docs":{},"s":{"docs":{},"编":{"docs":{},"写":{"docs":{},"自":{"docs":{},"己":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"来":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"网":{"docs":{},"络":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"开":{"docs":{},"放":{"docs":{},"式":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"新":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"或":{"docs":{},"返":{"docs":{},"回":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"对":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"如":{"docs":{},"需":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"配":{"docs":{},"置":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"t":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{},")":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"提":{"docs":{},"前":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"d":{"docs":{},"y":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}},"三":{"docs":{},"个":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},")":{"docs":{},",":{"docs":{},"甚":{"docs":{},"至":{"docs":{},"是":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"y":{"docs":{},"'":{"docs":{},"]":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"方":{"docs":{},"法":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},",":{"docs":{},"按":{"docs":{},"提":{"docs":{},"示":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"即":{"docs":{},"可":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"l":{"docs":{},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"d":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"(":{"docs":{},"'":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.01400147383935151}}}},"t":{"docs":{},"r":{"docs":{},"o":{"docs":{},"l":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.006632277081798084}}}},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"w":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"\"":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"'":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"e":{"docs":{},"l":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"o":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"g":{"docs":{},"l":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},")":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"v":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},"'":{"docs":{},"o":{"docs":{},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"p":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.004421518054532056}}}}}}}},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"q":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"n":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"u":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}},"g":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},"主":{"docs":{},"页":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"a":{"docs":{},"l":{"docs":{},"i":{"docs":{},"b":{"docs":{},"a":{"docs":{},"b":{"docs":{},"a":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},"d":{"docs":{},"!":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},")":{"docs":{},",":{"docs":{},"请":{"docs":{},"求":{"docs":{},"体":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},":":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"/":{"1":{"docs":{},".":{"1":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}}},"docs":{}},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"/":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.007369196757553427}}}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},":":{"8":{"0":{"0":{"8":{"docs":{},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"连":{"docs":{},"接":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"出":{"docs":{},"错":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"的":{"docs":{},"代":{"docs":{},"理":{"docs":{},"模":{"docs":{},"式":{"docs":{},"中":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"的":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"是":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"n":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"按":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"启":{"docs":{},"动":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"后":{"docs":{},"才":{"docs":{},"会":{"docs":{},"从":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"看":{"docs":{},"到":{"docs":{},"相":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"n":{"docs":{},"o":{"docs":{},"w":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"c":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}},"i":{"docs":{},"d":{"docs":{},"d":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"用":{"docs":{},"户":{"docs":{},"必":{"docs":{},"须":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"才":{"docs":{},"能":{"docs":{},"进":{"docs":{},"行":{"docs":{},"后":{"docs":{},"续":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0058953574060427415}}}}},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"e":{"docs":{},"x":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.008106116433308769}}}}}}}}}}}}}},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}},"n":{"docs":{},"e":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.004421518054532056}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},".":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},".":{"docs":{},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"s":{"docs":{},"[":{"docs":{},"'":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"r":{"docs":{},"e":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.008843036109064112}},"e":{"docs":{},".":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"[":{"docs":{},"'":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"p":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},".":{"docs":{},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"(":{"docs":{},"{":{"docs":{},"}":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}}}}}}}}}}},"n":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}},"s":{"docs":{},"x":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},":":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}}}}},"r":{"docs":{},"i":{"docs":{},"v":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},":":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.01105379513633014}}},"y":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"q":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}},"r":{"1":{"5":{"docs":{},")":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"docs":{}},"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}},"f":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.004421518054532056}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}},".":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},";":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}}}}}}}}}}}}},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}},"c":{"docs":{},"h":{"docs":{},"i":{"docs":{},"l":{"docs":{},"d":{"docs":{},"_":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"e":{"docs":{},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"'":{"docs":{},")":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},"(":{"docs":{},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},"(":{"docs":{},"{":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}},"e":{"docs":{},"\"":{"docs":{},"}":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},":":{"docs":{"./":{"ref":"./","tf":0.011790714812085483}}},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},")":{"docs":{"./":{"ref":"./","tf":0.004421518054532056}}},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},";":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.015475313190862197}}}}}}},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"c":{"docs":{},"a":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.012527634487840826}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"_":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"/":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"_":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"y":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"接":{"docs":{},"口":{"docs":{},"文":{"docs":{},"档":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},":":{"docs":{"./":{"ref":"./","tf":0.0058953574060427415}}}}}}},"v":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"e":{"docs":{},"t":{"docs":{},"t":{"docs":{},"i":{"docs":{},"m":{"docs":{},"e":{"docs":{},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{},"(":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"s":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}}}}}}}}},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"u":{"docs":{},"d":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0058953574060427415}}}}},"x":{"docs":{},"t":{"docs":{},"/":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"i":{"docs":{},"n":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"t":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"o":{"docs":{},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},";":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"或":{"docs":{},"者":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"表":{"docs":{},"示":{"docs":{},"是":{"docs":{},"否":{"docs":{},"需":{"docs":{},"要":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"y":{"docs":{},"p":{"docs":{},"e":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"u":{"docs":{},"i":{"docs":{},"中":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"v":{"docs":{},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"f":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"重":{"docs":{},"构":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"o":{"docs":{},"w":{"docs":{},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"o":{"docs":{},"r":{"docs":{},"l":{"docs":{},"d":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"{":{"docs":{"./":{"ref":"./","tf":0.06484893146647015}},"b":{"docs":{},"o":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"a":{"docs":{},"n":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0036845983787767134}}}}}}}}}},"n":{"docs":{},"u":{"docs":{},"m":{"docs":{},"b":{"docs":{},"e":{"docs":{},"r":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0058953574060427415}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}}},"|":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"}":{"docs":{"./":{"ref":"./","tf":0.02210759027266028}},")":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},";":{"docs":{"./":{"ref":"./","tf":0.006632277081798084}}}},",":{"docs":{"./":{"ref":"./","tf":0.014738393515106854}}},";":{"docs":{"./":{"ref":"./","tf":0.020633750921149593}}}},"下":{"docs":{},"载":{"docs":{},"后":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"可":{"docs":{},"以":{"docs":{},"直":{"docs":{},"接":{"docs":{},"单":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"并":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"是":{"docs":{},"最":{"docs":{},"简":{"docs":{},"单":{"docs":{},"的":{"docs":{},",":{"docs":{},"直":{"docs":{},"接":{"docs":{},"安":{"docs":{},"装":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"不":{"docs":{},"做":{"docs":{},"任":{"docs":{},"何":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{},"此":{"docs":{},"时":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"返":{"docs":{},"回":{"docs":{},"一":{"docs":{},"个":{"docs":{},"默":{"docs":{},"认":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"n":{"docs":{},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}},"同":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},"类":{"docs":{},"型":{"docs":{},"不":{"docs":{},"尽":{"docs":{},"相":{"docs":{},"同":{"docs":{},",":{"docs":{},"大":{"docs":{},"多":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"拓":{"docs":{},"展":{"docs":{},"名":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"中":{"docs":{},"把":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"开":{"docs":{},"关":{"docs":{},"打":{"docs":{},"开":{"docs":{},",":{"docs":{},"否":{"docs":{},"则":{"docs":{},"s":{"docs":{},"a":{"docs":{},"f":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{},"将":{"docs":{},"报":{"docs":{},"错":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"什":{"docs":{},"么":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"能":{"docs":{},"进":{"docs":{},"入":{"docs":{},"处":{"docs":{},"理":{"docs":{},"函":{"docs":{},"数":{"docs":{},"?":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"主":{"docs":{},"要":{"docs":{},"特":{"docs":{},"性":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"从":{"docs":{},"s":{"docs":{},"d":{"docs":{},"卡":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"存":{"docs":{},"储":{"docs":{},"设":{"docs":{},"备":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"启":{"docs":{},"动":{"docs":{},"完":{"docs":{},"成":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"都":{"docs":{},"在":{"docs":{},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"以":{"docs":{},"c":{"docs":{},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"的":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"t":{"docs":{},"c":{"docs":{},"h":{"docs":{},"y":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"插":{"docs":{},"件":{"docs":{},"为":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"下":{"docs":{},"任":{"docs":{},"意":{"docs":{},"一":{"docs":{},"项":{"docs":{},"都":{"docs":{},"能":{"docs":{},"用":{"docs":{},"来":{"docs":{},"改":{"docs":{},"变":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"特":{"docs":{},"性":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"几":{"docs":{},"种":{"docs":{},"返":{"docs":{},"回":{"docs":{},"都":{"docs":{},"是":{"docs":{},"合":{"docs":{},"法":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}}}}}},"作":{"docs":{},"为":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"全":{"docs":{},"局":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"通":{"docs":{},"过":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"使":{"docs":{},"用":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"在":{"docs":{},"线":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"路":{"docs":{},"径":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"修":{"docs":{},"改":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"协":{"docs":{},"议":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"如":{"docs":{},"强":{"docs":{},"制":{"docs":{},"改":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"内":{"docs":{},"容":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"关":{"docs":{},"闭":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"其":{"docs":{},"他":{"docs":{},"命":{"docs":{},"令":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"创":{"docs":{},"建":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"加":{"docs":{},"密":{"docs":{},"与":{"docs":{},"凭":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"即":{"docs":{},"将":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"供":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"作":{"docs":{},"为":{"docs":{},"使":{"docs":{},"用":{"docs":{},"。":{"docs":{},"详":{"docs":{},"见":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"a":{"docs":{},"p":{"docs":{},"i":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"#":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"_":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"原":{"docs":{},"始":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{},"对":{"docs":{},"象":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"双":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"响":{"docs":{},"应":{"docs":{},"前":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"只":{"docs":{},"有":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"才":{"docs":{},"会":{"docs":{},"尝":{"docs":{},"试":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"、":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"。":{"docs":{},"否":{"docs":{},"则":{"docs":{},"只":{"docs":{},"做":{"docs":{},"数":{"docs":{},"据":{"docs":{},"流":{"docs":{},"转":{"docs":{},"发":{"docs":{},",":{"docs":{},"无":{"docs":{},"法":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"数":{"docs":{},"据":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"函":{"docs":{},"数":{"docs":{},",":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"普":{"docs":{},"通":{"docs":{},"的":{"docs":{},"字":{"docs":{},"符":{"docs":{},"串":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"修":{"docs":{},"改":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"请":{"docs":{},"求":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"(":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"同":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"加":{"docs":{},"载":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"后":{"docs":{},"将":{"docs":{},"终":{"docs":{},"端":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}},"命":{"docs":{},"令":{"docs":{},"行":{"docs":{},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"在":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},"代":{"docs":{},"码":{"docs":{},"中":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"高":{"docs":{},"级":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"构":{"docs":{},"造":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"实":{"docs":{},"例":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},",":{"docs":{},"传":{"docs":{},"入":{"docs":{},"参":{"docs":{},"数":{"docs":{},"d":{"docs":{},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"o":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"并":{"docs":{},"提":{"docs":{},"供":{"docs":{},"对":{"docs":{},"应":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"基":{"docs":{},"于":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{},"s":{"docs":{},",":{"docs":{},"开":{"docs":{},"放":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"允":{"docs":{},"许":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"图":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"多":{"docs":{},"数":{"docs":{},"场":{"docs":{},"景":{"docs":{},"下":{"docs":{},",":{"docs":{},"错":{"docs":{},"误":{"docs":{},"会":{"docs":{},"在":{"docs":{},"请":{"docs":{},"求":{"docs":{},"目":{"docs":{},"标":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},"发":{"docs":{},"生":{"docs":{},",":{"docs":{},"比":{"docs":{},"如":{"docs":{},"d":{"docs":{},"n":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},"失":{"docs":{},"败":{"docs":{},"、":{"docs":{},"请":{"docs":{},"求":{"docs":{},"超":{"docs":{},"时":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"如":{"docs":{},"下":{"docs":{},"几":{"docs":{},"种":{"docs":{},"方":{"docs":{},"案":{"docs":{},"都":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"来":{"docs":{},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"何":{"docs":{},"引":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"果":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"返":{"docs":{},"回":{"docs":{},"了":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},",":{"docs":{},"则":{"docs":{},"立":{"docs":{},"即":{"docs":{},"把":{"docs":{},"此":{"docs":{},"响":{"docs":{},"应":{"docs":{},"返":{"docs":{},"回":{"docs":{},"到":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"(":{"docs":{},"而":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{},"真":{"docs":{},"正":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},")":{"docs":{},",":{"docs":{},"流":{"docs":{},"程":{"docs":{},"结":{"docs":{},"束":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{},"了":{"docs":{},"全":{"docs":{},"局":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"则":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"略":{"docs":{},"过":{"docs":{},"这":{"docs":{},"个":{"docs":{},"调":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"要":{"docs":{},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},",":{"docs":{},"请":{"docs":{},"在":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"启":{"docs":{},"动":{"docs":{},"前":{"docs":{},"自":{"docs":{},"行":{"docs":{},"调":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},"相":{"docs":{},"关":{"docs":{},"方":{"docs":{},"法":{"docs":{},"生":{"docs":{},"成":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"并":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"或":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"使":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"安":{"docs":{},"全":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"性":{"docs":{},"与":{"docs":{},"位":{"docs":{},"置":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"c":{"docs":{},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"对":{"docs":{},"于":{"docs":{},"d":{"docs":{},"e":{"docs":{},"b":{"docs":{},"i":{"docs":{},"a":{"docs":{},"n":{"docs":{},"或":{"docs":{},"者":{"docs":{},"u":{"docs":{},"b":{"docs":{},"u":{"docs":{},"n":{"docs":{},"t":{"docs":{},"u":{"docs":{},"系":{"docs":{},"统":{"docs":{},",":{"docs":{},"在":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"之":{"docs":{},"前":{"docs":{},",":{"docs":{},"可":{"docs":{},"能":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"发":{"docs":{},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"入":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"当":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"请":{"docs":{},"求":{"docs":{},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"时":{"docs":{},",":{"docs":{},"具":{"docs":{},"体":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"是":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"可":{"docs":{},"以":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"对":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"代":{"docs":{},"理":{"docs":{},"时":{"docs":{},",":{"docs":{},"w":{"docs":{},"s":{"docs":{},"s":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"不":{"docs":{},"会":{"docs":{},"被":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"记":{"docs":{},"录":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"开":{"docs":{},"启":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"访":{"docs":{},"问":{"docs":{},"特":{"docs":{},"定":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"站":{"docs":{},"点":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"提":{"docs":{},"示":{"docs":{},"该":{"docs":{},"站":{"docs":{},"点":{"docs":{},"不":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"安":{"docs":{},"全":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"这":{"docs":{},"通":{"docs":{},"常":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"设":{"docs":{},"置":{"docs":{},"不":{"docs":{},"能":{"docs":{},"被":{"docs":{},"正":{"docs":{},"确":{"docs":{},"识":{"docs":{},"别":{"docs":{},"导":{"docs":{},"致":{"docs":{},"的":{"docs":{},"(":{"docs":{},"比":{"docs":{},"如":{"docs":{},",":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"是":{"docs":{},"自":{"docs":{},"签":{"docs":{},"发":{"docs":{},"的":{"docs":{},")":{"docs":{},"。":{"docs":{},"如":{"docs":{},"果":{"docs":{},"您":{"docs":{},"信":{"docs":{},"任":{"docs":{},"该":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"以":{"docs":{},"下":{"docs":{},"方":{"docs":{},"式":{"docs":{},"来":{"docs":{},"继":{"docs":{},"续":{"docs":{},"访":{"docs":{},"问":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},",":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"端":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"快":{"docs":{},"速":{"docs":{},"开":{"docs":{},"始":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"我":{"docs":{},"们":{"docs":{},"自":{"docs":{},"然":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"借":{"docs":{},"助":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"实":{"docs":{},"现":{"docs":{},"这":{"docs":{},"个":{"docs":{},"效":{"docs":{},"果":{"docs":{},",":{"docs":{},"而":{"docs":{},"且":{"docs":{},"我":{"docs":{},"们":{"docs":{},"还":{"docs":{},"可":{"docs":{},"以":{"docs":{},"控":{"docs":{},"制":{"docs":{},"到":{"docs":{},"只":{"docs":{},"允":{"docs":{},"许":{"docs":{},"指":{"docs":{},"定":{"docs":{},"网":{"docs":{},"址":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"对":{"docs":{},"不":{"docs":{},"在":{"docs":{},"列":{"docs":{},"表":{"docs":{},"的":{"docs":{},"网":{"docs":{},"址":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"强":{"docs":{},"验":{"docs":{},"证":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"找":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"刚":{"docs":{},"导":{"docs":{},"入":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"docs":{},"信":{"docs":{},"任":{"docs":{},"(":{"docs":{},"a":{"docs":{},"l":{"docs":{},"w":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"把":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"响":{"docs":{},"应":{"docs":{},"信":{"docs":{},"息":{"docs":{},"返":{"docs":{},"回":{"docs":{},"给":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"协":{"docs":{},"议":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"拦":{"docs":{},"截":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"对":{"docs":{},"内":{"docs":{},"容":{"docs":{},"做":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"并":{"docs":{},"修":{"docs":{},"改":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"正":{"docs":{},"在":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"g":{"docs":{},"u":{"docs":{},"i":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"用":{"docs":{},"以":{"docs":{},"观":{"docs":{},"察":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"了":{"docs":{},"多":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"可":{"docs":{},"在":{"docs":{},"下":{"docs":{},"载":{"docs":{},"安":{"docs":{},"装":{"docs":{},"时":{"docs":{},"选":{"docs":{},"择":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"支":{"docs":{},"持":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"收":{"docs":{},"集":{"docs":{},"请":{"docs":{},"求":{"docs":{},"所":{"docs":{},"有":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"改":{"docs":{},"成":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"并":{"docs":{},"发":{"docs":{},"送":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"文":{"docs":{},"件":{"docs":{},"(":{"docs":{},"已":{"docs":{},"知":{"docs":{},"如":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"时":{"docs":{},",":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"否":{"docs":{},"启":{"docs":{},"用":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"屏":{"docs":{},"蔽":{"docs":{},"所":{"docs":{},"有":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"输":{"docs":{},"出":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"强":{"docs":{},"制":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},",":{"docs":{},"忽":{"docs":{},"略":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"请":{"docs":{},"求":{"docs":{},"中":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"文":{"docs":{},"档":{"docs":{},"的":{"docs":{},"适":{"docs":{},"用":{"docs":{},"范":{"docs":{},"围":{"docs":{},"是":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"质":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"来":{"docs":{},"加":{"docs":{},"载":{"docs":{},"模":{"docs":{},"块":{"docs":{},"并":{"docs":{},"体":{"docs":{},"验":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"证":{"docs":{},"书":{"docs":{},"认":{"docs":{},"证":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"注":{"docs":{},"意":{"docs":{},"的":{"docs":{},"是":{"docs":{},",":{"docs":{},"该":{"docs":{},"参":{"docs":{},"数":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"生":{"docs":{},"效":{"docs":{},"的":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"你":{"docs":{},"在":{"docs":{},"此":{"docs":{},"期":{"docs":{},"间":{"docs":{},"访":{"docs":{},"问":{"docs":{},"了":{"docs":{},"其":{"docs":{},"他":{"docs":{},"未":{"docs":{},"知":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"他":{"docs":{},"们":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{},",":{"docs":{},"这":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"带":{"docs":{},"来":{"docs":{},"安":{"docs":{},"全":{"docs":{},"隐":{"docs":{},"患":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"查":{"docs":{},"看":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"校":{"docs":{},"验":{"docs":{},"系":{"docs":{},"统":{"docs":{},"内":{"docs":{},"是":{"docs":{},"否":{"docs":{},"存":{"docs":{},"在":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"根":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"根":{"docs":{},"据":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"向":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"出":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"接":{"docs":{},"收":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"介":{"docs":{},"绍":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"此":{"docs":{},"处":{"docs":{},"无":{"docs":{},"法":{"docs":{},"控":{"docs":{},"制":{"docs":{},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"无":{"docs":{},"需":{"docs":{},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"注":{"docs":{},"意":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"前":{"docs":{},",":{"docs":{},"请":{"docs":{},"务":{"docs":{},"必":{"docs":{},"确":{"docs":{},"保":{"docs":{},"文":{"docs":{},"件":{"docs":{},"来":{"docs":{},"源":{"docs":{},"可":{"docs":{},"靠":{"docs":{},",":{"docs":{},"以":{"docs":{},"免":{"docs":{},"发":{"docs":{},"生":{"docs":{},"安":{"docs":{},"全":{"docs":{},"问":{"docs":{},"题":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}},"点":{"docs":{},"击":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"然":{"docs":{},"后":{"docs":{},",":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"生":{"docs":{},"成":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},",":{"docs":{},"完":{"docs":{},"成":{"docs":{},"后":{"docs":{},"请":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{},"文":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"用":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"发":{"docs":{},"请":{"docs":{},"求":{"docs":{},"测":{"docs":{},"试":{"docs":{},"的":{"docs":{},"方":{"docs":{},"法":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"测":{"docs":{},"试":{"docs":{},":":{"docs":{},"配":{"docs":{},"置":{"docs":{},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"会":{"docs":{},"被":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"少":{"docs":{},"部":{"docs":{},"分":{"docs":{},"仅":{"docs":{},"支":{"docs":{},"持":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"全":{"docs":{},"部":{"docs":{},"改":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"使":{"docs":{},"用":{"docs":{},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{},"代":{"docs":{},"替":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{},"测":{"docs":{},"试":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"docs":{}}}}}}}}}}}}},"头":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"最":{"docs":{},"后":{"docs":{},"追":{"docs":{},"加":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"签":{"docs":{},"名":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}}}}}}}}}}}}}}}}}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"都":{"docs":{},"改":{"docs":{},"成":{"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"请":{"docs":{},"求":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},",":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"中":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}},"相":{"docs":{},"比":{"3":{"docs":{},".":{"docs":{},"x":{"docs":{},"版":{"docs":{},"本":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"docs":{}}},"确":{"docs":{},"认":{"docs":{},"将":{"docs":{},"证":{"docs":{},"书":{"docs":{},"添":{"docs":{},"加":{"docs":{},"到":{"docs":{},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{},"或":{"docs":{},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"简":{"docs":{},"介":{"docs":{"./":{"ref":"./","tf":10}}},"化":{"docs":{},"了":{"docs":{},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"内":{"docs":{},"的":{"docs":{},"接":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"管":{"docs":{},"理":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"系":{"docs":{},"统":{"docs":{},"的":{"docs":{},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"方":{"docs":{},"法":{"docs":{},"调":{"docs":{},"用":{"docs":{},"时":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"弹":{"docs":{},"出":{"docs":{},"密":{"docs":{},"码":{"docs":{},"框":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"类":{"docs":{},"似":{"docs":{},"这":{"docs":{},"种":{"docs":{},"报":{"docs":{},"错":{"docs":{},"都":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"系":{"docs":{},"统":{"docs":{},"没":{"docs":{},"有":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"所":{"docs":{},"造":{"docs":{},"成":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"后":{"docs":{},",":{"docs":{},"期":{"docs":{},"望":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"被":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"后":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{},"同":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"一":{"docs":{},"致":{"docs":{},"。":{"docs":{},"未":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"会":{"docs":{},"再":{"docs":{},"进":{"docs":{},"入":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"中":{"docs":{},",":{"docs":{},"除":{"docs":{},"了":{"docs":{},"s":{"docs":{},"u":{"docs":{},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"y":{"docs":{},",":{"docs":{},"都":{"docs":{},"是":{"docs":{},"由":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"内":{"docs":{},"提":{"docs":{},"供":{"docs":{},"`":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"`":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},"全":{"docs":{},"面":{"docs":{},"支":{"docs":{},"持":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"和":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"应":{"docs":{},"该":{"docs":{},"符":{"docs":{},"合":{"docs":{},"c":{"docs":{},"m":{"docs":{},"d":{"docs":{},"规":{"docs":{},"范":{"docs":{},",":{"docs":{},"一":{"docs":{},"个":{"docs":{},"典":{"docs":{},"型":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"代":{"docs":{},"码":{"docs":{},"结":{"docs":{},"构":{"docs":{},"如":{"docs":{},"下":{"docs":{},"。":{"docs":{},"模":{"docs":{},"块":{"docs":{},"中":{"docs":{},"所":{"docs":{},"有":{"docs":{},"方":{"docs":{},"法":{"docs":{},"都":{"docs":{},"是":{"docs":{},"可":{"docs":{},"选":{"docs":{},"的":{"docs":{},",":{"docs":{},"只":{"docs":{},"需":{"docs":{},"实":{"docs":{},"现":{"docs":{},"业":{"docs":{},"务":{"docs":{},"感":{"docs":{},"兴":{"docs":{},"趣":{"docs":{},"的":{"docs":{},"部":{"docs":{},"分":{"docs":{},"即":{"docs":{},"可":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"介":{"docs":{},"绍":{"docs":{},"文":{"docs":{},"案":{"docs":{},",":{"docs":{},"用":{"docs":{},"于":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"提":{"docs":{},"示":{"docs":{},"用":{"docs":{},"户":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"能":{"docs":{},"力":{"docs":{},"范":{"docs":{},"围":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"原":{"docs":{},"理":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"警":{"docs":{},"告":{"docs":{},":":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},"和":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"全":{"docs":{},"息":{"docs":{},"息":{"docs":{},"相":{"docs":{},"关":{"docs":{},",":{"docs":{},"建":{"docs":{},"议":{"docs":{},"亲":{"docs":{},"自":{"docs":{},"生":{"docs":{},"成":{"docs":{},",":{"docs":{},"并":{"docs":{},"妥":{"docs":{},"善":{"docs":{},"保":{"docs":{},"管":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"属":{"docs":{},"性":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"下":{"docs":{},"载":{"docs":{},"到":{"docs":{},"指":{"docs":{},"定":{"docs":{},"目":{"docs":{},"录":{"docs":{},"后":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"从":{"docs":{},"其":{"docs":{},"他":{"docs":{},"入":{"docs":{},"口":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"使":{"docs":{},"用":{"docs":{},"的":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"或":{"docs":{},"者":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"出":{"docs":{},"错":{"docs":{},"的":{"docs":{},"事":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"的":{"docs":{},"原":{"docs":{},"始":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}},"目":{"docs":{},"标":{"docs":{},"的":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"受":{"docs":{},"制":{"docs":{},"于":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"无":{"docs":{},"法":{"docs":{},"获":{"docs":{},"取":{"docs":{},"完":{"docs":{},"整":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{},"会":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"这":{"docs":{},"个":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"他":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"对":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"进":{"docs":{},"行":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"运":{"docs":{},"行":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"这":{"docs":{},"里":{"docs":{},"提":{"docs":{},"供":{"docs":{},"一":{"docs":{},"些":{"docs":{},"样":{"docs":{},"例":{"docs":{},",":{"docs":{},"来":{"docs":{},"讲":{"docs":{},"解":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"常":{"docs":{},"见":{"docs":{},"用":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"通":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"请":{"docs":{},"求":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"初":{"docs":{},"始":{"docs":{},"化":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"其":{"docs":{},"配":{"docs":{},"置":{"docs":{},"也":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"性":{"docs":{},"的":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"网":{"docs":{},"站":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"为":{"docs":{},"全":{"docs":{},"局":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"x":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"端":{"docs":{},"口":{"docs":{},",":{"docs":{},"如":{"1":{"0":{"8":{"0":{"docs":{},"端":{"docs":{},"口":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"附":{"docs":{},"录":{"docs":{},":":{"docs":{},"如":{"docs":{},"何":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"限":{"docs":{},"速":{"docs":{},"值":{"docs":{},",":{"docs":{},"单":{"docs":{},"位":{"docs":{},"k":{"docs":{},"b":{"docs":{},"/":{"docs":{},"s":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"限":{"docs":{},"速":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"除":{"docs":{},"了":{"docs":{},"上":{"docs":{},"述":{"docs":{},"证":{"docs":{},"书":{"docs":{},"安":{"docs":{},"装":{"docs":{},"过":{"docs":{},"程":{"docs":{},",":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"需":{"docs":{},"要":{"docs":{},"编":{"docs":{},"写":{"docs":{},"一":{"docs":{},"个":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},",":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"首":{"docs":{},"先":{"docs":{},"和":{"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"类":{"docs":{},"似":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"先":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"进":{"docs":{},"行":{"docs":{},"下":{"docs":{},"载":{"docs":{},"。":{"docs":{},"然":{"docs":{},"后":{"docs":{},"不":{"docs":{},"同":{"docs":{},"的":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"方":{"docs":{},"式":{"docs":{},"可":{"docs":{},"能":{"docs":{},"有":{"docs":{},"所":{"docs":{},"不":{"docs":{},"同":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"步":{"docs":{},"骤":{"docs":{},"是":{"docs":{},"类":{"docs":{},"似":{"docs":{},"的":{"docs":{},",":{"docs":{},"我":{"docs":{},"们":{"docs":{},"列":{"docs":{},"举":{"docs":{},"了":{"docs":{},"几":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"驱":{"docs":{},"动":{"docs":{},"的":{"docs":{},",":{"docs":{},"函":{"docs":{},"数":{"docs":{},"需":{"docs":{},"要":{"docs":{},"满":{"docs":{},"足":{"docs":{},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"。":{"docs":{},"可":{"docs":{},"以":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},"或":{"docs":{},"是":{"docs":{},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"或":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"函":{"docs":{},"数":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"才":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"length":642},"corpusTokens":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"user","\"x","#全局包","#启动anyproxy,并解析所有https请求","#本地包","#生成rootca证书,生成后需要手动信任","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","*true*","+=",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","//这里也可以同时加上requestopt","/^win/.test(process.platform);","0)","1.","10.3信任ca证书","10000,","1080","127.0.0.1:8001/index.html","127.0.0.1:8001,访问","1,编写规则","2,","2.","200,","3,","4,","4.0的主要变化:","4.0,欢迎提供反馈","404","404;","443;","5000);","5s","80,","8001,","8002","=","===","=>",">",">=",">关于本机",">证书信任设置",">通用","_re","_req","_req:","_res:","a:","agent","agent\":","agent'","agent')","agent':","agent']","alive',","android","anyproxi","anyproxy',","anyproxy.io","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","anyproxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包","anyproxy可以作为一个npm模块使用,整合进其他工具。","anyproxy向客户端发送请求前,会调用beforesendresponse,并带上参数requestdetail","anyproxy向服务端发送请求前,会调用beforesendrequest,并带上参数requestdetail","anyproxy在与目标https服务器建立连接的过程中,如果发生错误,anyproxy会调用这个方法","anyproxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。","anyproxy收到https请求时,会调用beforedealhttpsrequest,并带上参数requestdetail","anyproxy是一个开放式的http代理服务器。","anyproxy调用规则模块beforesendrequest方法,由模块做处理,返回新的请求参数,或返回响应内容","anyproxy默认不对https请求做处理,如需看到明文信息,需要配置ca证书","apt","at',","attack),需要客户端提前信任anyproxy生成的ca","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","body:","body三个字段","body等","body),甚至是请求的目标地址等","by']","by:","by:anyproxi","ca","ca方法。","ca,按提示扫描二维码即可安装","cert","certdir","certdir);","class:","close","co","code","code)、响应头(respons","connect","connection':","connection:","console.error('error","console.log('th","const","control':","curl","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","data","data\"","data'","data',","delay","doc","enabl","enable:","english","error","error)","error);","event:","exec","exec('open","exec('start","expect","fals","false,","false;","faq","file:","finish","forceproxyhttp","forceproxyhttps:","forward","found","function","function()","g","gener","github主页:https://github.com/alibaba/anyproxi","h","hack","hacked!","hacked!';","header","header),请求体(request","header,","header:","headers:","header)、响应内容等","host","host:","hostname:","http","http/1.1","http://....j","http://127.0.0.1:8001","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://httpbin.org/us","https://sample.com/rule.j","https连接服务器出错","http代理配置","http的代理模式中,这里的request是connect请求","ignor","instal","intercept","intercept`参数,按npm模块启动时配置`forceproxyhttps`参数,所有https请求都会被替换证书并解析","intercept后才会从界面上看到相应内容。","io","ios系统信任ca证书","iswin","keypath)","know.com')","legaci","local","localrespons","method:","middl","middle),用户必须信任anyproxy生成的ca证书,才能进行后续流程","modifi","module.export","myrulepkg","new","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","nodej","npm","null;","object.assign({},","onconnecterror","onconnecterror(requestdetail,","onerror","onerror(requestdetail,","oppo","option","osx系统信任ca证书","over","path:","port","port:","post","privat","promise((resolve,","protocol","protocol\":","protocol:","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","r15),anyproxi","readi","redirect","ref:","reject)","rejectunauthor","request","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requestopt","requestoptions:","require('anyproxy');","require('child_process').exec;","require('myrulemodule'),","require('path').dirname(keypath);","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","root","rootca',","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","rule接口文档","rule样例","rule模块","sample.j","sample:","save","settimeout(()","silent","silent:","site","ssl","start","statu","statuscode:","step","string","sudo","summari","summary():","summary:","summary:str","test:","text/plain\"","throttl","throttle:","toward","true","true,","true;","true或者false,表示是否需要anyproxy替换证书并解析http","trust","trust)","type':","type:","ui中的","unauthor","url","url:","us","user","var","webinterfac","webinterface:","webport","webport:","web版界面端口号,默认8002","web版界面配置","web版界面重构","windows系统信任ca证书","world'","ws","wsintercept","wsintercept:","x","yieldabl","{","{boolean}","{number}","{object}","{string}","|","}","})","});","},","};","下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可","不做任何处理。此时anyproxy会返回一个默认的错误页。","不做任何处理,返回null","不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为","不开启websocket代理","中把anyproxy证书的开关打开,否则safari将报错。","为","为什么https请求不能进入处理函数?","主要特性包括:","举例","举例,请求","举例:请求","从sd卡安装证书。找到你下载的证书文件,进行安装","从存储设备安装。找到你下载的证书文件,进行安装","代理http","代理websocket","代理服务器发生错误","代理服务器启动完成","代理服务器都在wifi设置中配置","以chrome的switchyomega插件为例","以下任意一项都能用来改变https的处理特性:","以下几种返回都是合法的","作为npm模块使用","作为全局模块","你可以通过","使用npm包","使用举例","使用在线地址","使用本地数据","使用本地路径anyproxi","使用示例","修改发送到","修改请求bodi","修改请求协议","修改请求协议,如强制改用https发起请求","修改请求参数","修改请求头","修改请求数据","修改请求的目标地址","修改返回内容并延迟","修改返回头","修改返回状态码","修改返回的内容","修改返回的状态码","关闭代理服务器","关闭全局代理服务器","其他命令","创建代理服务器","加密与凭据","即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback","原始的服务端返回对象","双击打开rootca.crt","发送响应前处理","发送请求前拦截处理","只有返回true时,anyproxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。","可以是一个函数,也可以是一个普通的字符串","可修改内容包括请求头(request","可修改的内容包括http状态码(statu","同beforedealhttpsrequest中的参数","同beforesendrequest中的参数","启动","启动anyproxy,加载规则","启动代理服务器","启动后将终端http代理服务器配置为127.0.0.1:8001即可","命令行启动anyproxy时配置`","命令行启动anyproxy,默认端口号8001","命令行直接启动","在","在nodejs代码中启动","在wifi高级设置中,配置http代理即可","在构造anyproxy实例的时候,传入参数dangerouslyignoreunauthorized:true,","在请求处理过程中发生错误时,anyproxy会调用onerror方法,并提供对应的错误信息","基于node.js,开放二次开发能力,允许自定义请求处理逻辑","处理流程","处理流程图如下","多数场景下,错误会在请求目标服务器的时候发生,比如dns解析失败、请求超时等","如下几种方案都可以用来引用规则模块:","如下:","如何引用","如果beforesendrequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。","如果配置了全局解析https的参数,则anyproxy会略过这个调用","如要启用https解析,请在代理服务器启动前自行调用anyproxy.utils.certmgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxi","安全","安全性与位置信息","安卓系统信任ca证书","安装","安装ca:","对于debian或者ubuntu系统,在安装anyproxy之前,可能还需要安装","开发示例","引入","当http请求经过代理服务器时,具体处理过程是:","当代理服务器收到https请求时,anyproxy可以替换证书,对请求做明文解析。","当启用https代理时,wss也会被代理,但是不会被anyproxy记录。需要开启","当访问特定的https站点,anyproxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:","必选,代理服务器端口","快速开始","我们自然也可以借助自定义的rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。","所有http://httpbin.org","找到刚刚导入的anyproxy证书,配置为信任(alway","把","把响应信息返回给客户端","把所有发送到","把用http协议请求的","拦截https请求,对内容做修改","拦截发送到","拦截并修改服务端响应","拦截并修改正在发送的请求","提供gui界面,用以观察请求","提供了多种类型的证书文件,可在下载安装时选择。","提示","支持https的解析","收集请求所有请求参数,包括method,","改成https并发送","文件(已知如","时,requestdetail参数内容大致如下","时,responsedetail参数内容大致如下","是否启用web版界面,默认fals","是否处理https请求","是否屏蔽所有console输出,默认fals","是否开启websocket代理,默认fals","是否强制拦截所有的https,忽略规则模块的返回,默认fals","是否忽略请求中的证书错误,默认fals","是必选字段","服务端的返回信息,包括statuscod","本文档的适用范围是anyproxi","本质是中间人攻击(man","来加载模块并体验","来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。","查看请求信息","校验系统内是否存在anyproxy的根证书","样例","根据请求参数,向服务端发出请求,接收服务端响应。","模块介绍","此处无法控制向客户端的返回信息,无需返回值。","注意:http","注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题","测试规则","浏览器访问http://127.0.0.1:8002","点击web","然后,安装anyproxi","生成anyproxy的rootca,完成后请引导用户信任.crt文件","生成证书并解析所有https请求","用curl发请求测试的方法如下","用curl测试","用浏览器测试:配置浏览器http代理为","的https请求会被解析","的post数据","的user","的证书文件,少部分仅支持","的请求全部改到","的请求,使用本地数据代替服务端返回","的返回值里加上测试信息,并延迟5秒返回","的返回头里加上","的返回最后追加anyproxy的签名,并延迟5秒","的返回状态码都改成404","直接请求服务器:curl","直接返回客户端,不再发起请求,其中statuscod","相比3.x版本,anyproxi","确认将证书添加到login或system","示例","简介","简化了规则文件内的接口","管理anyproxy的证书","管理系统的全局代理配置,方法调用时可能会弹出密码框","类似这种报错都是因为系统没有信任anyproxy生成的ca所造成的","经过代理服务器后,期望的返回如下","自定义规则模块","被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。","规则文件中,除了summary,都是由","规则文件内提供`beforedealhttpsrequest`方法,返回","规则文件(rule)全面支持promise和gener","规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。","规则模块是用","规则模块的介绍文案,用于anyproxy提示用户,","规则模块的能力范围包括:","解析https请求的原理是中间人攻击(man","警告:ca证书和系统安全息息相关,建议亲自生成,并妥善保管","设置","设置属性","访问http://127.0.0.1:8002","证书下载到指定目录后,需要从其他入口进行安装,包括:","证书配置","请求bodi","请求url","请求使用的协议,http或者http","请求出错的事件","请求的原始request","请求目标的host,受制于协议,这里无法获取完整url","调用规则模块beforedealhttpsrequest方法,如果返回true,会明文解析这个请求,其他请求不处理","调用规则模块beforesendresponse方法,由模块对响应内容进行处理","运行","返回值","返回自定义错误页","这里提供一些样例,来讲解规则模块的常见用法","通过代理服务器请求:curl","通过启动参数","通过自定义的rule来修改","通过这种方式初始化的anyproxy,其配置也是全局性的,所有网站的证书问题都会被忽略","配置127.0.0.1:8001为全局http代理服务器","配置ios/android系统代理","配置osx系统代理","配置启动端口,如1080端口启动","配置浏览器http代理","附录:如何信任ca证书","限速值,单位kb/s,默认不限速","除了上述证书安装过程,还需要在","需要编写一个规则模块,在","首先和ios类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。","驱动的,函数需要满足yieldable。可以使用generator方法或是返回promise。","驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。",",web界面上能看到所有的请求信息",",界面上能看到刚才的请求信息"],"pipeline":["stopWordFilter","stemmer"]},"store":{"./":{"url":"./","title":"简介","keywords":"","body":"AnyProxy\n\n本文档的适用范围是AnyProxy 4.0,欢迎提供反馈\n\nRef: English Doc\nAnyProxy是一个开放式的HTTP代理服务器。\nGithub主页:https://github.com/alibaba/anyproxy\n主要特性包括:\n\n基于Node.js,开放二次开发能力,允许自定义请求处理逻辑\n支持Https的解析\n提供GUI界面,用以观察请求\n\n相比3.x版本,AnyProxy 4.0的主要变化:\n\n规则文件(Rule)全面支持Promise和Generator\n简化了规则文件内的接口\nWeb版界面重构\n\n\n快速开始\n作为全局模块\n安装\n对于Debian或者Ubuntu系统,在安装AnyProxy之前,可能还需要安装 nodejs-legacy\nsudo apt-get install nodejs-legacy\n\n然后,安装AnyProxy\nnpm install -g anyproxy\n\n启动\n\n命令行启动AnyProxy,默认端口号8001\n\nanyproxy\n\n\n启动后将终端http代理服务器配置为127.0.0.1:8001即可\n访问http://127.0.0.1:8002 ,web界面上能看到所有的请求信息\n\n其他命令\n\n配置启动端口,如1080端口启动\n\nanyproxy --port 1080\n\n作为npm模块使用\nAnyProxy可以作为一个npm模块使用,整合进其他工具。\n\n如要启用https解析,请在代理服务器启动前自行调用AnyProxy.utils.certMgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxy-ca方法。\n\n\n引入\n\nnpm i anyproxy --save\n\n\n使用举例\n\nconst AnyProxy = require('anyproxy');\nconst options = {\n port: 8001,\n rule: require('myRuleModule'),\n webInterface: {\n enable: true,\n webPort: 8002\n },\n throttle: 10000,\n forceProxyHttps: false,\n wsIntercept: false, // 不开启websocket代理\n silent: false\n};\nconst proxyServer = new AnyProxy.ProxyServer(options);\n\nproxyServer.on('ready', () => { /* */ });\nproxyServer.on('error', (e) => { /* */ });\nproxyServer.start();\n\n//when finished\nproxyServer.close();\n\n\nClass: AnyProxy.proxyServer\n\n创建代理服务器\nconst proxy = new AnyProxy.proxyServer(options)\n\n\noptions\n\nport {number} 必选,代理服务器端口\nrule {object} 自定义规则模块\nthrottle {number} 限速值,单位kb/s,默认不限速\nforceProxyHttps {boolean} 是否强制拦截所有的https,忽略规则模块的返回,默认false\nsilent {boolean} 是否屏蔽所有console输出,默认false\ndangerouslyIgnoreUnauthorized {boolean} 是否忽略请求中的证书错误,默认false\nwsIntercept {boolean} 是否开启websocket代理,默认false\nwebInterface {object} web版界面配置\nenable {boolean} 是否启用web版界面,默认false\nwebPort {number} web版界面端口号,默认8002\n\n\n\n\nEvent: ready\n\n代理服务器启动完成\n示例\n\nproxy.on('ready', function() { })\n\n\nEvent: error\n\n代理服务器发生错误\n示例\n\nproxy.on('error', function() { })\n\n\nMethod: start\n\n启动代理服务器\n示例\n\nproxy.start();\n\n\nMethod: close\n\n关闭代理服务器\n示例\n\nproxy.close();\n\n\n\n\nAnyProxy.utils.systemProxyMgr\n\n管理系统的全局代理配置,方法调用时可能会弹出密码框\n使用示例\n\n// 配置127.0.0.1:8001为全局http代理服务器\nAnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001');\n\n// 关闭全局代理服务器\nAnyProxy.utils.systemProxyMgr.disableGlobalProxy();\n\n\nAnyProxy.utils.certMgr\n\n管理AnyProxy的证书\nAnyProxy.utils.certMgr.ifRootCAFileExists()\n校验系统内是否存在AnyProxy的根证书\n\n\nAnyProxy.utils.certMgr.generateRootCA(callback)\n生成AnyProxy的rootCA,完成后请引导用户信任.crt文件\n\n\n样例\n\n const AnyProxy = require('anyproxy');\n const exec = require('child_process').exec;\n\n if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) {\n AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => {\n // let users to trust this CA before using proxy\n if (!error) {\n const certDir = require('path').dirname(keyPath);\n console.log('The cert is generated at', certDir);\n const isWin = /^win/.test(process.platform);\n if (isWin) {\n exec('start .', { cwd: certDir });\n } else {\n exec('open .', { cwd: certDir });\n }\n } else {\n console.error('error when generating rootCA', error);\n }\n });\n }\n\n\n\n代理HTTPS\n\nAnyProxy默认不对https请求做处理,如需看到明文信息,需要配置CA证书\n\n\n解析https请求的原理是中间人攻击(man-in-the-middle),用户必须信任AnyProxy生成的CA证书,才能进行后续流程\n\n\n生成证书并解析所有https请求\n\nanyproxy-ca #生成rootCA证书,生成后需要手动信任\nanyproxy --intercept #启动AnyProxy,并解析所有https请求\n\n\n附录:如何信任CA证书\n\n代理WebSocket\nanyproxy --ws-intercept\n\n\n当启用HTTPS代理时,wss也会被代理,但是不会被AnyProxy记录。需要开启--ws-intercept后才会从界面上看到相应内容。\n\nrule模块\nAnyProxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。\n\n注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题\n\n规则模块的能力范围包括:\n\n拦截并修改正在发送的请求\n可修改内容包括请求头(request header),请求体(request body),甚至是请求的目标地址等\n\n\n拦截并修改服务端响应\n可修改的内容包括http状态码(status code)、响应头(response header)、响应内容等\n\n\n拦截https请求,对内容做修改\n本质是中间人攻击(man-in-the-middle attack),需要客户端提前信任AnyProxy生成的CA\n\n\n\n开发示例\n\n举例\n\n需要编写一个规则模块,在 GET http://httpbin.org/user-agent 的返回值里加上测试信息,并延迟5秒返回\n\n\nStep 1,编写规则\n// file: sample.js\nmodule.exports = {\n summary: 'a rule to hack response',\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '- AnyProxy Hacked!';\n\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\n\nStep 2, 启动AnyProxy,加载规则\n\n运行 anyproxy --rule sample.js\n\n\nStep 3, 测试规则\n\n用curl测试\ncurl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n\n\n用浏览器测试:配置浏览器http代理为 127.0.0.1:8001,访问 http://httpbin.org/user-agent\n\n经过代理服务器后,期望的返回如下\n\n\n{\n \"user-agent\": \"curl/7.43.0\"\n}\n- AnyProxy Hacked!\n\nStep 4, 查看请求信息\n\n浏览器访问http://127.0.0.1:8002 ,界面上能看到刚才的请求信息\n\n\n\n处理流程\n\n处理流程图如下\n\n\n\n当http请求经过代理服务器时,具体处理过程是:\n\n收集请求所有请求参数,包括method, header, body等\nAnyProxy调用规则模块beforeSendRequest方法,由模块做处理,返回新的请求参数,或返回响应内容\n如果beforeSendRequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。\n根据请求参数,向服务端发出请求,接收服务端响应。\n调用规则模块beforeSendResponse方法,由模块对响应内容进行处理\n把响应信息返回给客户端\n\n\n当代理服务器收到https请求时,AnyProxy可以替换证书,对请求做明文解析。\n\n调用规则模块beforeDealHttpsRequest方法,如果返回true,会明文解析这个请求,其他请求不处理\n被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。\n\n\n\n如何引用\n如下几种方案都可以用来引用规则模块:\n\n使用本地路径anyproxy --rule ./rule.js\n\n\n使用在线地址\nanyproxy --rule https://sample.com/rule.js\n\n\n使用npm包\n\nAnyProxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包\n\nanyproxy --rule ./myRulePkg/ #本地包\nnpm i -g myRulePkg && anyproxy --rule myRulePkg #全局包\n\n\n\nrule接口文档\n规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。\nmodule.exports = {\n // 模块介绍\n summary: 'my customized rule for AnyProxy',\n // 发送请求前拦截处理\n *beforeSendRequest(requestDetail) { /* ... */ },\n // 发送响应前处理\n *beforeSendResponse(requestDetail, responseDetail) { /* ... */ },\n // 是否处理https请求\n *beforeDealHttpsRequest(requestDetail) { /* ... */ },\n // 请求出错的事件\n *onError(requestDetail, error) { /* ... */ },\n // https连接服务器出错\n *onConnectError(requestDetail, error) { /* ... */ }\n};\n\n\n规则文件中,除了summary,都是由 co 驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。\n\nsummary\nsummary(): string | summary:string\n\n规则模块的介绍文案,用于AnyProxy提示用户, 可以是一个函数,也可以是一个普通的字符串\n\nbeforeSendRequest\nbeforeSendRequest(requestDetail)\n\nAnyProxy向服务端发送请求前,会调用beforeSendRequest,并带上参数requestDetail\nrequestDetail\nprotocol {string} 请求使用的协议,http或者https\nrequestOptions {object} 即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback\nrequestData {object} 请求Body\nurl {string} 请求url\n_req {object} 请求的原始request\n\n\n举例:请求 anyproxy.io 时,requestDetail参数内容大致如下\n{\n protocol: 'http',\n url: 'http://anyproxy.io/',\n requestOptions: {\n hostname: 'anyproxy.io',\n port: 80,\n path: '/',\n method: 'GET',\n headers: {\n Host: 'anyproxy.io',\n 'Proxy-Connection': 'keep-alive',\n 'User-Agent': '...'\n }\n },\n requestData: '...',\n _req: { /* ... */}\n}\n\n\n以下几种返回都是合法的\n\n不做任何处理,返回null\n\nreturn null;\n\n\n修改请求协议,如强制改用https发起请求\n\nreturn {\n protocol: 'https'\n};\n\n\n修改请求参数\n\nvar newOption = Object.assign({}, requestDetail.requestOptions);\nnewOption.path = '/redirect/to/another/path';\nreturn {\n requestOptions: newOption\n};\n\n\n修改请求body\n\nreturn {\n requestData: 'my new request data'\n //这里也可以同时加上requestOptions\n};\n\n\n直接返回客户端,不再发起请求,其中statusCode header 是必选字段\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nbeforeSendResponse\nbeforeSendResponse(requestDetail, responseDetail)\n\nAnyProxy向客户端发送请求前,会调用beforeSendResponse,并带上参数requestDetail responseDetail\nrequestDetail 同beforeSendRequest中的参数\nresponseDetail\nresponse {object} 服务端的返回信息,包括statusCode header body三个字段\n_res {object} 原始的服务端返回对象\n\n\n举例,请求 anyproxy.io 时,responseDetail参数内容大致如下\n{\n response: {\n statusCode: 200,\n header: {\n 'Content-Type': 'image/gif',\n Connection: 'close',\n 'Cache-Control': '...'\n },\n body: '...'\n },\n _res: { /* ... */ }\n}\n\n\n以下几种返回都是合法的\n\n不做任何处理,返回null\n\nreturn null;\n\n\n修改返回的状态码\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.statusCode = 404;\nreturn {\n response: newResponse\n};\n\n\n修改返回的内容\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.body += '--from anyproxy--';\nreturn {\n response: newResponse\n};\n\n\n\nbeforeDealHttpsRequest\nbeforeDealHttpsRequest(requestDetail)\n\nAnyProxy收到https请求时,会调用beforeDealHttpsRequest,并带上参数requestDetail\n如果配置了全局解析https的参数,则AnyProxy会略过这个调用\n只有返回true时,AnyProxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。\n注意:https over http的代理模式中,这里的request是CONNECT请求\nrequestDetail\nhost {string} 请求目标的Host,受制于协议,这里无法获取完整url\n_req {object} 请求的原始request\n\n\n返回值\ntrue或者false,表示是否需要AnyProxy替换证书并解析https\n\n\n\nonError\nonError(requestDetail, error)\n\n在请求处理过程中发生错误时,AnyProxy会调用onError方法,并提供对应的错误信息\n多数场景下,错误会在请求目标服务器的时候发生,比如DNS解析失败、请求超时等\nrequestDetail 同beforeSendRequest中的参数\n以下几种返回都是合法的\n\n不做任何处理。此时AnyProxy会返回一个默认的错误页。\n\nreturn null;\n\n\n返回自定义错误页\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nonConnectError\nonConnectError(requestDetail, error)\n\nAnyProxy在与目标HTTPS服务器建立连接的过程中,如果发生错误,AnyProxy会调用这个方法\nrequestDetail 同beforeDealHttpsRequest中的参数\n此处无法控制向客户端的返回信息,无需返回值。\n\nrule样例\n\n这里提供一些样例,来讲解规则模块的常见用法\n你可以通过 anyproxy --rule http://....js 来加载模块并体验\n用curl发请求测试的方法如下\n直接请求服务器:curl http://httpbin.org/\n通过代理服务器请求:curl http://httpbin.org/ --proxy http://127.0.0.1:8001\n\n\n\n使用本地数据\n\n拦截发送到 http://httpbin.org 的请求,使用本地数据代替服务端返回\n\nanyproxy --rule rule_sample/sample_use_local_response.js\n\n/* \n sample: \n intercept all requests toward httpbin.org, use a local response\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n const localResponse = {\n statusCode: 200,\n header: { 'Content-Type': 'application/json' },\n body: '{\"hello\": \"this is local response\"}'\n };\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n return {\n response: localResponse\n };\n }\n },\n};\n\n修改请求头\n\n修改发送到 httpbin.org 的user-agent\n\nanyproxy --rule rule_sample/sample_modify_request_header.js\n\n/* \n sample: \n modify the user-agent in requests toward httpbin.org\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0';\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\n修改请求数据\n\n修改发送到 http://httpbin.org/post 的post数据\n\nanyproxy --rule rule_sample/sample_modify_request_data.js\n\n/*\n sample:\n modify the post data towards http://httpbin.org/post\n test:\n curl -H \"Content-Type: text/plain\" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001\n expected response:\n { \"data\": \"i-am-anyproxy-modified-post-data\" }\n*/\nmodule.exports = {\n summary: 'Rule to modify request data',\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) {\n return {\n requestData: 'i-am-anyproxy-modified-post-data'\n };\n }\n },\n};\n\n修改请求的目标地址\n\n把所有发送到 http://httpbin.org/ 的请求全部改到 http://httpbin.org/user-agent\n\nanyproxy --rule rule_sample/sample_modify_request_path.js\n\n/*\n sample:\n redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html\n test:\n curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n expected response:\n 'hello world' from 127.0.0.1:8001/index.html\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n requestDetail.protocol = 'http';\n newRequestOptions.hostname = '127.0.0.1'\n newRequestOptions.port = '8008';\n newRequestOptions.path = '/index.html';\n newRequestOptions.method = 'GET';\n return requestDetail;\n }\n },\n *beforeDealHttpsRequest(requestDetail) {\n return true;\n }\n};\n\n修改请求协议\n\n把用http协议请求的 http://httpbin.org 改成https并发送\n\nanyproxy --rule rule_sample/sample_modify_request_protocol.js\n\n/* \n sample: \n redirect all http requests of httpbin.org to https\n test:\n curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001\n expected response:\n { \"X-Forwarded-Protocol\": \"https\" }\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newOption = requestDetail.requestOptions;\n newOption.port = 443;\n return {\n protocol: 'https',\n requestOptions: newOption\n };\n }\n }\n};\n\n修改返回状态码\n\n把 所有http://httpbin.org 的返回状态码都改成404\n\nanyproxy --rule rule_sample/sample_modify_response_statuscode.js\n\n/* \n sample: \n modify all status code of http://httpbin.org/ to 404\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n HTTP/1.1 404 Not Found\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newResponse = responseDetail.response;\n newResponse.statusCode = 404;\n return {\n response: newResponse\n };\n }\n }\n};\n\n修改返回头\n\n在 http://httpbin.org/user-agent 的返回头里加上 X-Proxy-By:AnyProxy\n\nanyproxy --rule rule_sample/sample_modify_response_header.js\n\n/* \n sample: \n modify response header of http://httpbin.org/user-agent\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n X-Proxy-By: AnyProxy\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) {\n const newResponse = responseDetail.response;\n newResponse.header['X-Proxy-By'] = 'AnyProxy';\n return {\n response: newResponse\n };\n }\n }\n};\n\n修改返回内容并延迟\n\n在 http://httpbin.org/user-agent 的返回最后追加AnyProxy的签名,并延迟5秒\n\nanyproxy --rule rule_sample/sample_modify_response_data.js\n\n/* \n sample: \n modify response data of http://httpbin.org/user-agent\n test:\n curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n { \"user-agent\": \"curl/7.43.0\" } -- AnyProxy Hacked! --\n*/\n\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '-- AnyProxy Hacked! --';\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay the response for 5s\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\n证书配置\nOSX系统信任CA证书\n\n类似这种报错都是因为系统没有信任AnyProxy生成的CA所造成的\n\n\n\n警告:CA证书和系统安全息息相关,建议亲自生成,并妥善保管\n\n安装CA:\n\n双击打开rootCA.crt\n\n确认将证书添加到login或system\n\n\n\n\n找到刚刚导入的AnyProxy证书,配置为信任(Always Trust)\n\n\nWindows系统信任CA证书\n\n配置OSX系统代理\n\n在wifi高级设置中,配置http代理即可\n\n\n配置浏览器HTTP代理\n\n以Chrome的SwitchyOmega插件为例\n\n\niOS系统信任CA证书\n\n点击web ui中的 Root CA,按提示扫描二维码即可安装\n\n\niOS >= 10.3信任CA证书\n\n除了上述证书安装过程,还需要在 设置->通用->关于本机->证书信任设置 中把AnyProxy证书的开关打开,否则safari将报错。\n\n\n安卓系统信任CA证书\n首先和iOS类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。\n\n下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可\n证书下载到指定目录后,需要从其他入口进行安装,包括:\n设置 -> 安全性与位置信息 -> 加密与凭据 -> 从存储设备安装。找到你下载的证书文件,进行安装\n设置 -> 安全 -> 从SD卡安装证书。找到你下载的证书文件,进行安装\n\n\n\n不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为 .crt 的证书文件,少部分仅支持 .cer 文件(已知如 OPPO R15),AnyProxy 提供了多种类型的证书文件,可在下载安装时选择。\n配置iOS/Android系统代理\n\n代理服务器都在wifi设置中配置\n\niOS HTTP代理配置\n\n\n\n\nAndroid HTTP代理配置\n\n\nFAQ\nQ: 为什么https请求不能进入处理函数?\n A: 以下任意一项都能用来改变https的处理特性:\n 1. 命令行启动AnyProxy时配置`--intercept`参数,按npm模块启动时配置`forceProxyHttps`参数,所有Https请求都会被替换证书并解析\n 2. 规则文件内提供`beforeDealHttpsRequest`方法,返回 *true* 的https请求会被解析\nQ: 提示 function is not yieldable\n\nA: 规则模块是用 co 驱动的,函数需要满足yieldable。可以使用generator方法或是返回Promise。\n\nQ: The connection is not private\n当访问特定的HTTPS站点,AnyProxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:\n\n命令行直接启动\n通过启动参数 --ignore-unauthorized-ssl 来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。\nanyproxy -i --ignore-unauthorized-ssl\n\n\n在Nodejs代码中启动\n在构造AnyProxy实例的时候,传入参数dangerouslyIgnoreUnauthorized:true, 如下:\n const options = {\n ...,\n dangerouslyIgnoreUnauthorized: true\n };\n\n const anyproxyIns = new AnyProxy.ProxyCore(options);\n anyproxyIns.start();\n\n通过这种方式初始化的AnyProxy,其配置也是全局性的,所有网站的证书问题都会被忽略\n\n通过自定义的Rule来修改\n我们自然也可以借助自定义的Rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n // 设置属性 rejectUnauthorized 为 false\n newRequestOptions.rejectUnauthorized = false;\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\n\n\n"}}} ================================================ FILE: docs/cn/src_doc.md ================================================ AnyProxy =================== > 本文档的适用范围是AnyProxy 4.0,欢迎提供反馈 Ref: [English Doc](../en) AnyProxy是一个开放式的HTTP代理服务器。 Github主页:https://github.com/alibaba/anyproxy 主要特性包括: * 基于Node.js,开放二次开发能力,允许自定义请求处理逻辑 * 支持Https的解析 * 提供GUI界面,用以观察请求 相比3.x版本,AnyProxy 4.0的主要变化: * 规则文件(Rule)全面支持Promise和Generator * 简化了规则文件内的接口 * Web版界面重构 # 快速开始 ## 作为全局模块 ### 安装 对于Debian或者Ubuntu系统,在安装AnyProxy之前,可能还需要安装 `nodejs-legacy` ```bash sudo apt-get install nodejs-legacy ``` 然后,安装AnyProxy ```bash npm install -g anyproxy ``` ### 启动 * 命令行启动AnyProxy,默认端口号8001 ```bash anyproxy ``` * 启动后将终端http代理服务器配置为127.0.0.1:8001即可 * 访问http://127.0.0.1:8002 ,web界面上能看到所有的请求信息 ### 其他命令 * 配置启动端口,如1080端口启动 ```bash anyproxy --port 1080 ``` ## 作为npm模块使用 AnyProxy可以作为一个npm模块使用,整合进其他工具。 > 如要启用https解析,请在代理服务器启动前自行调用`AnyProxy.utils.certMgr`相关方法生成证书,并引导用户信任安装。或引导用户使用`anyproxy-ca`方法。 * 引入 ```bash npm i anyproxy --save ``` * 使用举例 ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, // 不开启websocket代理 silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * 创建代理服务器 ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} 必选,代理服务器端口 * `rule` {object} 自定义规则模块 * `throttle` {number} 限速值,单位kb/s,默认不限速 * `forceProxyHttps` {boolean} 是否强制拦截所有的https,忽略规则模块的返回,默认`false` * `silent` {boolean} 是否屏蔽所有console输出,默认`false` * `dangerouslyIgnoreUnauthorized` {boolean} 是否忽略请求中的证书错误,默认`false` * `wsIntercept` {boolean} 是否开启websocket代理,默认`false` * `webInterface` {object} web版界面配置 * `enable` {boolean} 是否启用web版界面,默认`false` * `webPort` {number} web版界面端口号,默认`8002` * Event: `ready` * 代理服务器启动完成 * 示例 ```js proxy.on('ready', function() { }) ``` * Event: `error` * 代理服务器发生错误 * 示例 ```js proxy.on('error', function() { }) ``` * Method: `start` * 启动代理服务器 * 示例 ```js proxy.start(); ``` * Method: `close` * 关闭代理服务器 * 示例 ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * 管理系统的全局代理配置,方法调用时可能会弹出密码框 * 使用示例 ```js // 配置127.0.0.1:8001为全局http代理服务器 AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // 关闭全局代理服务器 AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * 管理AnyProxy的证书 * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * 校验系统内是否存在AnyProxy的根证书 * `AnyProxy.utils.certMgr.generateRootCA(callback)` * 生成AnyProxy的rootCA,完成后请引导用户信任.crt文件 * 样例 ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # 代理HTTPS * AnyProxy默认不对https请求做处理,如需看到明文信息,需要配置CA证书 > 解析https请求的原理是中间人攻击(man-in-the-middle),用户必须信任AnyProxy生成的CA证书,才能进行后续流程 * 生成证书并解析所有https请求 ```bash anyproxy-ca #生成rootCA证书,生成后需要手动信任 anyproxy --intercept #启动AnyProxy,并解析所有https请求 ``` * [附录:如何信任CA证书](#证书配置) # 代理WebSocket ```bash anyproxy --ws-intercept ``` > 当启用`HTTPS`代理时,`wss`也会被代理,但是不会被AnyProxy记录。需要开启`--ws-intercept`后才会从界面上看到相应内容。 # rule模块 AnyProxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。 >注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题 规则模块的能力范围包括: * 拦截并修改正在发送的请求 * 可修改内容包括请求头(request header),请求体(request body),甚至是请求的目标地址等 * 拦截并修改服务端响应 * 可修改的内容包括http状态码(status code)、响应头(response header)、响应内容等 * 拦截https请求,对内容做修改 * 本质是中间人攻击(man-in-the-middle attack),需要客户端提前信任AnyProxy生成的CA ### 开发示例 * 举例 * 需要编写一个规则模块,在 GET http://httpbin.org/user-agent 的返回值里加上测试信息,并延迟5秒返回 * Step 1,编写规则 ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, 启动AnyProxy,加载规则 * 运行 `anyproxy --rule sample.js` * Step 3, 测试规则 * 用curl测试 ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * 用浏览器测试:配置浏览器http代理为 127.0.0.1:8001,访问 http://httpbin.org/user-agent * 经过代理服务器后,期望的返回如下 ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, 查看请求信息 * 浏览器访问http://127.0.0.1:8002 ,界面上能看到刚才的请求信息 ### 处理流程 * 处理流程图如下 * 当http请求经过代理服务器时,具体处理过程是: * 收集请求所有请求参数,包括method, header, body等 * AnyProxy调用规则模块`beforeSendRequest`方法,由模块做处理,返回新的请求参数,或返回响应内容 * 如果`beforeSendRequest`返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。 * 根据请求参数,向服务端发出请求,接收服务端响应。 * 调用规则模块`beforeSendResponse`方法,由模块对响应内容进行处理 * 把响应信息返回给客户端 * 当代理服务器收到https请求时,AnyProxy可以替换证书,对请求做明文解析。 * 调用规则模块`beforeDealHttpsRequest`方法,如果返回`true`,会明文解析这个请求,其他请求不处理 * 被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。 ### 如何引用 如下几种方案都可以用来引用规则模块: * 使用本地路径 ```bash anyproxy --rule ./rule.js ``` * 使用在线地址 ```bash anyproxy --rule https://sample.com/rule.js ``` * 使用npm包 * AnyProxy使用`require()`加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包 ```bash anyproxy --rule ./myRulePkg/ #本地包 npm i -g myRulePkg && anyproxy --rule myRulePkg #全局包 ``` # rule接口文档 规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。 ```js module.exports = { // 模块介绍 summary: 'my customized rule for AnyProxy', // 发送请求前拦截处理 *beforeSendRequest(requestDetail) { /* ... */ }, // 发送响应前处理 *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // 是否处理https请求 *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // 请求出错的事件 *onError(requestDetail, error) { /* ... */ }, // https连接服务器出错 *onConnectError(requestDetail, error) { /* ... */ } }; ``` > 规则文件中,除了summary,都是由 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。 ### summary #### summary(): string | summary:string * 规则模块的介绍文案,用于AnyProxy提示用户, 可以是一个函数,也可以是一个普通的字符串 ### beforeSendRequest #### beforeSendRequest(requestDetail) * AnyProxy向服务端发送请求前,会调用`beforeSendRequest`,并带上参数`requestDetail` * `requestDetail` * `protocol` {string} 请求使用的协议,http或者https * `requestOptions` {object} 即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} 请求Body * `url` {string} 请求url * `_req` {object} 请求的原始request * 举例:请求 *anyproxy.io* 时,`requestDetail`参数内容大致如下 ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改请求协议,如强制改用https发起请求 ```js return { protocol: 'https' }; ``` * 修改请求参数 ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * 修改请求body ```js return { requestData: 'my new request data' //这里也可以同时加上requestOptions }; ``` * 直接返回客户端,不再发起请求,其中`statusCode` `header` 是必选字段 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * AnyProxy向客户端发送请求前,会调用`beforeSendResponse`,并带上参数`requestDetail` `responseDetail` * `requestDetail` 同`beforeSendRequest`中的参数 * `responseDetail` * `response` {object} 服务端的返回信息,包括`statusCode` `header` `body`三个字段 * `_res` {object} 原始的服务端返回对象 * 举例,请求 *anyproxy.io* 时,`responseDetail`参数内容大致如下 ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改返回的状态码 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * 修改返回的内容 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * AnyProxy收到https请求时,会调用`beforeDealHttpsRequest`,并带上参数`requestDetail` * 如果配置了全局解析https的参数,则AnyProxy会略过这个调用 * 只有返回`true`时,AnyProxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。 * 注意:https over http的代理模式中,这里的request是CONNECT请求 * `requestDetail` * `host` {string} 请求目标的Host,受制于协议,这里无法获取完整url * `_req` {object} 请求的原始request * 返回值 * `true`或者`false`,表示是否需要AnyProxy替换证书并解析https ### onError #### onError(requestDetail, error) * 在请求处理过程中发生错误时,AnyProxy会调用`onError`方法,并提供对应的错误信息 * 多数场景下,错误会在请求目标服务器的时候发生,比如DNS解析失败、请求超时等 * `requestDetail` 同`beforeSendRequest`中的参数 * 以下几种返回都是合法的 * 不做任何处理。此时AnyProxy会返回一个默认的错误页。 ```js return null; ``` * 返回自定义错误页 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy在与目标HTTPS服务器建立连接的过程中,如果发生错误,AnyProxy会调用这个方法 * `requestDetail` 同`beforeDealHttpsRequest`中的参数 * 此处无法控制向客户端的返回信息,无需返回值。 # rule样例 * 这里提供一些样例,来讲解规则模块的常见用法 * 你可以通过 `anyproxy --rule http://....js` 来加载模块并体验 * 用curl发请求测试的方法如下 * 直接请求服务器:`curl http://httpbin.org/` * 通过代理服务器请求:`curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### 使用本地数据 * 拦截发送到 http://httpbin.org 的请求,使用本地数据代替服务端返回 {{sample-rule:rule_sample/sample_use_local_response.js}} ### 修改请求头 * 修改发送到 httpbin.org 的user-agent {{sample-rule:rule_sample/sample_modify_request_header.js}} ### 修改请求数据 * 修改发送到 http://httpbin.org/post 的post数据 {{sample-rule:rule_sample/sample_modify_request_data.js}} ### 修改请求的目标地址 * 把所有发送到 http://httpbin.org/ 的请求全部改到 http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_request_path.js}} ### 修改请求协议 * 把用http协议请求的 http://httpbin.org 改成https并发送 {{sample-rule:rule_sample/sample_modify_request_protocol.js}} ### 修改返回状态码 * 把 所有http://httpbin.org 的返回状态码都改成404 {{sample-rule:rule_sample/sample_modify_response_statuscode.js}} ### 修改返回头 * 在 http://httpbin.org/user-agent 的返回头里加上 X-Proxy-By:AnyProxy {{sample-rule:rule_sample/sample_modify_response_header.js}} ### 修改返回内容并延迟 * 在 http://httpbin.org/user-agent 的返回最后追加AnyProxy的签名,并延迟5秒 {{sample-rule:rule_sample/sample_modify_response_data.js}} # 证书配置 ### OSX系统信任CA证书 * 类似这种报错都是因为系统没有信任AnyProxy生成的CA所造成的 > 警告:CA证书和系统安全息息相关,建议亲自生成,并妥善保管 安装CA: * 双击打开*rootCA.crt* * 确认将证书添加到login或system * 找到刚刚导入的AnyProxy证书,配置为信任(Always Trust) ### Windows系统信任CA证书 ### 配置OSX系统代理 * 在wifi高级设置中,配置http代理即可 ### 配置浏览器HTTP代理 * 以Chrome的[SwitchyOmega插件](https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### iOS系统信任CA证书 * 点击web ui中的 *Root CA*,按提示扫描二维码即可安装 ### iOS >= 10.3信任CA证书 * 除了上述证书安装过程,还需要在 *设置->通用->关于本机->证书信任设置* 中把AnyProxy证书的开关打开,否则safari将报错。 ### 安卓系统信任CA证书 首先和iOS类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。 * 下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可 * 证书下载到指定目录后,需要从其他入口进行安装,包括: * 设置 -> 安全性与位置信息 -> 加密与凭据 -> 从存储设备安装。找到你下载的证书文件,进行安装 * 设置 -> 安全 -> 从SD卡安装证书。找到你下载的证书文件,进行安装 不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为 .crt 的证书文件,少部分仅支持 .cer 文件(已知如 OPPO R15),AnyProxy 提供了多种类型的证书文件,可在下载安装时选择。 ### 配置iOS/Android系统代理 * 代理服务器都在wifi设置中配置 * iOS HTTP代理配置 * Android HTTP代理配置 # FAQ #### Q: 为什么https请求不能进入处理函数? A: 以下任意一项都能用来改变https的处理特性: 1. 命令行启动AnyProxy时配置`--intercept`参数,按npm模块启动时配置`forceProxyHttps`参数,所有Https请求都会被替换证书并解析 2. 规则文件内提供`beforeDealHttpsRequest`方法,返回 *true* 的https请求会被解析 #### Q: 提示 *function is not yieldable* * A: 规则模块是用 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以使用generator方法或是返回Promise。 #### Q: The connection is not private 当访问特定的HTTPS站点,AnyProxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问: - 命令行直接启动 通过启动参数 `--ignore-unauthorized-ssl` 来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。 ```bash anyproxy -i --ignore-unauthorized-ssl ``` - 在Nodejs代码中启动 在构造AnyProxy实例的时候,传入参数`dangerouslyIgnoreUnauthorized:true`, 如下: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *通过这种方式初始化的AnyProxy,其配置也是全局性的,所有网站的证书问题都会被忽略* - 通过自定义的Rule来修改 我们自然也可以借助自定义的Rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。 ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // 设置属性 rejectUnauthorized 为 false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` ================================================ FILE: docs/en/index.html ================================================ Introduction · AnyProxy

    AnyProxy

    AnyProxy is a fully configurable http/https proxy in NodeJS.

    Ref: 中文文档

    Github:

    Features:

    • Offer you the ablity to handle http traffic by invoking a js module
    • Intercept https
    • GUI webinterface

    Change Logs since 3.x:

    • Support Promise and Generator in rule module
    • Simplified interface in rule module
    • A newly designed web interface

    Getting Start

    install

    To Debian and Ubuntu users, you may need to install nodejs-legacy at the same time

    sudo apt-get install nodejs-legacy
    

    Then install the AnyProxy

    npm install -g anyproxy
    

    launch

    • start AnyProxy in command line, with default port 8001
    anyproxy
    
    • now you can use http proxy server by 127.0.0.1:8001
    • visit http://127.0.0.1:8002 to see the http requests

    options

    • specify the port of http proxy
    anyproxy --port 1080
    

    Use AnyProxy as an npm module

    AnyProxy can be used as an npm module

    To enable https feature, please guide users to use anyproxy-ca in cli. Or use methods under AnyProxy.utils.certMgr to generate certificates.

    • install
    npm i anyproxy --save
    
    • sample
    const AnyProxy = require('anyproxy');
    const options = {
      port: 8001,
      rule: require('myRuleModule'),
      webInterface: {
        enable: true,
        webPort: 8002
      },
      throttle: 10000,
      forceProxyHttps: false,
      wsIntercept: false,
      silent: false
    };
    const proxyServer = new AnyProxy.ProxyServer(options);
    
    proxyServer.on('ready', () => { /* */ });
    proxyServer.on('error', (e) => { /* */ });
    proxyServer.start();
    
    //when finished
    proxyServer.close();
    
    • Class: AnyProxy.proxyServer

      • create a proxy server

        const proxy = new AnyProxy.proxyServer(options)
        
      • options

        • port {number} required, port number of proxy server
        • rule {object} your rule module
        • throttle {number} throttle in kb/s, unlimited for default
        • forceProxyHttps {boolean} in force intercept all https request, default to false
        • silent {boolean} if keep silent in console, false for default false
        • dangerouslyIgnoreUnauthorized {boolean} if ignore certificate error in request, default to false
        • wsIntercept {boolean} whether to intercept websocket, default to false
        • webInterface {object} config for web interface
          • enable {boolean} if enable web interface, default to false
          • webPort {number} port number for web interface
      • Event: ready

        • emit when proxy server is ready
        • sample
        proxy.on('ready', function() { })
        
      • Event: error

        • emit when error happened inside proxy server
        • sample
        proxy.on('error', function() { })
        
      • Method: start

        • start proxy server
        • sample
        proxy.start();
        
      • Method: close

        • close proxy server
        • sample
        proxy.close();
        
    • AnyProxy.utils.systemProxyMgr

      • manage the system proxy config. sudo password may be required
      • sample
      // set 127.0.0.1:8001 as system http server
      AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001');
      
      // disable global proxy server
      AnyProxy.utils.systemProxyMgr.disableGlobalProxy();
      
    • AnyProxy.utils.certMgr

      • Manage certificates of AnyProxy
      • AnyProxy.utils.certMgr.ifRootCAFileExists()
        • detect if AnyProx rootCA exists
      • AnyProxy.utils.certMgr.generateRootCA(callback)
        • generate a rootCA
      • Sample
        const AnyProxy = require('anyproxy');
        const exec = require('child_process').exec;
      
        if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) {
          AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => {
            // let users to trust this CA before using proxy
            if (!error) {
              const certDir = require('path').dirname(keyPath);
              console.log('The cert is generated at', certDir);
              const isWin = /^win/.test(process.platform);
              if (isWin) {
                exec('start .', { cwd: certDir });
              } else {
                exec('open .', { cwd: certDir });
              }
            } else {
              console.error('error when generating rootCA', error);
            }
          });
        }
      

    Proxy Https

    • AnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate.

    Under the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network.

    • generate certifycates and intercept
    anyproxy-ca #generate root CA. manually trust it after that.
    anyproxy --intercept #launch anyproxy and intercept all https traffic
    

    Proxy WebSocket

    anyproxy --ws-intercept
    

    The wss requests will be handled automatically when the HTTPS intercept is turned on, but AnyProxy will not record the data by default. You need to specify the --ws-intercept to tell AnyProxy to record it.

    Rule Introduction

    AnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests.

    Make sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk.

    Rule module could do the following stuff:

    • intercept and modify the request which is being sent
      • editable fields include request header, body, target address
    • intercept and modify the response from server
      • editable fields include response status code, header, body
    • intercept https requests, modify request and response

    sample

    • Target

    • Step 1,Write the rule file, save as sample.js

      // file: sample.js
      module.exports = {
        summary: 'a rule to hack response',
        *beforeSendResponse(requestDetail, responseDetail) {
          if (requestDetail.url === 'http://httpbin.org/user-agent') {
            const newResponse = responseDetail.response;
            newResponse.body += '- AnyProxy Hacked!';
      
            return new Promise((resolve, reject) => {
              setTimeout(() => { // delay
                resolve({ response: newResponse });
              }, 5000);
            });
          }
        },
      };
      
    • Step 2, start AnyProxy and load the rule file

      • run anyproxy --rule sample.js
    • Step 3, test

      • use curl

        curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
        
      • use browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent

      • the expected response from proxy is

      {
        "user-agent": "curl/7.43.0"
      }
      - AnyProxy Hacked!
      
    • Step 4, view the request log

    how does it work

    • The flow chart is as follows

    • When got an http request, the entire process of proxy server is

      • AnyProxy collects all the quest info, include method, header, body
      • AnyProxy calls beforeSendRequest of the rule module. Rule module deal the request, return new request param or response content
      • If beforeSendRequest returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here.
      • Send request to target server, collect response
      • Call beforeSendResponse of the rule module. Rule module deal the response data
      • Send response to client
    • When AnyProxy get https request, it could replace the certificate and decrypt the request data

      • AnyProxy calls beforeDealHttpsRequest of the rule module
      • If the function returns true, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed.

    how to load rule module

    • use local file

      anyproxy --rule ./rule.js
      
    • use an online rule file

      anyproxy --rule https://sample.com/rule.js
      
    • use an npm module

      • AnyProxy uses require() to load rule module. You could either load a local npm module or a global-installed one.
      anyproxy --rule ./myRulePkg/ #local module
      npm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module
      

    Rule module interface

    A typical rule module is as follows. All the functions are optional, just write the part you are interested in.

    module.exports = {
      // introduction
      summary: 'my customized rule for AnyProxy',
      // intercept before send request to server
      *beforeSendRequest(requestDetail) { /* ... */ },
      // deal response before send to client
      *beforeSendResponse(requestDetail, responseDetail) { /* ... */ },
      // if deal https request
      *beforeDealHttpsRequest(requestDetail) { /* ... */ },
      // error happened when dealing requests
      *onError(requestDetail, error) { /* ... */ },
      // error happened when connect to https server
      *onConnectError(requestDetail, error) { /* ... */ }
    };
    

    All functions in your rule file, except summary, are all driven by co . They should be yieldable, i.e. return a promise or be a generator function.

    summary

    summary

    • Introduction of this rule file. AnyProxy will read this field and give some tip to user.

    beforeSendRequest

    beforeSendRequest(requestDetail)

    • Before sending request to server, AnyProxy will call beforeSendRequest with param requestDetail
    • requestDetail
    • e.g. When requesting anyproxy.io, requestDetail is something like the following

      {
        protocol: 'http',
        url: 'http://anyproxy.io/',
        requestOptions: {
          hostname: 'anyproxy.io',
          port: 80,
          path: '/',
          method: 'GET',
          headers: {
            Host: 'anyproxy.io',
            'Proxy-Connection': 'keep-alive',
            'User-Agent': '...'
          }
        },
        requestData: '...',
        _req: { /* ... */}
      }
      
    • Any of these return values are valid

      • do nothing, and return null
      return null;
      
      • modify the request protocol,i.e. force use https
      return {
        protocol: 'https'
      };
      
      • modify request param
      var newOption = Object.assign({}, requestDetail.requestOptions);
      newOption.path = '/redirect/to/another/path';
      return {
        requestOptions: newOption
      };
      
      • modify request body
      return {
        requestData: 'my new request data'
        // requestOptions can also be used here
      };
      
      • give response to the client, not sending request any longer. statusCode headersare required is this situation.
      return {
        response: {
          statusCode: 200,
          header: { 'content-type': 'text/html' },
          body: 'this could be a <string> or <buffer>'
        }
      };
      

    beforeSendResponse

    beforeSendResponse(requestDetail, responseDetail)

    • Before sending response to client, AnyProxy will call beforeSendResponse with param requestDetail responseDetail
    • requestDetail is the same param as in beforeSendRequest
    • responseDetail
      • response {object} the response from server, includes statusCode header body
      • _res {object} the native node.js response object
    • e.g. When requesting anyproxy.io, responseDetail is something like the following

      {
        response: {
          statusCode: 200,
          header: {
            'Content-Type': 'image/gif',
            Connection: 'close',
            'Cache-Control': '...'
          },
          body: '...'
        },
        _res: { /* ... */ }
      }
      
    • Any of these return values are valid

      • do nothing, and return null
      return null;
      
      • modify the response status code
      var newResponse = Object.assign({}, responseDetail.response);
      newResponse.statusCode = 404;
      return {
        response: newResponse
      };
      
      • modify the response content
      var newResponse = Object.assign({}, responseDetail.response);
      newResponse.body += '--from anyproxy--';
      return {
        response: newResponse
      };
      

    beforeDealHttpsRequest

    beforeDealHttpsRequest(requestDetail)

    • When receiving https request, AnyProxy will call beforeDealHttpsRequest with param requestDetail
    • If configed with forceProxyHttps in launching, AnyProxy will skip calling this method
    • Only by returning true, AnyProxy will try to replace the certificate and intercept the https request.
    • requestDetail
      • host {string} the target host to request. Due to the request protocol, full url couldn't be got here
      • _req {object} the native node.js request object. The _req here refers to the CONNECT request.
    • return value
      • true or false, whether AnyProxy should intercept the https request

    onError

    onError(requestDetail, error)

    • AnyProxy will call this method when an error happened in request handling.
    • Errors usually are issued during requesting, e.g. DNS failure, request timeout
    • requestDetail is the same one as in beforeSendRequest
    • Any of these return values are valid

      • do nothing, and AnyProxy will response a default error page
      return null;
      
      • return a customized error page
      return {
        response: {
          statusCode: 200,
          header: { 'content-type': 'text/html' },
          body: 'this could be a <string> or <buffer>'
        }
      };
      

    onConnectError

    onConnectError(requestDetail, error)

    • AnyProxy will call this method when failed to connect target server in https request
    • requestDetail is the same one as in beforeDealHttpsRequest
    • no return value is required

    Rule Samples

    • here are some samples about frequently used rule file
    • try these samples by anyproxy --rule http://....js
    • how to test with curl:
      • request the server directly curl http://httpbin.org/
      • request the server via proxy curl http://httpbin.org/ --proxy http://127.0.0.1:8001

    use local response

    anyproxy --rule rule_sample/sample_use_local_response.js
    
    /* 
      sample: 
        intercept all requests toward httpbin.org, use a local response
      test:
        curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
    */
    module.exports = {
      *beforeSendRequest(requestDetail) {
        const localResponse = {
          statusCode: 200,
          header: { 'Content-Type': 'application/json' },
          body: '{"hello": "this is local response"}'
        };
        if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
          return {
            response: localResponse
          };
        }
      },
    };
    

    modify request header

    • modify the user-agent sent to httpbin.org
    anyproxy --rule rule_sample/sample_modify_request_header.js
    
    /* 
      sample: 
        modify the user-agent in requests toward httpbin.org
      test:
        curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001
    */
    module.exports = {
      *beforeSendRequest(requestDetail) {
        if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
          const newRequestOptions = requestDetail.requestOptions;
          newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0';
          return {
            requestOptions: newRequestOptions
          };
        }
      },
    };
    

    modify request body

    anyproxy --rule rule_sample/sample_modify_request_data.js
    
    /*
      sample:
        modify the post data towards http://httpbin.org/post
      test:
        curl -H "Content-Type: text/plain" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001
      expected response:
        { "data": "i-am-anyproxy-modified-post-data" }
    */
    module.exports = {
      summary: 'Rule to modify request data',
      *beforeSendRequest(requestDetail) {
        if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) {
          return {
            requestData: 'i-am-anyproxy-modified-post-data'
          };
        }
      },
    };
    

    modify the request target

    anyproxy --rule rule_sample/sample_modify_request_path.js
    
    /*
      sample:
        redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html
      test:
        curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001
      expected response:
        'hello world' from 127.0.0.1:8001/index.html
    */
    module.exports = {
      *beforeSendRequest(requestDetail) {
        if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) {
          const newRequestOptions = requestDetail.requestOptions;
          requestDetail.protocol = 'http';
          newRequestOptions.hostname = '127.0.0.1'
          newRequestOptions.port = '8008';
          newRequestOptions.path = '/index.html';
          newRequestOptions.method = 'GET';
          return requestDetail;
        }
      },
      *beforeDealHttpsRequest(requestDetail) {
        return true;
      }
    };
    

    modify request protocol

    anyproxy --rule rule_sample/sample_modify_request_protocol.js
    
    /* 
      sample: 
        redirect all http requests of httpbin.org to https
      test:
        curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001
      expected response:
        { "X-Forwarded-Protocol": "https" }
    */
    module.exports = {
      *beforeSendRequest(requestDetail) {
        if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
          const newOption = requestDetail.requestOptions;
          newOption.port = 443;
          return {
            protocol: 'https',
            requestOptions: newOption
          };
        }
      }
    };
    

    modify response status code

    anyproxy --rule rule_sample/sample_modify_response_statuscode.js
    
    /* 
      sample: 
        modify all status code of http://httpbin.org/ to 404
      test:
        curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
      expected response:
        HTTP/1.1 404 Not Found
    */
    module.exports = {
      *beforeSendResponse(requestDetail, responseDetail) {
        if (requestDetail.url.indexOf('http://httpbin.org') === 0) {
          const newResponse = responseDetail.response;
          newResponse.statusCode = 404;
          return {
            response: newResponse
          };
        }
      }
    };
    

    modify the response header

    anyproxy --rule rule_sample/sample_modify_response_header.js
    
    /* 
      sample: 
        modify response header of http://httpbin.org/user-agent
      test:
        curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
      expected response:
        X-Proxy-By: AnyProxy
    */
    module.exports = {
      *beforeSendResponse(requestDetail, responseDetail) {
        if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) {
          const newResponse = responseDetail.response;
          newResponse.header['X-Proxy-By'] = 'AnyProxy';
          return {
            response: newResponse
          };
        }
      }
    };
    

    modify response data and delay

    anyproxy --rule rule_sample/sample_modify_response_data.js
    
    /* 
      sample: 
        modify response data of http://httpbin.org/user-agent
      test:
        curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001
      expected response:
        { "user-agent": "curl/7.43.0" } -- AnyProxy Hacked! --
    */
    
    module.exports = {
      *beforeSendResponse(requestDetail, responseDetail) {
        if (requestDetail.url === 'http://httpbin.org/user-agent') {
          const newResponse = responseDetail.response;
          newResponse.body += '-- AnyProxy Hacked! --';
          return new Promise((resolve, reject) => {
            setTimeout(() => { // delay the response for 5s
              resolve({ response: newResponse });
            }, 5000);
          });
        }
      },
    };
    

    Config Certification

    Config root CA in OSX

    • this kind of errors is usually caused by untrusted root CA

    Warning: please keep your root CA safe since it may influence your system security.

    install :

    • double click rootCA.crt

    • add cert into login or system

    • find the newly imported AnyProxy certificates, configured as Always Trust

    Config root CA in windows

    Config OSX system proxy

    • the config is in wifi - advanced

    config http proxy server

    trust root CA in iOS

    • Click Root CA in web ui, and follow the instruction to install

    trust root CA in iOS after 10.3

    • Besides installing root CA, you have to "turn on" the certificate for web manually in settings - general - about - Certificate Trust Settings. Otherwire, safari will not trust the root CA generated by AnyProxy.

    trust root CA in Android

    First of all, you need to download the root CA by clicking Root CA in web ui, and then scan the QR code. Installing CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path.

    • The downloaded CA file can be directly installed by clicking, this is the easist way
    • You need to install the CA file from other menu, such as:
      • Settings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install
      • Settings -> Security -> Install from SD card, and find you CA file to install

    There are several file extensions of CA file which may not be compatible with all kinds of Android phones. .crt file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing.

    config iOS/Android proxy server

    • proxy settings are placed in wifi setting

    • iOS

    • Android

    FAQ

    Q: can not deal https request in rule module.

    • A: Any of these options could be used to change the way AnyProxy deall https requests
      1. config --intercept when luanching AnyProxy via cli, or use forceProxyHttps when using as an npm module
      2. place a beforeDealHttpsRequest function in your rule file and determine which request to intercept by your own.

    Q: get an error says function is not yieldable

    • A: Rule module is driven by co. The functions inside should be yieldable, i.e. return a promise or be a generator function.

    Q: The connection is not private

    AnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below.

    • If you run AnyProxy by command line Pass in the option --ignore-unauthorized-ssl to ignore the certification errors, please mind that the option will be active for all connections.

      anyproxy -i --ignore-unauthorized-ssl
      
    • If you run AnyProxy by Nodejs Pass in the option dangerouslyIgnoreUnauthorized:true, like this:

      const options = {
       ...,
       dangerouslyIgnoreUnauthorized: true
      };
      
      const anyproxyIns = new AnyProxy.ProxyCore(options);
      anyproxyIns.start();
      

      This is also a global option, all certification errors will be ignored

    • With the help of AnyProxy Rule You can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs Http.rquest, as we do in AnyProxy. A simple demo below:

      module.exports = {
        *beforeSendRequest(requestDetail) {
          if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) {
            const newRequestOptions = requestDetail.requestOptions;
            // set rejectUnauthorized as false
            newRequestOptions.rejectUnauthorized = false;
            return {
              requestOptions: newRequestOptions
            };
          }
        },
      };
      

      And we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to!

    results matching ""

      No results matching ""

      ================================================ FILE: docs/en/search_index.json ================================================ {"index":{"version":"0.5.12","fields":[{"name":"title","boost":10},{"name":"keywords","boost":15},{"name":"body","boost":1}],"ref":"url","documentStore":{"store":{"./":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"turn","\"user","\"x","#gener","#global","#launch","#local","&","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","+=",",",".",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","/^win/.test(process.platform);","0)","10.3","10000,","1080","127.0.0.1:8001","127.0.0.1:8001,","127.0.0.1:8001/index.html","1,write","2,","200,","3,","3.x:","4,","404","404;","443;","5","5000);","5s","80,","8001","8001,","8002","=","===","=>",">","[switchyomega]","_re","_req","_req:","_res:","a:","abil","abliti","activ","add","address","advanc","advance.","agent","agent\":","agent'","agent')","agent':","agent']","agent,","alive',","all,","alway","android","anyprox","anyproxi","anyproxy',","anyproxy,","anyproxy.","anyproxy.io,","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","append","appendix:how","apt","as:","at',","attack","attack.","automat","base","be","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","below,","below.","below:","besid","bodi","body,","body:","bonou","browser","browser.","by']","by:","by:anyproxi","ca","ca,","ca.","call","card,","caus","cert","certdir","certdir);","certif","certificate.","certificates,","certificates.","certifyc","chang","chart","choos","chrome","class:","cli,","cli.","click","clicking,","client","client,","close","co","co.","code","code,","code.","collect","command","common","compat","config","config.","configur","connect","connection':","connection:","connections.","console,","console.error('error","console.log('th","const","content","content,","control':","couldn't","course.","creat","credenti","curl","curl:","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","dangerouslyignoreunauthorized:true,","data","data\"","data'","data',","deal","dealed.","deall","debian","decrypt","default","default.","defin","delay","demo","design","detect","determin","differ","directli","disabl","dn","doubl","download","driven","due","dure","e.g.","easist","edit","emit","enabl","enable:","encrypt","end","entir","error","error)","error);","errors,","event:","example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例","except","exec","exec('open","exec('start","exist","expect","extens","extent","face","fail","failure,","fals","false,","false;","faq","feature,","features:","few","field","file","file,","file.","file:","find","finish","first","flow","follow","follows.","forc","forceproxyhttp","forceproxyhttps:","forward","found","frequent","full","fulli","function","function()","function.","g","gener","get","github:","give","global","go,","gui","guid","h","hack","hacked!","hacked!';","handl","handling.","happen","header","header,","header:","headers:","headersar","help","here","here,","here.","hood,","host","host:","hostname:","http","http.rquest,","http/1.1","http/http","http://....j","http://127.0.0.1:8001","http://127.0.0.1:8002","http://127.0.0.1:8002,","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://github.com/alibaba/anyproxi","https://httpbin.org/us","https://nodejs.org/api/http.html#http_http_request_options_callback","https://sample.com/rule.j","i.e.","ignor","import","in.","includ","influenc","info","info,","insid","instal","installing.","instruct","intercept","interest","interfac","interface,","introduct","invok","io","ios/android","issu","iswin","it,","it.","javascript.","js","kb/s,","keep","keypath)","kind","know","know.com')","known","launch","launching,","legaci","line","line,","list","load","local","localrespons","locat","log","logic","login","longer.","luanch","make","man","manag","manual","menu","menu,","messag","method","method,","method:","middl","mind","modifi","modul","module,","module.","module.export","myrulepkg","nativ","need","network.","new","newli","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","node.j","nodej","nodejs.","nothing,","now","npm","null","null;","number","object","object.","object.assign({},","offer","on","on\"","on,","onconnecterror","onconnecterror(requestdetail,","one.","onerror","onerror(requestdetail,","onlin","oppo","option","option,","optional,","osx","otherwire,","otherwise,","own.","page","param","part","pass","password","path.","path:","phones.","place","pleas","point","popular,","port","port:","post","privat","process","promis","promise((resolve,","propmt","protocol","protocol\":","protocol,","protocol:","protocol,i.e.","provid","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","qr","quest","r15.","read","readi","receiv","record","redirect","ref:","refer","reject)","rejectunauthor","replac","request","request,","request.","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requesting,","requestopt","requestoptions:","requests,","requests.","requir","require('anyproxy');","require('child_process').exec;","require('http').request","require('myrulemodule'),","require('path').dirname(keypath);","require()","required,","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","right","risk.","root","rootca","rootca',","rootca.crt","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","run","safari","safe","same","sampl","sample.j","sample:","save","scan","scenario,","sd","second","seconds.","secur","security.","see","self","send","sent","server","server,","server.","set","settimeout(()","settings.","sever","side","signed.","silent","silent:","similar","simpl","simplifi","site","site(s)","situation.","skip","someth","source.","specifi","ssl","start","statu","statuscod","statuscode:","step","storage,","stuff:","such","sudo","summari","summary,","summary:","support","sure","system","system,","take","target","tell","test","test:","text","text/plain\"","that.","this:","throttl","throttle:","time","timeout","tip","to!","toward","traffic","tri","true","true,","true;","trust","turn","type","type':","type:","typic","ubuntu","ui,","unauthor","under","unknown","unlimit","unsecur","untrust","url","url:","us","use,","user","user.","users,","usual","valid","valu","var","via","view","visit","want","warning:","way","web","webinterfac","webinterface:","webport","webport:","websocket","websocket,","whether","wifi","window","without","work","world'","write","written","ws","wsintercept","wsintercept:","wss","x","yieldabl","yieldable,","you'r","{","{boolean}","{number}","{object}","{string}","}","})","});","},","};","中文文档",":"]},"length":1},"tokenStore":{"root":{"0":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}},"1":{"0":{"0":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"8":{"0":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}},"docs":{},".":{"3":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"docs":{}}},"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},",":{"docs":{},"访":{"docs":{},"问":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},",":{"docs":{},"编":{"docs":{},"写":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"w":{"docs":{},"r":{"docs":{},"i":{"docs":{},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"2":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"3":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{},"x":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"docs":{}},"4":{"3":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"0":{"docs":{},"的":{"docs":{},"主":{"docs":{},"要":{"docs":{},"变":{"docs":{},"化":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},",":{"docs":{},"欢":{"docs":{},"迎":{"docs":{},"提":{"docs":{},"供":{"docs":{},"反":{"docs":{},"馈":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"docs":{}}},"5":{"0":{"0":{"0":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"docs":{}},"docs":{}},"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"2":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},"\"":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"/":{"7":{"docs":{},".":{"4":{"3":{"docs":{},".":{"0":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"docs":{}}},"docs":{}},"docs":{}}},"docs":{}}}}}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"#":{"docs":{},"全":{"docs":{},"局":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"地":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"生":{"docs":{},"成":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"生":{"docs":{},"成":{"docs":{},"后":{"docs":{},"需":{"docs":{},"要":{"docs":{},"手":{"docs":{},"动":{"docs":{},"信":{"docs":{},"任":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"l":{"docs":{},"a":{"docs":{},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"&":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"&":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"8":{"0":{"0":{"1":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"8":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},".":{"docs":{},".":{"docs":{},".":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"/":{"docs":{},"t":{"docs":{},"o":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"o":{"docs":{},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{},"/":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"a":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"/":{"0":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}}},"docs":{}}},"docs":{}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"l":{"docs":{},"i":{"docs":{},"c":{"docs":{},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"/":{"docs":{},"j":{"docs":{},"s":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"?":{"docs":{},"s":{"docs":{},"h":{"docs":{},"o":{"docs":{},"w":{"docs":{},"_":{"docs":{},"e":{"docs":{},"n":{"docs":{},"v":{"docs":{},"=":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"m":{"docs":{},"a":{"docs":{},"g":{"docs":{},"e":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"f":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"k":{"docs":{},"e":{"docs":{},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"m":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"e":{"docs":{},"x":{"docs":{},"t":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"{":{"docs":{},"\"":{"docs":{},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"(":{"docs":{},"!":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},".":{"docs":{},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"e":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},".":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},"o":{"docs":{},"f":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"/":{"docs":{},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"*":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.007178106774338268}},"}":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"+":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},".":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"/":{"docs":{},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"/":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.00762673844773441}}},"/":{"docs":{"./":{"ref":"./","tf":0.006280843427545985}},"w":{"docs":{},"h":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"这":{"docs":{},"里":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"同":{"docs":{},"时":{"docs":{},"加":{"docs":{},"上":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"^":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},"/":{"docs":{},".":{"docs":{},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},".":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"t":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"m":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"=":{"docs":{"./":{"ref":"./","tf":0.020188425302826378}},"=":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}},">":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},">":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"=":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"关":{"docs":{},"于":{"docs":{},"本":{"docs":{},"机":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"信":{"docs":{},"任":{"docs":{},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"通":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"q":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"l":{"docs":{},"i":{"docs":{},"v":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"w":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.03230148048452221}}},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"c":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"r":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"(":{"docs":{},"(":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"s":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},".":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"使":{"docs":{},"用":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},"加":{"docs":{},"载":{"docs":{},"本":{"docs":{},"地":{"docs":{},"规":{"docs":{},"则":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"在":{"docs":{},"参":{"docs":{},"数":{"docs":{},"里":{"docs":{},"传":{"docs":{},"入":{"docs":{},"一":{"docs":{},"个":{"docs":{},"本":{"docs":{},"地":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{},"路":{"docs":{},"径":{"docs":{},",":{"docs":{},"或":{"docs":{},"是":{"docs":{},"某":{"docs":{},"个":{"docs":{},"全":{"docs":{},"局":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"作":{"docs":{},"为":{"docs":{},"一":{"docs":{},"个":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{},",":{"docs":{},"整":{"docs":{},"合":{"docs":{},"进":{"docs":{},"其":{"docs":{},"他":{"docs":{},"工":{"docs":{},"具":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"在":{"docs":{},"与":{"docs":{},"目":{"docs":{},"标":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"建":{"docs":{},"立":{"docs":{},"连":{"docs":{},"接":{"docs":{},"的":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"这":{"docs":{},"个":{"docs":{},"方":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"了":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"的":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"j":{"docs":{},"s":{"docs":{},"编":{"docs":{},"写":{"docs":{},"自":{"docs":{},"己":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"来":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"网":{"docs":{},"络":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"开":{"docs":{},"放":{"docs":{},"式":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"新":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"或":{"docs":{},"返":{"docs":{},"回":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"对":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"如":{"docs":{},"需":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"配":{"docs":{},"置":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"x":{"docs":{},":":{"docs":{},"h":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"t":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},")":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"提":{"docs":{},"前":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"b":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"l":{"docs":{},"i":{"docs":{},"t":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"c":{"docs":{},"t":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"d":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"a":{"docs":{},"n":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"t":{"docs":{},"o":{"docs":{},"m":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"b":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"d":{"docs":{},"y":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},"三":{"docs":{},"个":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},")":{"docs":{},",":{"docs":{},"甚":{"docs":{},"至":{"docs":{},"是":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},"n":{"docs":{},"o":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"y":{"docs":{},"'":{"docs":{},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"a":{"docs":{},"s":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"o":{"docs":{},"w":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.01031852848811126}},"方":{"docs":{},"法":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},",":{"docs":{},"按":{"docs":{},"提":{"docs":{},"示":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"即":{"docs":{},"可":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}}}},"r":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"f":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"i":{"docs":{},"c":{"docs":{},"a":{"docs":{},"t":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"y":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"l":{"docs":{},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"i":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"d":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"(":{"docs":{},"'":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.008524001794526694}}}},"t":{"docs":{},"r":{"docs":{},"o":{"docs":{},"l":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"f":{"docs":{},"i":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"l":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"l":{"docs":{},"d":{"docs":{},"n":{"docs":{},"'":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"s":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"w":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"h":{"docs":{},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"o":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"d":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"e":{"docs":{},"l":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"e":{"docs":{},"d":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"b":{"docs":{},"i":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"c":{"docs":{},"r":{"docs":{},"y":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"f":{"docs":{},"a":{"docs":{},"u":{"docs":{},"l":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"m":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"m":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"u":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"w":{"docs":{},"n":{"docs":{},"l":{"docs":{},"o":{"docs":{},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}},"i":{"docs":{},"f":{"docs":{},"f":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"l":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"s":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"r":{"docs":{},"i":{"docs":{},"v":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"u":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"g":{"docs":{},"l":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"c":{"docs":{},"r":{"docs":{},"y":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.006729475100942127}},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"v":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"'":{"docs":{},"o":{"docs":{},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"p":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"(":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"c":{"docs":{},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},".":{"docs":{},"g":{"docs":{},"o":{"docs":{},"o":{"docs":{},"g":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"/":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"/":{"docs":{},"p":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"k":{"docs":{},"g":{"docs":{},"c":{"docs":{},"e":{"docs":{},"m":{"docs":{},"l":{"docs":{},"o":{"docs":{},"k":{"docs":{},"b":{"docs":{},"a":{"docs":{},"d":{"docs":{},"o":{"docs":{},"h":{"docs":{},"g":{"docs":{},"k":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"j":{"docs":{},"o":{"docs":{},"m":{"docs":{},"c":{"docs":{},"l":{"docs":{},"g":{"docs":{},"j":{"docs":{},"g":{"docs":{},"i":{"docs":{},"f":{"docs":{},")":{"docs":{},"为":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},".":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"a":{"docs":{},"s":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"d":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"m":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"q":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"u":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.006280843427545985}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"r":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"e":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},"主":{"docs":{},"页":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"a":{"docs":{},"l":{"docs":{},"i":{"docs":{},"b":{"docs":{},"a":{"docs":{},"b":{"docs":{},"a":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"o":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"d":{"docs":{},"!":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"n":{"docs":{},"d":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},")":{"docs":{},",":{"docs":{},"请":{"docs":{},"求":{"docs":{},"体":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"l":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.014804845222072678}},"/":{"1":{"docs":{},".":{"1":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}}},"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}},"2":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"/":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},":":{"8":{"0":{"0":{"8":{"docs":{},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"g":{"docs":{},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"a":{"docs":{},"l":{"docs":{},"i":{"docs":{},"b":{"docs":{},"a":{"docs":{},"b":{"docs":{},"a":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"a":{"docs":{},"p":{"docs":{},"i":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"#":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"_":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"连":{"docs":{},"接":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"出":{"docs":{},"错":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"的":{"docs":{},"代":{"docs":{},"理":{"docs":{},"模":{"docs":{},"式":{"docs":{},"中":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"的":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"是":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},".":{"docs":{},"r":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}},"n":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.008075370121130552}},"l":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"u":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"按":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"启":{"docs":{},"动":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"后":{"docs":{},"才":{"docs":{},"会":{"docs":{},"从":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"看":{"docs":{},"到":{"docs":{},"相":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"f":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"o":{"docs":{},"d":{"docs":{},"u":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":10.001345895020188}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"l":{"docs":{},"u":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}},"f":{"docs":{},"l":{"docs":{},"u":{"docs":{},"e":{"docs":{},"n":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"v":{"docs":{},"o":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},".":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"m":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"n":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"b":{"docs":{},"/":{"docs":{},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"c":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"a":{"docs":{},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"u":{"docs":{},"a":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"n":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"s":{"docs":{},"a":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"d":{"docs":{},"d":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"用":{"docs":{},"户":{"docs":{},"必":{"docs":{},"须":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"才":{"docs":{},"能":{"docs":{},"进":{"docs":{},"行":{"docs":{},"后":{"docs":{},"续":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.012113055181695828}}}}},"u":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"e":{"docs":{},"x":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"a":{"docs":{},"k":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"a":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"u":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"n":{"docs":{},"e":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},".":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},".":{"docs":{},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"s":{"docs":{},"[":{"docs":{},"'":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"e":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"e":{"docs":{},".":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"[":{"docs":{},"'":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}}}}}},"l":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"e":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}},"t":{"docs":{},"w":{"docs":{},"o":{"docs":{},"r":{"docs":{},"k":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"t":{"docs":{},"h":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"p":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},";":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"m":{"docs":{},"b":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"(":{"docs":{},"{":{"docs":{},"}":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}}}}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"p":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"a":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"f":{"docs":{},"f":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{},"w":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}}},"w":{"docs":{},"n":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"g":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"r":{"docs":{},"a":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"w":{"docs":{},"o":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"p":{"docs":{},"u":{"docs":{},"l":{"docs":{},"a":{"docs":{},"r":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"i":{"docs":{},"v":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{},"i":{"docs":{},".":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.016150740242261104}}},"y":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"p":{"docs":{},"m":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"v":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"o":{"docs":{},"n":{"docs":{},"e":{"docs":{},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"l":{"docs":{},"a":{"docs":{},"c":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"e":{"docs":{},"a":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"q":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"1":{"5":{"docs":{},")":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"f":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.02332884701659937}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}},".":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},";":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}},".":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}},"c":{"docs":{},"h":{"docs":{},"i":{"docs":{},"l":{"docs":{},"d":{"docs":{},"_":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"e":{"docs":{},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"'":{"docs":{},")":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},"(":{"docs":{},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},"(":{"docs":{},"{":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.014804845222072678}},"e":{"docs":{},"\"":{"docs":{},"}":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.007178106774338268}}},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},";":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.016150740242261104}}}}}},"c":{"docs":{},"e":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005832211754149843}},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.021085688649618663}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"_":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"/":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"_":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"y":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"接":{"docs":{},"口":{"docs":{},"文":{"docs":{},"档":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"i":{"docs":{},"g":{"docs":{},"h":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"k":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},":":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"v":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"f":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"t":{"docs":{},"i":{"docs":{},"m":{"docs":{},"e":{"docs":{},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{},"(":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"n":{"docs":{},"g":{"docs":{},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"t":{"docs":{},"y":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"f":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"r":{"docs":{},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.008075370121130552}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"(":{"docs":{},"s":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"g":{"docs":{},"n":{"docs":{},"e":{"docs":{},"d":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"m":{"docs":{},"i":{"docs":{},"l":{"docs":{},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{},"r":{"docs":{},"a":{"docs":{},"g":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"f":{"docs":{},"f":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"d":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"p":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"c":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{},"o":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"k":{"docs":{},"i":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"u":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"e":{"docs":{},"c":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}},"x":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"/":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"i":{"docs":{},"n":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"t":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"a":{"docs":{},"t":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"i":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}}}},"!":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"或":{"docs":{},"者":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"表":{"docs":{},"示":{"docs":{},"是":{"docs":{},"否":{"docs":{},"需":{"docs":{},"要":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"a":{"docs":{},"f":{"docs":{},"f":{"docs":{},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"y":{"docs":{},"p":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"a":{"docs":{},"k":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}}},"i":{"docs":{},"m":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"i":{"docs":{},"中":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"k":{"docs":{},"n":{"docs":{},"o":{"docs":{},"w":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"l":{"docs":{},"i":{"docs":{},"m":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"s":{"docs":{},"e":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.009421265141318977}},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"b":{"docs":{},"u":{"docs":{},"n":{"docs":{},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"v":{"docs":{},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"l":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"u":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}},"i":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"e":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"s":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"f":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"重":{"docs":{},"构":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"o":{"docs":{},"r":{"docs":{},"l":{"docs":{},"d":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"a":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"n":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"y":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"h":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"r":{"docs":{},"i":{"docs":{},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}},"o":{"docs":{},"u":{"docs":{},"'":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"{":{"docs":{"./":{"ref":"./","tf":0.03947958725886048}},"b":{"docs":{},"o":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"a":{"docs":{},"n":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}},"n":{"docs":{},"u":{"docs":{},"m":{"docs":{},"b":{"docs":{},"e":{"docs":{},"r":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}},"|":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"}":{"docs":{"./":{"ref":"./","tf":0.013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},";":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}}}},",":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}}},";":{"docs":{"./":{"ref":"./","tf":0.01256168685509197}}}},"下":{"docs":{},"载":{"docs":{},"后":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"可":{"docs":{},"以":{"docs":{},"直":{"docs":{},"接":{"docs":{},"单":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"并":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"是":{"docs":{},"最":{"docs":{},"简":{"docs":{},"单":{"docs":{},"的":{"docs":{},",":{"docs":{},"直":{"docs":{},"接":{"docs":{},"安":{"docs":{},"装":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"不":{"docs":{},"做":{"docs":{},"任":{"docs":{},"何":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{},"此":{"docs":{},"时":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"返":{"docs":{},"回":{"docs":{},"一":{"docs":{},"个":{"docs":{},"默":{"docs":{},"认":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"n":{"docs":{},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}},"同":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},"类":{"docs":{},"型":{"docs":{},"不":{"docs":{},"尽":{"docs":{},"相":{"docs":{},"同":{"docs":{},",":{"docs":{},"大":{"docs":{},"多":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"拓":{"docs":{},"展":{"docs":{},"名":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"中":{"docs":{},"把":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"开":{"docs":{},"关":{"docs":{},"打":{"docs":{},"开":{"docs":{},",":{"docs":{},"否":{"docs":{},"则":{"docs":{},"s":{"docs":{},"a":{"docs":{},"f":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{},"将":{"docs":{},"报":{"docs":{},"错":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"文":{"docs":{},"文":{"docs":{},"档":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"什":{"docs":{},"么":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"能":{"docs":{},"进":{"docs":{},"入":{"docs":{},"处":{"docs":{},"理":{"docs":{},"函":{"docs":{},"数":{"docs":{},"?":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"主":{"docs":{},"要":{"docs":{},"特":{"docs":{},"性":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"从":{"docs":{},"s":{"docs":{},"d":{"docs":{},"卡":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"存":{"docs":{},"储":{"docs":{},"设":{"docs":{},"备":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"启":{"docs":{},"动":{"docs":{},"完":{"docs":{},"成":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"都":{"docs":{},"在":{"docs":{},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"以":{"docs":{},"c":{"docs":{},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"的":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"t":{"docs":{},"c":{"docs":{},"h":{"docs":{},"y":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"插":{"docs":{},"件":{"docs":{},"为":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"下":{"docs":{},"任":{"docs":{},"意":{"docs":{},"一":{"docs":{},"项":{"docs":{},"都":{"docs":{},"能":{"docs":{},"用":{"docs":{},"来":{"docs":{},"改":{"docs":{},"变":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"特":{"docs":{},"性":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"几":{"docs":{},"种":{"docs":{},"返":{"docs":{},"回":{"docs":{},"都":{"docs":{},"是":{"docs":{},"合":{"docs":{},"法":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}}}}}},"作":{"docs":{},"为":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"全":{"docs":{},"局":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"通":{"docs":{},"过":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"使":{"docs":{},"用":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"在":{"docs":{},"线":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"路":{"docs":{},"径":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"修":{"docs":{},"改":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"协":{"docs":{},"议":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"如":{"docs":{},"强":{"docs":{},"制":{"docs":{},"改":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"内":{"docs":{},"容":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"关":{"docs":{},"闭":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"其":{"docs":{},"他":{"docs":{},"命":{"docs":{},"令":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"创":{"docs":{},"建":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"加":{"docs":{},"密":{"docs":{},"与":{"docs":{},"凭":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"即":{"docs":{},"将":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"供":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"作":{"docs":{},"为":{"docs":{},"使":{"docs":{},"用":{"docs":{},"。":{"docs":{},"详":{"docs":{},"见":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"a":{"docs":{},"p":{"docs":{},"i":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"#":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"_":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"原":{"docs":{},"始":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{},"对":{"docs":{},"象":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"双":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"响":{"docs":{},"应":{"docs":{},"前":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"只":{"docs":{},"有":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"才":{"docs":{},"会":{"docs":{},"尝":{"docs":{},"试":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"、":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"。":{"docs":{},"否":{"docs":{},"则":{"docs":{},"只":{"docs":{},"做":{"docs":{},"数":{"docs":{},"据":{"docs":{},"流":{"docs":{},"转":{"docs":{},"发":{"docs":{},",":{"docs":{},"无":{"docs":{},"法":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"数":{"docs":{},"据":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"函":{"docs":{},"数":{"docs":{},",":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"普":{"docs":{},"通":{"docs":{},"的":{"docs":{},"字":{"docs":{},"符":{"docs":{},"串":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"修":{"docs":{},"改":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"请":{"docs":{},"求":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"(":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"同":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"加":{"docs":{},"载":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"后":{"docs":{},"将":{"docs":{},"终":{"docs":{},"端":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}},"命":{"docs":{},"令":{"docs":{},"行":{"docs":{},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"在":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},"代":{"docs":{},"码":{"docs":{},"中":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"高":{"docs":{},"级":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"构":{"docs":{},"造":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"实":{"docs":{},"例":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},",":{"docs":{},"传":{"docs":{},"入":{"docs":{},"参":{"docs":{},"数":{"docs":{},"d":{"docs":{},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"o":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"并":{"docs":{},"提":{"docs":{},"供":{"docs":{},"对":{"docs":{},"应":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"基":{"docs":{},"于":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{},"s":{"docs":{},",":{"docs":{},"开":{"docs":{},"放":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"允":{"docs":{},"许":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"图":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"多":{"docs":{},"数":{"docs":{},"场":{"docs":{},"景":{"docs":{},"下":{"docs":{},",":{"docs":{},"错":{"docs":{},"误":{"docs":{},"会":{"docs":{},"在":{"docs":{},"请":{"docs":{},"求":{"docs":{},"目":{"docs":{},"标":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},"发":{"docs":{},"生":{"docs":{},",":{"docs":{},"比":{"docs":{},"如":{"docs":{},"d":{"docs":{},"n":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},"失":{"docs":{},"败":{"docs":{},"、":{"docs":{},"请":{"docs":{},"求":{"docs":{},"超":{"docs":{},"时":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"如":{"docs":{},"下":{"docs":{},"几":{"docs":{},"种":{"docs":{},"方":{"docs":{},"案":{"docs":{},"都":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"来":{"docs":{},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"何":{"docs":{},"引":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"果":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"返":{"docs":{},"回":{"docs":{},"了":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},",":{"docs":{},"则":{"docs":{},"立":{"docs":{},"即":{"docs":{},"把":{"docs":{},"此":{"docs":{},"响":{"docs":{},"应":{"docs":{},"返":{"docs":{},"回":{"docs":{},"到":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"(":{"docs":{},"而":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{},"真":{"docs":{},"正":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},")":{"docs":{},",":{"docs":{},"流":{"docs":{},"程":{"docs":{},"结":{"docs":{},"束":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{},"了":{"docs":{},"全":{"docs":{},"局":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"则":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"略":{"docs":{},"过":{"docs":{},"这":{"docs":{},"个":{"docs":{},"调":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"要":{"docs":{},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},",":{"docs":{},"请":{"docs":{},"在":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"启":{"docs":{},"动":{"docs":{},"前":{"docs":{},"自":{"docs":{},"行":{"docs":{},"调":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},"相":{"docs":{},"关":{"docs":{},"方":{"docs":{},"法":{"docs":{},"生":{"docs":{},"成":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"并":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"或":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"使":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"安":{"docs":{},"全":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"性":{"docs":{},"与":{"docs":{},"位":{"docs":{},"置":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"c":{"docs":{},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"对":{"docs":{},"于":{"docs":{},"d":{"docs":{},"e":{"docs":{},"b":{"docs":{},"i":{"docs":{},"a":{"docs":{},"n":{"docs":{},"或":{"docs":{},"者":{"docs":{},"u":{"docs":{},"b":{"docs":{},"u":{"docs":{},"n":{"docs":{},"t":{"docs":{},"u":{"docs":{},"系":{"docs":{},"统":{"docs":{},",":{"docs":{},"在":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"之":{"docs":{},"前":{"docs":{},",":{"docs":{},"可":{"docs":{},"能":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"发":{"docs":{},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"入":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"当":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"请":{"docs":{},"求":{"docs":{},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"时":{"docs":{},",":{"docs":{},"具":{"docs":{},"体":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"是":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"可":{"docs":{},"以":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"对":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"代":{"docs":{},"理":{"docs":{},"时":{"docs":{},",":{"docs":{},"w":{"docs":{},"s":{"docs":{},"s":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"不":{"docs":{},"会":{"docs":{},"被":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"记":{"docs":{},"录":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"开":{"docs":{},"启":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"访":{"docs":{},"问":{"docs":{},"特":{"docs":{},"定":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"站":{"docs":{},"点":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"提":{"docs":{},"示":{"docs":{},"该":{"docs":{},"站":{"docs":{},"点":{"docs":{},"不":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"安":{"docs":{},"全":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"这":{"docs":{},"通":{"docs":{},"常":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"设":{"docs":{},"置":{"docs":{},"不":{"docs":{},"能":{"docs":{},"被":{"docs":{},"正":{"docs":{},"确":{"docs":{},"识":{"docs":{},"别":{"docs":{},"导":{"docs":{},"致":{"docs":{},"的":{"docs":{},"(":{"docs":{},"比":{"docs":{},"如":{"docs":{},",":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"是":{"docs":{},"自":{"docs":{},"签":{"docs":{},"发":{"docs":{},"的":{"docs":{},")":{"docs":{},"。":{"docs":{},"如":{"docs":{},"果":{"docs":{},"您":{"docs":{},"信":{"docs":{},"任":{"docs":{},"该":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"以":{"docs":{},"下":{"docs":{},"方":{"docs":{},"式":{"docs":{},"来":{"docs":{},"继":{"docs":{},"续":{"docs":{},"访":{"docs":{},"问":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},",":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"端":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"快":{"docs":{},"速":{"docs":{},"开":{"docs":{},"始":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"我":{"docs":{},"们":{"docs":{},"自":{"docs":{},"然":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"借":{"docs":{},"助":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"实":{"docs":{},"现":{"docs":{},"这":{"docs":{},"个":{"docs":{},"效":{"docs":{},"果":{"docs":{},",":{"docs":{},"而":{"docs":{},"且":{"docs":{},"我":{"docs":{},"们":{"docs":{},"还":{"docs":{},"可":{"docs":{},"以":{"docs":{},"控":{"docs":{},"制":{"docs":{},"到":{"docs":{},"只":{"docs":{},"允":{"docs":{},"许":{"docs":{},"指":{"docs":{},"定":{"docs":{},"网":{"docs":{},"址":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"对":{"docs":{},"不":{"docs":{},"在":{"docs":{},"列":{"docs":{},"表":{"docs":{},"的":{"docs":{},"网":{"docs":{},"址":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"强":{"docs":{},"验":{"docs":{},"证":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"找":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"刚":{"docs":{},"导":{"docs":{},"入":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"docs":{},"信":{"docs":{},"任":{"docs":{},"(":{"docs":{},"a":{"docs":{},"l":{"docs":{},"w":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"把":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"响":{"docs":{},"应":{"docs":{},"信":{"docs":{},"息":{"docs":{},"返":{"docs":{},"回":{"docs":{},"给":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"协":{"docs":{},"议":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"拦":{"docs":{},"截":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"对":{"docs":{},"内":{"docs":{},"容":{"docs":{},"做":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"并":{"docs":{},"修":{"docs":{},"改":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"正":{"docs":{},"在":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"g":{"docs":{},"u":{"docs":{},"i":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"用":{"docs":{},"以":{"docs":{},"观":{"docs":{},"察":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"了":{"docs":{},"多":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"可":{"docs":{},"在":{"docs":{},"下":{"docs":{},"载":{"docs":{},"安":{"docs":{},"装":{"docs":{},"时":{"docs":{},"选":{"docs":{},"择":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"支":{"docs":{},"持":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"收":{"docs":{},"集":{"docs":{},"请":{"docs":{},"求":{"docs":{},"所":{"docs":{},"有":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"改":{"docs":{},"成":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"并":{"docs":{},"发":{"docs":{},"送":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"文":{"docs":{},"件":{"docs":{},"(":{"docs":{},"已":{"docs":{},"知":{"docs":{},"如":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"时":{"docs":{},",":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"否":{"docs":{},"启":{"docs":{},"用":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"屏":{"docs":{},"蔽":{"docs":{},"所":{"docs":{},"有":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"输":{"docs":{},"出":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"强":{"docs":{},"制":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},",":{"docs":{},"忽":{"docs":{},"略":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"请":{"docs":{},"求":{"docs":{},"中":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"文":{"docs":{},"档":{"docs":{},"的":{"docs":{},"适":{"docs":{},"用":{"docs":{},"范":{"docs":{},"围":{"docs":{},"是":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"质":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"来":{"docs":{},"加":{"docs":{},"载":{"docs":{},"模":{"docs":{},"块":{"docs":{},"并":{"docs":{},"体":{"docs":{},"验":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"证":{"docs":{},"书":{"docs":{},"认":{"docs":{},"证":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"注":{"docs":{},"意":{"docs":{},"的":{"docs":{},"是":{"docs":{},",":{"docs":{},"该":{"docs":{},"参":{"docs":{},"数":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"生":{"docs":{},"效":{"docs":{},"的":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"你":{"docs":{},"在":{"docs":{},"此":{"docs":{},"期":{"docs":{},"间":{"docs":{},"访":{"docs":{},"问":{"docs":{},"了":{"docs":{},"其":{"docs":{},"他":{"docs":{},"未":{"docs":{},"知":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"他":{"docs":{},"们":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{},",":{"docs":{},"这":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"带":{"docs":{},"来":{"docs":{},"安":{"docs":{},"全":{"docs":{},"隐":{"docs":{},"患":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"查":{"docs":{},"看":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"校":{"docs":{},"验":{"docs":{},"系":{"docs":{},"统":{"docs":{},"内":{"docs":{},"是":{"docs":{},"否":{"docs":{},"存":{"docs":{},"在":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"根":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"根":{"docs":{},"据":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"向":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"出":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"接":{"docs":{},"收":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"介":{"docs":{},"绍":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"此":{"docs":{},"处":{"docs":{},"无":{"docs":{},"法":{"docs":{},"控":{"docs":{},"制":{"docs":{},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"无":{"docs":{},"需":{"docs":{},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"注":{"docs":{},"意":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"前":{"docs":{},",":{"docs":{},"请":{"docs":{},"务":{"docs":{},"必":{"docs":{},"确":{"docs":{},"保":{"docs":{},"文":{"docs":{},"件":{"docs":{},"来":{"docs":{},"源":{"docs":{},"可":{"docs":{},"靠":{"docs":{},",":{"docs":{},"以":{"docs":{},"免":{"docs":{},"发":{"docs":{},"生":{"docs":{},"安":{"docs":{},"全":{"docs":{},"问":{"docs":{},"题":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}},"点":{"docs":{},"击":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"然":{"docs":{},"后":{"docs":{},",":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"生":{"docs":{},"成":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},",":{"docs":{},"完":{"docs":{},"成":{"docs":{},"后":{"docs":{},"请":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{},"文":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"用":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"发":{"docs":{},"请":{"docs":{},"求":{"docs":{},"测":{"docs":{},"试":{"docs":{},"的":{"docs":{},"方":{"docs":{},"法":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"测":{"docs":{},"试":{"docs":{},":":{"docs":{},"配":{"docs":{},"置":{"docs":{},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"会":{"docs":{},"被":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"少":{"docs":{},"部":{"docs":{},"分":{"docs":{},"仅":{"docs":{},"支":{"docs":{},"持":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"全":{"docs":{},"部":{"docs":{},"改":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"使":{"docs":{},"用":{"docs":{},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{},"代":{"docs":{},"替":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{},"测":{"docs":{},"试":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"docs":{}}}}}}}}}}}}},"头":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"最":{"docs":{},"后":{"docs":{},"追":{"docs":{},"加":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"签":{"docs":{},"名":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}}}}}}}}}}}}}}}}}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"都":{"docs":{},"改":{"docs":{},"成":{"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"请":{"docs":{},"求":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},",":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"中":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}},"相":{"docs":{},"比":{"3":{"docs":{},".":{"docs":{},"x":{"docs":{},"版":{"docs":{},"本":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"docs":{}}},"确":{"docs":{},"认":{"docs":{},"将":{"docs":{},"证":{"docs":{},"书":{"docs":{},"添":{"docs":{},"加":{"docs":{},"到":{"docs":{},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{},"或":{"docs":{},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"简":{"docs":{},"介":{"docs":{"./":{"ref":"./","tf":10}}},"化":{"docs":{},"了":{"docs":{},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"内":{"docs":{},"的":{"docs":{},"接":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"管":{"docs":{},"理":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"系":{"docs":{},"统":{"docs":{},"的":{"docs":{},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"方":{"docs":{},"法":{"docs":{},"调":{"docs":{},"用":{"docs":{},"时":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"弹":{"docs":{},"出":{"docs":{},"密":{"docs":{},"码":{"docs":{},"框":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"类":{"docs":{},"似":{"docs":{},"这":{"docs":{},"种":{"docs":{},"报":{"docs":{},"错":{"docs":{},"都":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"系":{"docs":{},"统":{"docs":{},"没":{"docs":{},"有":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"所":{"docs":{},"造":{"docs":{},"成":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"后":{"docs":{},",":{"docs":{},"期":{"docs":{},"望":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"被":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"后":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{},"同":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"一":{"docs":{},"致":{"docs":{},"。":{"docs":{},"未":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"会":{"docs":{},"再":{"docs":{},"进":{"docs":{},"入":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"中":{"docs":{},",":{"docs":{},"除":{"docs":{},"了":{"docs":{},"s":{"docs":{},"u":{"docs":{},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"y":{"docs":{},",":{"docs":{},"都":{"docs":{},"是":{"docs":{},"由":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"内":{"docs":{},"提":{"docs":{},"供":{"docs":{},"`":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"`":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},"全":{"docs":{},"面":{"docs":{},"支":{"docs":{},"持":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"和":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"应":{"docs":{},"该":{"docs":{},"符":{"docs":{},"合":{"docs":{},"c":{"docs":{},"m":{"docs":{},"d":{"docs":{},"规":{"docs":{},"范":{"docs":{},",":{"docs":{},"一":{"docs":{},"个":{"docs":{},"典":{"docs":{},"型":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"代":{"docs":{},"码":{"docs":{},"结":{"docs":{},"构":{"docs":{},"如":{"docs":{},"下":{"docs":{},"。":{"docs":{},"模":{"docs":{},"块":{"docs":{},"中":{"docs":{},"所":{"docs":{},"有":{"docs":{},"方":{"docs":{},"法":{"docs":{},"都":{"docs":{},"是":{"docs":{},"可":{"docs":{},"选":{"docs":{},"的":{"docs":{},",":{"docs":{},"只":{"docs":{},"需":{"docs":{},"实":{"docs":{},"现":{"docs":{},"业":{"docs":{},"务":{"docs":{},"感":{"docs":{},"兴":{"docs":{},"趣":{"docs":{},"的":{"docs":{},"部":{"docs":{},"分":{"docs":{},"即":{"docs":{},"可":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"介":{"docs":{},"绍":{"docs":{},"文":{"docs":{},"案":{"docs":{},",":{"docs":{},"用":{"docs":{},"于":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"提":{"docs":{},"示":{"docs":{},"用":{"docs":{},"户":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"能":{"docs":{},"力":{"docs":{},"范":{"docs":{},"围":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"原":{"docs":{},"理":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"警":{"docs":{},"告":{"docs":{},":":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},"和":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"全":{"docs":{},"息":{"docs":{},"息":{"docs":{},"相":{"docs":{},"关":{"docs":{},",":{"docs":{},"建":{"docs":{},"议":{"docs":{},"亲":{"docs":{},"自":{"docs":{},"生":{"docs":{},"成":{"docs":{},",":{"docs":{},"并":{"docs":{},"妥":{"docs":{},"善":{"docs":{},"保":{"docs":{},"管":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"属":{"docs":{},"性":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"下":{"docs":{},"载":{"docs":{},"到":{"docs":{},"指":{"docs":{},"定":{"docs":{},"目":{"docs":{},"录":{"docs":{},"后":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"从":{"docs":{},"其":{"docs":{},"他":{"docs":{},"入":{"docs":{},"口":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"使":{"docs":{},"用":{"docs":{},"的":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"或":{"docs":{},"者":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"出":{"docs":{},"错":{"docs":{},"的":{"docs":{},"事":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"的":{"docs":{},"原":{"docs":{},"始":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}},"目":{"docs":{},"标":{"docs":{},"的":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"受":{"docs":{},"制":{"docs":{},"于":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"无":{"docs":{},"法":{"docs":{},"获":{"docs":{},"取":{"docs":{},"完":{"docs":{},"整":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{},"会":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"这":{"docs":{},"个":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"他":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"对":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"进":{"docs":{},"行":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"运":{"docs":{},"行":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"这":{"docs":{},"里":{"docs":{},"提":{"docs":{},"供":{"docs":{},"一":{"docs":{},"些":{"docs":{},"样":{"docs":{},"例":{"docs":{},",":{"docs":{},"来":{"docs":{},"讲":{"docs":{},"解":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"常":{"docs":{},"见":{"docs":{},"用":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"通":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"请":{"docs":{},"求":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"初":{"docs":{},"始":{"docs":{},"化":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"其":{"docs":{},"配":{"docs":{},"置":{"docs":{},"也":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"性":{"docs":{},"的":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"网":{"docs":{},"站":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"为":{"docs":{},"全":{"docs":{},"局":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"x":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"端":{"docs":{},"口":{"docs":{},",":{"docs":{},"如":{"1":{"0":{"8":{"0":{"docs":{},"端":{"docs":{},"口":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"附":{"docs":{},"录":{"docs":{},":":{"docs":{},"如":{"docs":{},"何":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"限":{"docs":{},"速":{"docs":{},"值":{"docs":{},",":{"docs":{},"单":{"docs":{},"位":{"docs":{},"k":{"docs":{},"b":{"docs":{},"/":{"docs":{},"s":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"限":{"docs":{},"速":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"除":{"docs":{},"了":{"docs":{},"上":{"docs":{},"述":{"docs":{},"证":{"docs":{},"书":{"docs":{},"安":{"docs":{},"装":{"docs":{},"过":{"docs":{},"程":{"docs":{},",":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"需":{"docs":{},"要":{"docs":{},"编":{"docs":{},"写":{"docs":{},"一":{"docs":{},"个":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},",":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"首":{"docs":{},"先":{"docs":{},"和":{"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"类":{"docs":{},"似":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"先":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"进":{"docs":{},"行":{"docs":{},"下":{"docs":{},"载":{"docs":{},"。":{"docs":{},"然":{"docs":{},"后":{"docs":{},"不":{"docs":{},"同":{"docs":{},"的":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"方":{"docs":{},"式":{"docs":{},"可":{"docs":{},"能":{"docs":{},"有":{"docs":{},"所":{"docs":{},"不":{"docs":{},"同":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"步":{"docs":{},"骤":{"docs":{},"是":{"docs":{},"类":{"docs":{},"似":{"docs":{},"的":{"docs":{},",":{"docs":{},"我":{"docs":{},"们":{"docs":{},"列":{"docs":{},"举":{"docs":{},"了":{"docs":{},"几":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"驱":{"docs":{},"动":{"docs":{},"的":{"docs":{},",":{"docs":{},"函":{"docs":{},"数":{"docs":{},"需":{"docs":{},"要":{"docs":{},"满":{"docs":{},"足":{"docs":{},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"。":{"docs":{},"可":{"docs":{},"以":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},"或":{"docs":{},"是":{"docs":{},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"或":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"函":{"docs":{},"数":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"才":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"[":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"t":{"docs":{},"c":{"docs":{},"h":{"docs":{},"y":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"j":{"docs":{},"a":{"docs":{},"v":{"docs":{},"a":{"docs":{},"s":{"docs":{},"c":{"docs":{},"r":{"docs":{},"i":{"docs":{},"p":{"docs":{},"t":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"length":1376},"corpusTokens":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"turn","\"user","\"x","#gener","#global","#launch","#local","#全局包","#启动anyproxy,并解析所有https请求","#本地包","#生成rootca证书,生成后需要手动信任","&","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","*true*","+=",",",".",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","//这里也可以同时加上requestopt","/^win/.test(process.platform);","0)","1.","10.3","10.3信任ca证书","10000,","1080","127.0.0.1:8001","127.0.0.1:8001,","127.0.0.1:8001/index.html","127.0.0.1:8001,访问","1,write","1,编写规则","2,","2.","200,","3,","3.x:","4,","4.0的主要变化:","4.0,欢迎提供反馈","404","404;","443;","5","5000);","5s","80,","8001","8001,","8002","=","===","=>",">",">=",">关于本机",">证书信任设置",">通用","[switchyomega]","_re","_req","_req:","_res:","a:","abil","abliti","activ","add","address","advanc","advance.","agent","agent\":","agent'","agent')","agent':","agent']","agent,","alive',","all,","alway","android","anyprox","anyproxi","anyproxy',","anyproxy,","anyproxy.","anyproxy.io","anyproxy.io,","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","anyproxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包","anyproxy可以作为一个npm模块使用,整合进其他工具。","anyproxy向客户端发送请求前,会调用beforesendresponse,并带上参数requestdetail","anyproxy向服务端发送请求前,会调用beforesendrequest,并带上参数requestdetail","anyproxy在与目标https服务器建立连接的过程中,如果发生错误,anyproxy会调用这个方法","anyproxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。","anyproxy收到https请求时,会调用beforedealhttpsrequest,并带上参数requestdetail","anyproxy是一个开放式的http代理服务器。","anyproxy调用规则模块beforesendrequest方法,由模块做处理,返回新的请求参数,或返回响应内容","anyproxy默认不对https请求做处理,如需看到明文信息,需要配置ca证书","append","appendix:how","apt","as:","at',","attack","attack.","attack),需要客户端提前信任anyproxy生成的ca","automat","base","be","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","below,","below.","below:","besid","bodi","body,","body:","body三个字段","body等","body),甚至是请求的目标地址等","bonou","browser","browser.","by']","by:","by:anyproxi","ca","ca,","ca.","call","card,","caus","ca方法。","ca,按提示扫描二维码即可安装","cert","certdir","certdir);","certif","certificate.","certificates,","certificates.","certifyc","chang","chart","choos","chrome","class:","cli,","cli.","click","clicking,","client","client,","close","co","co.","code","code)、响应头(respons","code,","code.","collect","command","common","compat","config","config.","configur","connect","connection':","connection:","connections.","console,","console.error('error","console.log('th","const","content","content,","control':","couldn't","course.","creat","credenti","curl","curl:","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","dangerouslyignoreunauthorized:true,","data","data\"","data'","data',","deal","dealed.","deall","debian","decrypt","default","default.","defin","delay","demo","design","detect","determin","differ","directli","disabl","dn","doc","doubl","download","driven","due","dure","e.g.","easist","edit","emit","enabl","enable:","encrypt","end","english","entir","error","error)","error);","errors,","event:","example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例","except","exec","exec('open","exec('start","exist","expect","extens","extent","face","fail","failure,","fals","false,","false;","faq","feature,","features:","few","field","file","file,","file.","file:","find","finish","first","flow","follow","follows.","forc","forceproxyhttp","forceproxyhttps:","forward","found","frequent","full","fulli","function","function()","function.","g","gener","get","github:","github主页:https://github.com/alibaba/anyproxi","give","global","go,","gui","guid","h","hack","hacked!","hacked!';","handl","handling.","happen","header","header),请求体(request","header,","header:","headers:","headersar","header)、响应内容等","help","here","here,","here.","hood,","host","host:","hostname:","http","http.rquest,","http/1.1","http/http","http://....j","http://127.0.0.1:8001","http://127.0.0.1:8002","http://127.0.0.1:8002,","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://github.com/alibaba/anyproxi","https://httpbin.org/us","https://nodejs.org/api/http.html#http_http_request_options_callback","https://sample.com/rule.j","https连接服务器出错","http代理配置","http的代理模式中,这里的request是connect请求","i.e.","ignor","import","in.","includ","influenc","info","info,","insid","instal","installing.","instruct","intercept","intercept`参数,按npm模块启动时配置`forceproxyhttps`参数,所有https请求都会被替换证书并解析","intercept后才会从界面上看到相应内容。","interest","interfac","interface,","introduct","invok","io","ios/android","ios系统信任ca证书","issu","iswin","it,","it.","javascript.","js","kb/s,","keep","keypath)","kind","know","know.com')","known","launch","launching,","legaci","line","line,","list","load","local","localrespons","locat","log","logic","login","longer.","luanch","make","man","manag","manual","menu","menu,","messag","method","method,","method:","middl","middle),用户必须信任anyproxy生成的ca证书,才能进行后续流程","mind","modifi","modul","module,","module.","module.export","myrulepkg","nativ","need","network.","new","newli","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","node.j","nodej","nodejs.","nothing,","now","npm","null","null;","number","object","object.","object.assign({},","offer","on","on\"","on,","onconnecterror","onconnecterror(requestdetail,","one.","onerror","onerror(requestdetail,","onlin","oppo","option","option,","optional,","osx","osx系统信任ca证书","otherwire,","otherwise,","over","own.","page","param","part","pass","password","path.","path:","phones.","place","pleas","point","popular,","port","port:","post","privat","process","promis","promise((resolve,","propmt","protocol","protocol\":","protocol,","protocol:","protocol,i.e.","provid","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","qr","quest","r15.","r15),anyproxi","read","readi","receiv","record","redirect","ref:","refer","reject)","rejectunauthor","replac","request","request,","request.","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requesting,","requestopt","requestoptions:","requests,","requests.","requir","require('anyproxy');","require('child_process').exec;","require('http').request","require('myrulemodule'),","require('path').dirname(keypath);","require()","required,","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","right","risk.","root","rootca","rootca',","rootca.crt","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","rule接口文档","rule样例","rule模块","run","safari","safe","same","sampl","sample.j","sample:","save","scan","scenario,","sd","second","seconds.","secur","security.","see","self","send","sent","server","server,","server.","set","settimeout(()","settings.","sever","side","signed.","silent","silent:","similar","simpl","simplifi","site","site(s)","situation.","skip","someth","source.","specifi","ssl","start","statu","statuscod","statuscode:","step","storage,","string","stuff:","such","sudo","summari","summary():","summary,","summary:","summary:str","support","sure","system","system,","take","target","tell","test","test:","text","text/plain\"","that.","this:","throttl","throttle:","time","timeout","tip","to!","toward","traffic","tri","true","true,","true;","true或者false,表示是否需要anyproxy替换证书并解析http","trust","trust)","turn","type","type':","type:","typic","ubuntu","ui,","ui中的","unauthor","under","unknown","unlimit","unsecur","untrust","url","url:","us","use,","user","user.","users,","usual","valid","valu","var","via","view","visit","want","warning:","way","web","webinterfac","webinterface:","webport","webport:","websocket","websocket,","web版界面端口号,默认8002","web版界面配置","web版界面重构","whether","wifi","window","windows系统信任ca证书","without","work","world'","write","written","ws","wsintercept","wsintercept:","wss","x","yieldabl","yieldable,","you'r","{","{boolean}","{number}","{object}","{string}","|","}","})","});","},","};","下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可","不做任何处理。此时anyproxy会返回一个默认的错误页。","不做任何处理,返回null","不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为","不开启websocket代理","中把anyproxy证书的开关打开,否则safari将报错。","中文文档","为","为什么https请求不能进入处理函数?","主要特性包括:","举例","举例,请求","举例:请求","从sd卡安装证书。找到你下载的证书文件,进行安装","从存储设备安装。找到你下载的证书文件,进行安装","代理http","代理websocket","代理服务器发生错误","代理服务器启动完成","代理服务器都在wifi设置中配置","以chrome的switchyomega插件为例","以下任意一项都能用来改变https的处理特性:","以下几种返回都是合法的","作为npm模块使用","作为全局模块","你可以通过","使用npm包","使用举例","使用在线地址","使用本地数据","使用本地路径anyproxi","使用示例","修改发送到","修改请求bodi","修改请求协议","修改请求协议,如强制改用https发起请求","修改请求参数","修改请求头","修改请求数据","修改请求的目标地址","修改返回内容并延迟","修改返回头","修改返回状态码","修改返回的内容","修改返回的状态码","关闭代理服务器","关闭全局代理服务器","其他命令","创建代理服务器","加密与凭据","即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback","原始的服务端返回对象","双击打开rootca.crt","发送响应前处理","发送请求前拦截处理","只有返回true时,anyproxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。","可以是一个函数,也可以是一个普通的字符串","可修改内容包括请求头(request","可修改的内容包括http状态码(statu","同beforedealhttpsrequest中的参数","同beforesendrequest中的参数","启动","启动anyproxy,加载规则","启动代理服务器","启动后将终端http代理服务器配置为127.0.0.1:8001即可","命令行启动anyproxy时配置`","命令行启动anyproxy,默认端口号8001","命令行直接启动","在","在nodejs代码中启动","在wifi高级设置中,配置http代理即可","在构造anyproxy实例的时候,传入参数dangerouslyignoreunauthorized:true,","在请求处理过程中发生错误时,anyproxy会调用onerror方法,并提供对应的错误信息","基于node.js,开放二次开发能力,允许自定义请求处理逻辑","处理流程","处理流程图如下","多数场景下,错误会在请求目标服务器的时候发生,比如dns解析失败、请求超时等","如下几种方案都可以用来引用规则模块:","如下:","如何引用","如果beforesendrequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。","如果配置了全局解析https的参数,则anyproxy会略过这个调用","如要启用https解析,请在代理服务器启动前自行调用anyproxy.utils.certmgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxi","安全","安全性与位置信息","安卓系统信任ca证书","安装","安装ca:","对于debian或者ubuntu系统,在安装anyproxy之前,可能还需要安装","开发示例","引入","当http请求经过代理服务器时,具体处理过程是:","当代理服务器收到https请求时,anyproxy可以替换证书,对请求做明文解析。","当启用https代理时,wss也会被代理,但是不会被anyproxy记录。需要开启","当访问特定的https站点,anyproxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:","必选,代理服务器端口","快速开始","我们自然也可以借助自定义的rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。","所有http://httpbin.org","找到刚刚导入的anyproxy证书,配置为信任(alway","把","把响应信息返回给客户端","把所有发送到","把用http协议请求的","拦截https请求,对内容做修改","拦截发送到","拦截并修改服务端响应","拦截并修改正在发送的请求","提供gui界面,用以观察请求","提供了多种类型的证书文件,可在下载安装时选择。","提示","支持https的解析","收集请求所有请求参数,包括method,","改成https并发送","文件(已知如","时,requestdetail参数内容大致如下","时,responsedetail参数内容大致如下","是否启用web版界面,默认fals","是否处理https请求","是否屏蔽所有console输出,默认fals","是否开启websocket代理,默认fals","是否强制拦截所有的https,忽略规则模块的返回,默认fals","是否忽略请求中的证书错误,默认fals","是必选字段","服务端的返回信息,包括statuscod","本文档的适用范围是anyproxi","本质是中间人攻击(man","来加载模块并体验","来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。","查看请求信息","校验系统内是否存在anyproxy的根证书","样例","根据请求参数,向服务端发出请求,接收服务端响应。","模块介绍","此处无法控制向客户端的返回信息,无需返回值。","注意:http","注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题","测试规则","浏览器访问http://127.0.0.1:8002","点击web","然后,安装anyproxi","生成anyproxy的rootca,完成后请引导用户信任.crt文件","生成证书并解析所有https请求","用curl发请求测试的方法如下","用curl测试","用浏览器测试:配置浏览器http代理为","的https请求会被解析","的post数据","的user","的证书文件,少部分仅支持","的请求全部改到","的请求,使用本地数据代替服务端返回","的返回值里加上测试信息,并延迟5秒返回","的返回头里加上","的返回最后追加anyproxy的签名,并延迟5秒","的返回状态码都改成404","直接请求服务器:curl","直接返回客户端,不再发起请求,其中statuscod","相比3.x版本,anyproxi","确认将证书添加到login或system","示例","简介","简化了规则文件内的接口","管理anyproxy的证书","管理系统的全局代理配置,方法调用时可能会弹出密码框","类似这种报错都是因为系统没有信任anyproxy生成的ca所造成的","经过代理服务器后,期望的返回如下","自定义规则模块","被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。","规则文件中,除了summary,都是由","规则文件内提供`beforedealhttpsrequest`方法,返回","规则文件(rule)全面支持promise和gener","规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。","规则模块是用","规则模块的介绍文案,用于anyproxy提示用户,","规则模块的能力范围包括:","解析https请求的原理是中间人攻击(man","警告:ca证书和系统安全息息相关,建议亲自生成,并妥善保管","设置","设置属性","访问http://127.0.0.1:8002","证书下载到指定目录后,需要从其他入口进行安装,包括:","证书配置","请求bodi","请求url","请求使用的协议,http或者http","请求出错的事件","请求的原始request","请求目标的host,受制于协议,这里无法获取完整url","调用规则模块beforedealhttpsrequest方法,如果返回true,会明文解析这个请求,其他请求不处理","调用规则模块beforesendresponse方法,由模块对响应内容进行处理","运行","返回值","返回自定义错误页","这里提供一些样例,来讲解规则模块的常见用法","通过代理服务器请求:curl","通过启动参数","通过自定义的rule来修改","通过这种方式初始化的anyproxy,其配置也是全局性的,所有网站的证书问题都会被忽略","配置127.0.0.1:8001为全局http代理服务器","配置ios/android系统代理","配置osx系统代理","配置启动端口,如1080端口启动","配置浏览器http代理","附录:如何信任ca证书","限速值,单位kb/s,默认不限速","除了上述证书安装过程,还需要在","需要编写一个规则模块,在","首先和ios类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。","驱动的,函数需要满足yieldable。可以使用generator方法或是返回promise。","驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。",",web界面上能看到所有的请求信息",",界面上能看到刚才的请求信息",":"],"pipeline":["stopWordFilter","stemmer"]},"store":{"./":{"url":"./","title":"Introduction","keywords":"","body":"AnyProxy\nAnyProxy is a fully configurable http/https proxy in NodeJS.\nRef: 中文文档\nGithub:\n\nhttps://github.com/alibaba/anyproxy\n\nFeatures:\n\nOffer you the ablity to handle http traffic by invoking a js module\nIntercept https\nGUI webinterface\n\nChange Logs since 3.x:\n\nSupport Promise and Generator in rule module\nSimplified interface in rule module\nA newly designed web interface\n\n\nGetting Start\ninstall\nTo Debian and Ubuntu users, you may need to install nodejs-legacy at the same time\nsudo apt-get install nodejs-legacy\n\nThen install the AnyProxy\nnpm install -g anyproxy\n\nlaunch\n\nstart AnyProxy in command line, with default port 8001\n\nanyproxy\n\n\nnow you can use http proxy server by 127.0.0.1:8001\nvisit http://127.0.0.1:8002 to see the http requests\n\noptions\n\nspecify the port of http proxy\n\nanyproxy --port 1080\n\nUse AnyProxy as an npm module\nAnyProxy can be used as an npm module\n\nTo enable https feature, please guide users to use anyproxy-ca in cli. Or use methods under AnyProxy.utils.certMgr to generate certificates.\n\n\ninstall\n\nnpm i anyproxy --save\n\n\nsample\n\nconst AnyProxy = require('anyproxy');\nconst options = {\n port: 8001,\n rule: require('myRuleModule'),\n webInterface: {\n enable: true,\n webPort: 8002\n },\n throttle: 10000,\n forceProxyHttps: false,\n wsIntercept: false,\n silent: false\n};\nconst proxyServer = new AnyProxy.ProxyServer(options);\n\nproxyServer.on('ready', () => { /* */ });\nproxyServer.on('error', (e) => { /* */ });\nproxyServer.start();\n\n//when finished\nproxyServer.close();\n\n\nClass: AnyProxy.proxyServer\n\ncreate a proxy server\nconst proxy = new AnyProxy.proxyServer(options)\n\n\noptions\n\nport {number} required, port number of proxy server\nrule {object} your rule module\nthrottle {number} throttle in kb/s, unlimited for default\nforceProxyHttps {boolean} in force intercept all https request, default to false\nsilent {boolean} if keep silent in console, false for default false\ndangerouslyIgnoreUnauthorized {boolean} if ignore certificate error in request, default to false\nwsIntercept {boolean} whether to intercept websocket, default to false\nwebInterface {object} config for web interface\nenable {boolean} if enable web interface, default to false\nwebPort {number} port number for web interface\n\n\n\n\nEvent: ready\n\nemit when proxy server is ready\nsample\n\nproxy.on('ready', function() { })\n\n\nEvent: error\n\nemit when error happened inside proxy server\nsample\n\nproxy.on('error', function() { })\n\n\nMethod: start\n\nstart proxy server\nsample\n\nproxy.start();\n\n\nMethod: close\n\nclose proxy server\nsample\n\nproxy.close();\n\n\n\n\nAnyProxy.utils.systemProxyMgr\n\nmanage the system proxy config. sudo password may be required\nsample\n\n// set 127.0.0.1:8001 as system http server\nAnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001');\n\n// disable global proxy server\nAnyProxy.utils.systemProxyMgr.disableGlobalProxy();\n\n\nAnyProxy.utils.certMgr\n\nManage certificates of AnyProxy\nAnyProxy.utils.certMgr.ifRootCAFileExists()\ndetect if AnyProx rootCA exists\n\n\nAnyProxy.utils.certMgr.generateRootCA(callback)\ngenerate a rootCA\n\n\nSample\n\n const AnyProxy = require('anyproxy');\n const exec = require('child_process').exec;\n\n if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) {\n AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => {\n // let users to trust this CA before using proxy\n if (!error) {\n const certDir = require('path').dirname(keyPath);\n console.log('The cert is generated at', certDir);\n const isWin = /^win/.test(process.platform);\n if (isWin) {\n exec('start .', { cwd: certDir });\n } else {\n exec('open .', { cwd: certDir });\n }\n } else {\n console.error('error when generating rootCA', error);\n }\n });\n }\n\n\n\nProxy Https\n\nAnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate.\n\n\nUnder the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network.\n\n\ngenerate certifycates and intercept\n\nanyproxy-ca #generate root CA. manually trust it after that.\nanyproxy --intercept #launch anyproxy and intercept all https traffic\n\n\nAppendix:how to trust CA\n\nProxy WebSocket\nanyproxy --ws-intercept\n\n\nThe wss requests will be handled automatically when the HTTPS intercept is turned on, but AnyProxy will not record the data by default. You need to specify the --ws-intercept to tell AnyProxy to record it.\n\nRule Introduction\nAnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests.\n\nMake sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk.\n\nRule module could do the following stuff:\n\nintercept and modify the request which is being sent\neditable fields include request header, body, target address\n\n\nintercept and modify the response from server\neditable fields include response status code, header, body\n\n\nintercept https requests, modify request and response\n\nsample\n\nTarget\n\nwrite a rule module to append some text to the response of GET http://httpbin.org/user-agent, and delay the response for 5 seconds\n\n\nStep 1,Write the rule file, save as sample.js\n// file: sample.js\nmodule.exports = {\n summary: 'a rule to hack response',\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '- AnyProxy Hacked!';\n\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\n\nStep 2, start AnyProxy and load the rule file\n\nrun anyproxy --rule sample.js\n\n\nStep 3, test\n\nuse curl\ncurl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n\n\nuse browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent\n\nthe expected response from proxy is\n\n\n{\n \"user-agent\": \"curl/7.43.0\"\n}\n- AnyProxy Hacked!\n\nStep 4, view the request log\n\nvisit http://127.0.0.1:8002, the request just sent should be listed here\n\n\n\nhow does it work\n\nThe flow chart is as follows\n\n\n\nWhen got an http request, the entire process of proxy server is\n\nAnyProxy collects all the quest info, include method, header, body\nAnyProxy calls beforeSendRequest of the rule module. Rule module deal the request, return new request param or response content\nIf beforeSendRequest returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here.\nSend request to target server, collect response\nCall beforeSendResponse of the rule module. Rule module deal the response data\nSend response to client\n\n\nWhen AnyProxy get https request, it could replace the certificate and decrypt the request data\n\nAnyProxy calls beforeDealHttpsRequest of the rule module\nIf the function returns true, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed.\n\n\n\nhow to load rule module\n\nuse local file\nanyproxy --rule ./rule.js\n\n\nuse an online rule file\nanyproxy --rule https://sample.com/rule.js\n\n\nuse an npm module\n\nAnyProxy uses require() to load rule module. You could either load a local npm module or a global-installed one.\n\nanyproxy --rule ./myRulePkg/ #local module\nnpm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module\n\n\n\nRule module interface\nA typical rule module is as follows. All the functions are optional, just write the part you are interested in.\nmodule.exports = {\n // introduction\n summary: 'my customized rule for AnyProxy',\n // intercept before send request to server\n *beforeSendRequest(requestDetail) { /* ... */ },\n // deal response before send to client\n *beforeSendResponse(requestDetail, responseDetail) { /* ... */ },\n // if deal https request\n *beforeDealHttpsRequest(requestDetail) { /* ... */ },\n // error happened when dealing requests\n *onError(requestDetail, error) { /* ... */ },\n // error happened when connect to https server\n *onConnectError(requestDetail, error) { /* ... */ }\n};\n\n\nAll functions in your rule file, except summary, are all driven by co . They should be yieldable, i.e. return a promise or be a generator function.\n\nsummary\nsummary\n\nIntroduction of this rule file. AnyProxy will read this field and give some tip to user.\n\nbeforeSendRequest\nbeforeSendRequest(requestDetail)\n\nBefore sending request to server, AnyProxy will call beforeSendRequest with param requestDetail\nrequestDetail\nprotocol {string} the protocol to use, http or https\nrequestOptions {object} the options of the request-to-go, a param of require('http').request . ref: https://nodejs.org/api/http.html#http_http_request_options_callback\nrequestData {object} request body\nurl {string} request url\n_req {object} the native node.js request object\n\n\ne.g. When requesting anyproxy.io, requestDetail is something like the following\n{\n protocol: 'http',\n url: 'http://anyproxy.io/',\n requestOptions: {\n hostname: 'anyproxy.io',\n port: 80,\n path: '/',\n method: 'GET',\n headers: {\n Host: 'anyproxy.io',\n 'Proxy-Connection': 'keep-alive',\n 'User-Agent': '...'\n }\n },\n requestData: '...',\n _req: { /* ... */}\n}\n\n\nAny of these return values are valid\n\ndo nothing, and return null\n\nreturn null;\n\n\nmodify the request protocol,i.e. force use https\n\nreturn {\n protocol: 'https'\n};\n\n\nmodify request param\n\nvar newOption = Object.assign({}, requestDetail.requestOptions);\nnewOption.path = '/redirect/to/another/path';\nreturn {\n requestOptions: newOption\n};\n\n\nmodify request body\n\nreturn {\n requestData: 'my new request data'\n // requestOptions can also be used here\n};\n\n\ngive response to the client, not sending request any longer. statusCode headersare required is this situation.\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nbeforeSendResponse\nbeforeSendResponse(requestDetail, responseDetail)\n\nBefore sending response to client, AnyProxy will call beforeSendResponse with param requestDetail responseDetail\nrequestDetail is the same param as in beforeSendRequest\nresponseDetail\nresponse {object} the response from server, includes statusCode header body\n_res {object} the native node.js response object\n\n\ne.g. When requesting anyproxy.io, responseDetail is something like the following\n{\n response: {\n statusCode: 200,\n header: {\n 'Content-Type': 'image/gif',\n Connection: 'close',\n 'Cache-Control': '...'\n },\n body: '...'\n },\n _res: { /* ... */ }\n}\n\n\nAny of these return values are valid\n\ndo nothing, and return null\n\nreturn null;\n\n\nmodify the response status code\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.statusCode = 404;\nreturn {\n response: newResponse\n};\n\n\nmodify the response content\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.body += '--from anyproxy--';\nreturn {\n response: newResponse\n};\n\n\n\nbeforeDealHttpsRequest\nbeforeDealHttpsRequest(requestDetail)\n\nWhen receiving https request, AnyProxy will call beforeDealHttpsRequest with param requestDetail\nIf configed with forceProxyHttps in launching, AnyProxy will skip calling this method\nOnly by returning true, AnyProxy will try to replace the certificate and intercept the https request.\nrequestDetail\nhost {string} the target host to request. Due to the request protocol, full url couldn't be got here\n_req {object} the native node.js request object. The _req here refers to the CONNECT request.\n\n\nreturn value\ntrue or false, whether AnyProxy should intercept the https request\n\n\n\nonError\nonError(requestDetail, error)\n\nAnyProxy will call this method when an error happened in request handling.\nErrors usually are issued during requesting, e.g. DNS failure, request timeout\nrequestDetail is the same one as in beforeSendRequest\nAny of these return values are valid\n\ndo nothing, and AnyProxy will response a default error page\n\nreturn null;\n\n\nreturn a customized error page\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nonConnectError\nonConnectError(requestDetail, error)\n\nAnyProxy will call this method when failed to connect target server in https request\nrequestDetail is the same one as in beforeDealHttpsRequest\nno return value is required\n\nRule Samples\n\nhere are some samples about frequently used rule file\ntry these samples by anyproxy --rule http://....js\nhow to test with curl:\nrequest the server directly curl http://httpbin.org/\nrequest the server via proxy curl http://httpbin.org/ --proxy http://127.0.0.1:8001\n\n\n\nuse local response\n\nintercept the request towards http://httpbin.org , return the local-defined response\n\nanyproxy --rule rule_sample/sample_use_local_response.js\n\n/* \n sample: \n intercept all requests toward httpbin.org, use a local response\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n const localResponse = {\n statusCode: 200,\n header: { 'Content-Type': 'application/json' },\n body: '{\"hello\": \"this is local response\"}'\n };\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n return {\n response: localResponse\n };\n }\n },\n};\n\nmodify request header\n\nmodify the user-agent sent to httpbin.org\n\nanyproxy --rule rule_sample/sample_modify_request_header.js\n\n/* \n sample: \n modify the user-agent in requests toward httpbin.org\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0';\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\nmodify request body\n\nmodify the post body of http://httpbin.org/post\n\nanyproxy --rule rule_sample/sample_modify_request_data.js\n\n/*\n sample:\n modify the post data towards http://httpbin.org/post\n test:\n curl -H \"Content-Type: text/plain\" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001\n expected response:\n { \"data\": \"i-am-anyproxy-modified-post-data\" }\n*/\nmodule.exports = {\n summary: 'Rule to modify request data',\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) {\n return {\n requestData: 'i-am-anyproxy-modified-post-data'\n };\n }\n },\n};\n\nmodify the request target\n\nsend all the request towards http://httpbin.org/ to http://httpbin.org/user-agent\n\nanyproxy --rule rule_sample/sample_modify_request_path.js\n\n/*\n sample:\n redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html\n test:\n curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n expected response:\n 'hello world' from 127.0.0.1:8001/index.html\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n requestDetail.protocol = 'http';\n newRequestOptions.hostname = '127.0.0.1'\n newRequestOptions.port = '8008';\n newRequestOptions.path = '/index.html';\n newRequestOptions.method = 'GET';\n return requestDetail;\n }\n },\n *beforeDealHttpsRequest(requestDetail) {\n return true;\n }\n};\n\nmodify request protocol\n\nmodify the http request towards http://httpbin.org to https\n\nanyproxy --rule rule_sample/sample_modify_request_protocol.js\n\n/* \n sample: \n redirect all http requests of httpbin.org to https\n test:\n curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001\n expected response:\n { \"X-Forwarded-Protocol\": \"https\" }\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newOption = requestDetail.requestOptions;\n newOption.port = 443;\n return {\n protocol: 'https',\n requestOptions: newOption\n };\n }\n }\n};\n\nmodify response status code\n\nmodify all status code from http://httpbin.org to 404\n\nanyproxy --rule rule_sample/sample_modify_response_statuscode.js\n\n/* \n sample: \n modify all status code of http://httpbin.org/ to 404\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n HTTP/1.1 404 Not Found\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newResponse = responseDetail.response;\n newResponse.statusCode = 404;\n return {\n response: newResponse\n };\n }\n }\n};\n\nmodify the response header\n\nadd X-Proxy-By:AnyProxy to the response header from http://httpbin.org/user-agent\n\nanyproxy --rule rule_sample/sample_modify_response_header.js\n\n/* \n sample: \n modify response header of http://httpbin.org/user-agent\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n X-Proxy-By: AnyProxy\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) {\n const newResponse = responseDetail.response;\n newResponse.header['X-Proxy-By'] = 'AnyProxy';\n return {\n response: newResponse\n };\n }\n }\n};\n\nmodify response data and delay\n\nappend some info to the response of http://httpbin.org/user-agent, then delay the response for 5 seconds.\n\nanyproxy --rule rule_sample/sample_modify_response_data.js\n\n/* \n sample: \n modify response data of http://httpbin.org/user-agent\n test:\n curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n { \"user-agent\": \"curl/7.43.0\" } -- AnyProxy Hacked! --\n*/\n\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '-- AnyProxy Hacked! --';\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay the response for 5s\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\nConfig Certification\nConfig root CA in OSX\n\nthis kind of errors is usually caused by untrusted root CA\n\n\n\nWarning: please keep your root CA safe since it may influence your system security.\n\ninstall :\n\ndouble click rootCA.crt\n\nadd cert into login or system\n\n\n\n\nfind the newly imported AnyProxy certificates, configured as Always Trust\n\n\nConfig root CA in windows\n\nConfig OSX system proxy\n\nthe config is in wifi - advanced\n\n\nconfig http proxy server\n\ntake Chrome extent [SwitchyOmega] as an example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例\n\n\ntrust root CA in iOS\n\nClick Root CA in web ui, and follow the instruction to install\n\n\ntrust root CA in iOS after 10.3\n\nBesides installing root CA, you have to \"turn on\" the certificate for web manually in settings - general - about - Certificate Trust Settings. Otherwire, safari will not trust the root CA generated by AnyProxy.\n\n\ntrust root CA in Android\nFirst of all, you need to download the root CA by clicking Root CA in web ui, and then scan the QR code.\nInstalling CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path.\n\nThe downloaded CA file can be directly installed by clicking, this is the easist way\nYou need to install the CA file from other menu, such as:\nSettings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install\nSettings -> Security -> Install from SD card, and find you CA file to install\n\n\n\nThere are several file extensions of CA file which may not be compatible with all kinds of Android phones. .crt file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing.\nconfig iOS/Android proxy server\n\nproxy settings are placed in wifi setting\n\niOS\n\n\n\n\nAndroid\n\n\nFAQ\nQ: can not deal https request in rule module.\n\nA: Any of these options could be used to change the way AnyProxy deall https requests\nconfig --intercept when luanching AnyProxy via cli, or use forceProxyHttps when using as an npm module\nplace a beforeDealHttpsRequest function in your rule file and determine which request to intercept by your own.\n\n\n\nQ: get an error says function is not yieldable\n\nA: Rule module is driven by co. The functions inside should be yieldable, i.e. return a promise or be a generator function.\n\nQ: The connection is not private\nAnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below.\n\nIf you run AnyProxy by command line\nPass in the option --ignore-unauthorized-ssl to ignore the certification errors, please mind that the option will be active for all connections.\nanyproxy -i --ignore-unauthorized-ssl\n\n\nIf you run AnyProxy by Nodejs\nPass in the option dangerouslyIgnoreUnauthorized:true, like this:\nconst options = {\n ...,\n dangerouslyIgnoreUnauthorized: true\n};\n\nconst anyproxyIns = new AnyProxy.ProxyCore(options);\nanyproxyIns.start();\n\nThis is also a global option, all certification errors will be ignored\n\nWith the help of AnyProxy Rule\nYou can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs Http.rquest, as we do in AnyProxy. A simple demo below:\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n // set rejectUnauthorized as false\n newRequestOptions.rejectUnauthorized = false;\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\nAnd we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to!\n\n\n"}}} ================================================ FILE: docs/en/src_doc.md ================================================ AnyProxy =================== AnyProxy is a fully configurable http/https proxy in NodeJS. Ref: [中文文档](../cn) Github: * https://github.com/alibaba/anyproxy Features: * Offer you the ablity to handle http traffic by invoking a js module * Intercept https * GUI webinterface Change Logs since 3.x: * Support Promise and Generator in rule module * Simplified interface in rule module * A newly designed web interface # Getting Start ### install To Debian and Ubuntu users, you may need to install `nodejs-legacy` at the same time ```bash sudo apt-get install nodejs-legacy ``` Then install the AnyProxy ```bash npm install -g anyproxy ``` ### launch * start AnyProxy in command line, with default port 8001 ```bash anyproxy ``` * now you can use http proxy server by 127.0.0.1:8001 * visit http://127.0.0.1:8002 to see the http requests ### options * specify the port of http proxy ```bash anyproxy --port 1080 ``` ### Use AnyProxy as an npm module AnyProxy can be used as an npm module > To enable https feature, please guide users to use `anyproxy-ca` in cli. Or use methods under `AnyProxy.utils.certMgr` to generate certificates. * install ```bash npm i anyproxy --save ``` * sample ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * create a proxy server ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} required, port number of proxy server * `rule` {object} your rule module * `throttle` {number} throttle in kb/s, unlimited for default * `forceProxyHttps` {boolean} in force intercept all https request, default to `false` * `silent` {boolean} if keep silent in console, false for default `false` * `dangerouslyIgnoreUnauthorized` {boolean} if ignore certificate error in request, default to `false` * `wsIntercept` {boolean} whether to intercept websocket, default to `false` * `webInterface` {object} config for web interface * `enable` {boolean} if enable web interface, default to `false` * `webPort` {number} port number for web interface * Event: `ready` * emit when proxy server is ready * sample ```js proxy.on('ready', function() { }) ``` * Event: `error` * emit when error happened inside proxy server * sample ```js proxy.on('error', function() { }) ``` * Method: `start` * start proxy server * sample ```js proxy.start(); ``` * Method: `close` * close proxy server * sample ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * manage the system proxy config. sudo password may be required * sample ```js // set 127.0.0.1:8001 as system http server AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // disable global proxy server AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * Manage certificates of AnyProxy * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * detect if AnyProx rootCA exists * `AnyProxy.utils.certMgr.generateRootCA(callback)` * generate a rootCA * Sample ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # Proxy Https * AnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate. > Under the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network. * generate certifycates and intercept ```bash anyproxy-ca #generate root CA. manually trust it after that. anyproxy --intercept #launch anyproxy and intercept all https traffic ``` * [Appendix:how to trust CA](#config-certification) # Proxy WebSocket ```bash anyproxy --ws-intercept ``` > The `wss` requests will be handled automatically when the `HTTPS` intercept is turned on, but AnyProxy will not record the data by default. You need to specify the `--ws-intercept` to tell AnyProxy to record it. # Rule Introduction AnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests. > Make sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk. Rule module could do the following stuff: * intercept and modify the request which is being sent * editable fields include request header, body, target address * intercept and modify the response from server * editable fields include response status code, header, body * intercept https requests, modify request and response ### sample * Target * write a rule module to append some text to the response of GET http://httpbin.org/user-agent, and delay the response for 5 seconds * Step 1,Write the rule file, save as sample.js ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, start AnyProxy and load the rule file * run `anyproxy --rule sample.js` * Step 3, test * use curl ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * use browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent * the expected response from proxy is ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, view the request log * visit http://127.0.0.1:8002, the request just sent should be listed here ### how does it work * The flow chart is as follows * When got an http request, the entire process of proxy server is * AnyProxy collects all the quest info, include method, header, body * AnyProxy calls `beforeSendRequest` of the rule module. Rule module deal the request, return new request param or response content * If `beforeSendRequest` returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here. * Send request to target server, collect response * Call `beforeSendResponse` of the rule module. Rule module deal the response data * Send response to client * When AnyProxy get https request, it could replace the certificate and decrypt the request data * AnyProxy calls `beforeDealHttpsRequest` of the rule module * If the function returns `true`, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed. ### how to load rule module * use local file ```bash anyproxy --rule ./rule.js ``` * use an online rule file ```bash anyproxy --rule https://sample.com/rule.js ``` * use an npm module * AnyProxy uses `require()` to load rule module. You could either load a local npm module or a global-installed one. ```bash anyproxy --rule ./myRulePkg/ #local module npm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module ``` # Rule module interface A typical rule module is as follows. All the functions are optional, just write the part you are interested in. ```js module.exports = { // introduction summary: 'my customized rule for AnyProxy', // intercept before send request to server *beforeSendRequest(requestDetail) { /* ... */ }, // deal response before send to client *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // if deal https request *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // error happened when dealing requests *onError(requestDetail, error) { /* ... */ }, // error happened when connect to https server *onConnectError(requestDetail, error) { /* ... */ } }; ``` > All functions in your rule file, except summary, are all driven by [co](https://www.npmjs.com/package/co) . They should be yieldable, i.e. return a promise or be a generator function. ### summary #### summary * Introduction of this rule file. AnyProxy will read this field and give some tip to user. ### beforeSendRequest #### beforeSendRequest(requestDetail) * Before sending request to server, AnyProxy will call `beforeSendRequest` with param `requestDetail` * `requestDetail` * `protocol` {string} the protocol to use, http or https * `requestOptions` {object} the options of the request-to-go, a param of require('http').request . ref: https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} request body * `url` {string} request url * `_req` {object} the native node.js request object * e.g. When requesting *anyproxy.io*, `requestDetail` is something like the following ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the request protocol,i.e. force use https ```js return { protocol: 'https' }; ``` * modify request param ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * modify request body ```js return { requestData: 'my new request data' // requestOptions can also be used here }; ``` * give response to the client, not sending request any longer. `statusCode` `headers`are required is this situation. ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * Before sending response to client, AnyProxy will call `beforeSendResponse` with param `requestDetail` `responseDetail` * `requestDetail` is the same param as in `beforeSendRequest` * `responseDetail` * `response` {object} the response from server, includes `statusCode` `header` `body` * `_res` {object} the native node.js response object * e.g. When requesting *anyproxy.io*, `responseDetail` is something like the following ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the response status code ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * modify the response content ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * When receiving https request, AnyProxy will call `beforeDealHttpsRequest` with param `requestDetail` * If configed with `forceProxyHttps` in launching, AnyProxy will skip calling this method * Only by returning true, AnyProxy will try to replace the certificate and intercept the https request. * `requestDetail` * `host` {string} the target host to request. Due to the request protocol, full url couldn't be got here * `_req` {object} the native node.js request object. The `_req` here refers to the CONNECT request. * return value * `true` or `false`, whether AnyProxy should intercept the https request ### onError #### onError(requestDetail, error) * AnyProxy will call this method when an error happened in request handling. * Errors usually are issued during requesting, e.g. DNS failure, request timeout * `requestDetail` is the same one as in `beforeSendRequest` * Any of these return values are valid * do nothing, and AnyProxy will response a default error page ```js return null; ``` * return a customized error page ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy will call this method when failed to connect target server in https request * `requestDetail` is the same one as in `beforeDealHttpsRequest` * no return value is required # Rule Samples * here are some samples about frequently used rule file * try these samples by `anyproxy --rule http://....js` * how to test with curl: * request the server directly `curl http://httpbin.org/` * request the server via proxy `curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### use local response * intercept the request towards http://httpbin.org , return the local-defined response {{sample-rule:rule_sample/sample_use_local_response.js}} ### modify request header * modify the user-agent sent to httpbin.org {{sample-rule:rule_sample/sample_modify_request_header.js}} ### modify request body * modify the post body of http://httpbin.org/post {{sample-rule:rule_sample/sample_modify_request_data.js}} ### modify the request target * send all the request towards http://httpbin.org/ to http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_request_path.js}} ### modify request protocol * modify the http request towards http://httpbin.org to https {{sample-rule:rule_sample/sample_modify_request_protocol.js}} ### modify response status code * modify all status code from http://httpbin.org to 404 {{sample-rule:rule_sample/sample_modify_response_statuscode.js}} ### modify the response header * add X-Proxy-By:AnyProxy to the response header from http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_response_header.js}} ### modify response data and delay * append some info to the response of http://httpbin.org/user-agent, then delay the response for 5 seconds. {{sample-rule:rule_sample/sample_modify_response_data.js}} # Config Certification ### Config root CA in OSX * this kind of errors is usually caused by untrusted root CA > Warning: please keep your root CA safe since it may influence your system security. install : * double click *rootCA.crt* * add cert into login or system * find the newly imported AnyProxy certificates, configured as **Always Trust** ### Config root CA in windows ### Config OSX system proxy * the config is in wifi - advanced ### config http proxy server * take Chrome extent [SwitchyOmega] as an example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### trust root CA in iOS * Click *Root CA* in web ui, and follow the instruction to install ### trust root CA in iOS after 10.3 * Besides installing root CA, you have to "turn on" the certificate for web manually in *settings - general - about - Certificate Trust Settings*. Otherwire, safari will not trust the root CA generated by AnyProxy. ### trust root CA in Android First of all, you need to download the root CA by clicking *Root CA* in web ui, and then scan the QR code. Installing CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path. * The downloaded CA file can be directly installed by clicking, this is the easist way * You need to install the CA file from other menu, such as: * Settings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install * Settings -> Security -> Install from SD card, and find you CA file to install There are several file extensions of CA file which may not be compatible with all kinds of Android phones. `.crt` file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing. ### config iOS/Android proxy server * proxy settings are placed in wifi setting * iOS * Android # FAQ ### Q: can not deal https request in rule module. * A: Any of these options could be used to change the way AnyProxy deall https requests 1. config `--intercept` when luanching AnyProxy via cli, or use `forceProxyHttps` when using as an npm module 2. place a `beforeDealHttpsRequest` function in your rule file and determine which request to intercept by your own. ### Q: get an error says *function is not yieldable* * A: Rule module is driven by [co](https://www.npmjs.com/package/co). The functions inside should be yieldable, i.e. return a promise or be a generator function. ### Q: The connection is not private AnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below. - If you run AnyProxy by command line Pass in the option `--ignore-unauthorized-ssl` to ignore the certification errors, please mind that the option will be active for all connections. ```bash anyproxy -i --ignore-unauthorized-ssl ``` - If you run AnyProxy by Nodejs Pass in the option `dangerouslyIgnoreUnauthorized:true`, like this: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *This is also a global option, all certification errors will be ignored* - With the help of AnyProxy Rule You can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs `Http.rquest`, as we do in AnyProxy. A simple demo below: ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // set rejectUnauthorized as false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` And we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to! ================================================ FILE: docs/gitbook/gitbook-plugin-fontsettings/fontsettings.js ================================================ require(['gitbook', 'jquery'], function(gitbook, $) { // Configuration var MAX_SIZE = 4, MIN_SIZE = 0, BUTTON_ID; // Current fontsettings state var fontState; // Default themes var THEMES = [ { config: 'white', text: 'White', id: 0 }, { config: 'sepia', text: 'Sepia', id: 1 }, { config: 'night', text: 'Night', id: 2 } ]; // Default font families var FAMILIES = [ { config: 'serif', text: 'Serif', id: 0 }, { config: 'sans', text: 'Sans', id: 1 } ]; // Return configured themes function getThemes() { return THEMES; } // Modify configured themes function setThemes(themes) { THEMES = themes; updateButtons(); } // Return configured font families function getFamilies() { return FAMILIES; } // Modify configured font families function setFamilies(families) { FAMILIES = families; updateButtons(); } // Save current font settings function saveFontSettings() { gitbook.storage.set('fontState', fontState); update(); } // Increase font size function enlargeFontSize(e) { e.preventDefault(); if (fontState.size >= MAX_SIZE) return; fontState.size++; saveFontSettings(); } // Decrease font size function reduceFontSize(e) { e.preventDefault(); if (fontState.size <= MIN_SIZE) return; fontState.size--; saveFontSettings(); } // Change font family function changeFontFamily(configName, e) { if (e && e instanceof Event) { e.preventDefault(); } var familyId = getFontFamilyId(configName); fontState.family = familyId; saveFontSettings(); } // Change type of color theme function changeColorTheme(configName, e) { if (e && e instanceof Event) { e.preventDefault(); } var $book = gitbook.state.$book; // Remove currently applied color theme if (fontState.theme !== 0) $book.removeClass('color-theme-'+fontState.theme); // Set new color theme var themeId = getThemeId(configName); fontState.theme = themeId; if (fontState.theme !== 0) $book.addClass('color-theme-'+fontState.theme); saveFontSettings(); } // Return the correct id for a font-family config key // Default to first font-family function getFontFamilyId(configName) { // Search for plugin configured font family var configFamily = $.grep(FAMILIES, function(family) { return family.config == configName; })[0]; // Fallback to default font family return (!!configFamily)? configFamily.id : 0; } // Return the correct id for a theme config key // Default to first theme function getThemeId(configName) { // Search for plugin configured theme var configTheme = $.grep(THEMES, function(theme) { return theme.config == configName; })[0]; // Fallback to default theme return (!!configTheme)? configTheme.id : 0; } function update() { var $book = gitbook.state.$book; $('.font-settings .font-family-list li').removeClass('active'); $('.font-settings .font-family-list li:nth-child('+(fontState.family+1)+')').addClass('active'); $book[0].className = $book[0].className.replace(/\bfont-\S+/g, ''); $book.addClass('font-size-'+fontState.size); $book.addClass('font-family-'+fontState.family); if(fontState.theme !== 0) { $book[0].className = $book[0].className.replace(/\bcolor-theme-\S+/g, ''); $book.addClass('color-theme-'+fontState.theme); } } function init(config) { // Search for plugin configured font family var configFamily = getFontFamilyId(config.family), configTheme = getThemeId(config.theme); // Instantiate font state object fontState = gitbook.storage.get('fontState', { size: config.size || 2, family: configFamily, theme: configTheme }); update(); } function updateButtons() { // Remove existing fontsettings buttons if (!!BUTTON_ID) { gitbook.toolbar.removeButton(BUTTON_ID); } // Create buttons in toolbar BUTTON_ID = gitbook.toolbar.createButton({ icon: 'fa fa-font', label: 'Font Settings', className: 'font-settings', dropdown: [ [ { text: 'A', className: 'font-reduce', onClick: reduceFontSize }, { text: 'A', className: 'font-enlarge', onClick: enlargeFontSize } ], $.map(FAMILIES, function(family) { family.onClick = function(e) { return changeFontFamily(family.config, e); }; return family; }), $.map(THEMES, function(theme) { theme.onClick = function(e) { return changeColorTheme(theme.config, e); }; return theme; }) ] }); } // Init configuration at start gitbook.events.bind('start', function(e, config) { var opts = config.fontsettings; // Generate buttons at start updateButtons(); // Init current settings init(opts); }); // Expose API gitbook.fontsettings = { enlargeFontSize: enlargeFontSize, reduceFontSize: reduceFontSize, setTheme: changeColorTheme, setFamily: changeFontFamily, getThemes: getThemes, setThemes: setThemes, getFamilies: getFamilies, setFamilies: setFamilies }; }); ================================================ FILE: docs/gitbook/gitbook-plugin-fontsettings/website.css ================================================ /* * Theme 1 */ .color-theme-1 .dropdown-menu { background-color: #111111; border-color: #7e888b; } .color-theme-1 .dropdown-menu .dropdown-caret .caret-inner { border-bottom: 9px solid #111111; } .color-theme-1 .dropdown-menu .buttons { border-color: #7e888b; } .color-theme-1 .dropdown-menu .button { color: #afa790; } .color-theme-1 .dropdown-menu .button:hover { color: #73553c; } /* * Theme 2 */ .color-theme-2 .dropdown-menu { background-color: #2d3143; border-color: #272a3a; } .color-theme-2 .dropdown-menu .dropdown-caret .caret-inner { border-bottom: 9px solid #2d3143; } .color-theme-2 .dropdown-menu .buttons { border-color: #272a3a; } .color-theme-2 .dropdown-menu .button { color: #62677f; } .color-theme-2 .dropdown-menu .button:hover { color: #f4f4f5; } .book .book-header .font-settings .font-enlarge { line-height: 30px; font-size: 1.4em; } .book .book-header .font-settings .font-reduce { line-height: 30px; font-size: 1em; } .book.color-theme-1 .book-body { color: #704214; background: #f3eacb; } .book.color-theme-1 .book-body .page-wrapper .page-inner section { background: #f3eacb; } .book.color-theme-2 .book-body { color: #bdcadb; background: #1c1f2b; } .book.color-theme-2 .book-body .page-wrapper .page-inner section { background: #1c1f2b; } .book.font-size-0 .book-body .page-inner section { font-size: 1.2rem; } .book.font-size-1 .book-body .page-inner section { font-size: 1.4rem; } .book.font-size-2 .book-body .page-inner section { font-size: 1.6rem; } .book.font-size-3 .book-body .page-inner section { font-size: 2.2rem; } .book.font-size-4 .book-body .page-inner section { font-size: 4rem; } .book.font-family-0 { font-family: Georgia, serif; } .book.font-family-1 { font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal { color: #704214; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal a { color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h1, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h2, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h3, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h4, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h5, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h6 { color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h1, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h2 { border-color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal h6 { color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal hr { background-color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal blockquote { border-color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code { background: #fdf6e3; color: #657b83; border-color: #f8df9c; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal .highlight { background-color: inherit; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal table th, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal table td { border-color: #f5d06c; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal table tr { color: inherit; background-color: #fdf6e3; border-color: #444444; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal table tr:nth-child(2n) { background-color: #fbeecb; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal { color: #bdcadb; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal a { color: #3eb1d0; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h1, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h2, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h3, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h4, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h5, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h6 { color: #fffffa; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h1, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h2 { border-color: #373b4e; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal h6 { color: #373b4e; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal hr { background-color: #373b4e; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal blockquote { border-color: #373b4e; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code { color: #9dbed8; background: #2d3143; border-color: #2d3143; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal .highlight { background-color: #282a39; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal table th, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal table td { border-color: #3b3f54; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal table tr { color: #b6c2d2; background-color: #2d3143; border-color: #3b3f54; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal table tr:nth-child(2n) { background-color: #35394b; } .book.color-theme-1 .book-header { color: #afa790; background: transparent; } .book.color-theme-1 .book-header .btn { color: #afa790; } .book.color-theme-1 .book-header .btn:hover { color: #73553c; background: none; } .book.color-theme-1 .book-header h1 { color: #704214; } .book.color-theme-2 .book-header { color: #7e888b; background: transparent; } .book.color-theme-2 .book-header .btn { color: #3b3f54; } .book.color-theme-2 .book-header .btn:hover { color: #fffff5; background: none; } .book.color-theme-2 .book-header h1 { color: #bdcadb; } .book.color-theme-1 .book-body .navigation { color: #afa790; } .book.color-theme-1 .book-body .navigation:hover { color: #73553c; } .book.color-theme-2 .book-body .navigation { color: #383f52; } .book.color-theme-2 .book-body .navigation:hover { color: #fffff5; } /* * Theme 1 */ .book.color-theme-1 .book-summary { color: #afa790; background: #111111; border-right: 1px solid rgba(0, 0, 0, 0.07); } .book.color-theme-1 .book-summary .book-search { background: transparent; } .book.color-theme-1 .book-summary .book-search input, .book.color-theme-1 .book-summary .book-search input:focus { border: 1px solid transparent; } .book.color-theme-1 .book-summary ul.summary li.divider { background: #7e888b; box-shadow: none; } .book.color-theme-1 .book-summary ul.summary li i.fa-check { color: #33cc33; } .book.color-theme-1 .book-summary ul.summary li.done > a { color: #877f6a; } .book.color-theme-1 .book-summary ul.summary li a, .book.color-theme-1 .book-summary ul.summary li span { color: #877f6a; background: transparent; font-weight: normal; } .book.color-theme-1 .book-summary ul.summary li.active > a, .book.color-theme-1 .book-summary ul.summary li a:hover { color: #704214; background: transparent; font-weight: normal; } /* * Theme 2 */ .book.color-theme-2 .book-summary { color: #bcc1d2; background: #2d3143; border-right: none; } .book.color-theme-2 .book-summary .book-search { background: transparent; } .book.color-theme-2 .book-summary .book-search input, .book.color-theme-2 .book-summary .book-search input:focus { border: 1px solid transparent; } .book.color-theme-2 .book-summary ul.summary li.divider { background: #272a3a; box-shadow: none; } .book.color-theme-2 .book-summary ul.summary li i.fa-check { color: #33cc33; } .book.color-theme-2 .book-summary ul.summary li.done > a { color: #62687f; } .book.color-theme-2 .book-summary ul.summary li a, .book.color-theme-2 .book-summary ul.summary li span { color: #c1c6d7; background: transparent; font-weight: 600; } .book.color-theme-2 .book-summary ul.summary li.active > a, .book.color-theme-2 .book-summary ul.summary li a:hover { color: #f4f4f5; background: #252737; font-weight: 600; } ================================================ FILE: docs/gitbook/gitbook-plugin-highlight/ebook.css ================================================ pre, code { /* http://jmblog.github.io/color-themes-for-highlightjs */ /* Tomorrow Comment */ /* Tomorrow Red */ /* Tomorrow Orange */ /* Tomorrow Yellow */ /* Tomorrow Green */ /* Tomorrow Aqua */ /* Tomorrow Blue */ /* Tomorrow Purple */ } pre .hljs-comment, code .hljs-comment, pre .hljs-title, code .hljs-title { color: #8e908c; } pre .hljs-variable, code .hljs-variable, pre .hljs-attribute, code .hljs-attribute, pre .hljs-tag, code .hljs-tag, pre .hljs-regexp, code .hljs-regexp, pre .hljs-deletion, code .hljs-deletion, pre .ruby .hljs-constant, code .ruby .hljs-constant, pre .xml .hljs-tag .hljs-title, code .xml .hljs-tag .hljs-title, pre .xml .hljs-pi, code .xml .hljs-pi, pre .xml .hljs-doctype, code .xml .hljs-doctype, pre .html .hljs-doctype, code .html .hljs-doctype, pre .css .hljs-id, code .css .hljs-id, pre .css .hljs-class, code .css .hljs-class, pre .css .hljs-pseudo, code .css .hljs-pseudo { color: #c82829; } pre .hljs-number, code .hljs-number, pre .hljs-preprocessor, code .hljs-preprocessor, pre .hljs-pragma, code .hljs-pragma, pre .hljs-built_in, code .hljs-built_in, pre .hljs-literal, code .hljs-literal, pre .hljs-params, code .hljs-params, pre .hljs-constant, code .hljs-constant { color: #f5871f; } pre .ruby .hljs-class .hljs-title, code .ruby .hljs-class .hljs-title, pre .css .hljs-rules .hljs-attribute, code .css .hljs-rules .hljs-attribute { color: #eab700; } pre .hljs-string, code .hljs-string, pre .hljs-value, code .hljs-value, pre .hljs-inheritance, code .hljs-inheritance, pre .hljs-header, code .hljs-header, pre .hljs-addition, code .hljs-addition, pre .ruby .hljs-symbol, code .ruby .hljs-symbol, pre .xml .hljs-cdata, code .xml .hljs-cdata { color: #718c00; } pre .css .hljs-hexcolor, code .css .hljs-hexcolor { color: #3e999f; } pre .hljs-function, code .hljs-function, pre .python .hljs-decorator, code .python .hljs-decorator, pre .python .hljs-title, code .python .hljs-title, pre .ruby .hljs-function .hljs-title, code .ruby .hljs-function .hljs-title, pre .ruby .hljs-title .hljs-keyword, code .ruby .hljs-title .hljs-keyword, pre .perl .hljs-sub, code .perl .hljs-sub, pre .javascript .hljs-title, code .javascript .hljs-title, pre .coffeescript .hljs-title, code .coffeescript .hljs-title { color: #4271ae; } pre .hljs-keyword, code .hljs-keyword, pre .javascript .hljs-function, code .javascript .hljs-function { color: #8959a8; } pre .hljs, code .hljs { display: block; background: white; color: #4d4d4c; padding: 0.5em; } pre .coffeescript .javascript, code .coffeescript .javascript, pre .javascript .xml, code .javascript .xml, pre .tex .hljs-formula, code .tex .hljs-formula, pre .xml .javascript, code .xml .javascript, pre .xml .vbscript, code .xml .vbscript, pre .xml .css, code .xml .css, pre .xml .hljs-cdata, code .xml .hljs-cdata { opacity: 0.5; } ================================================ FILE: docs/gitbook/gitbook-plugin-highlight/website.css ================================================ .book .book-body .page-wrapper .page-inner section.normal pre, .book .book-body .page-wrapper .page-inner section.normal code { /* http://jmblog.github.com/color-themes-for-google-code-highlightjs */ /* Tomorrow Comment */ /* Tomorrow Red */ /* Tomorrow Orange */ /* Tomorrow Yellow */ /* Tomorrow Green */ /* Tomorrow Aqua */ /* Tomorrow Blue */ /* Tomorrow Purple */ } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-comment, .book .book-body .page-wrapper .page-inner section.normal code .hljs-comment, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .hljs-title { color: #8e908c; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-variable, .book .book-body .page-wrapper .page-inner section.normal code .hljs-variable, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-attribute, .book .book-body .page-wrapper .page-inner section.normal code .hljs-attribute, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-tag, .book .book-body .page-wrapper .page-inner section.normal code .hljs-tag, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-regexp, .book .book-body .page-wrapper .page-inner section.normal code .hljs-regexp, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-deletion, .book .book-body .page-wrapper .page-inner section.normal code .hljs-deletion, .book .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-constant, .book .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-constant, .book .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-tag .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .xml .hljs-tag .hljs-title, .book .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-pi, .book .book-body .page-wrapper .page-inner section.normal code .xml .hljs-pi, .book .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-doctype, .book .book-body .page-wrapper .page-inner section.normal code .xml .hljs-doctype, .book .book-body .page-wrapper .page-inner section.normal pre .html .hljs-doctype, .book .book-body .page-wrapper .page-inner section.normal code .html .hljs-doctype, .book .book-body .page-wrapper .page-inner section.normal pre .css .hljs-id, .book .book-body .page-wrapper .page-inner section.normal code .css .hljs-id, .book .book-body .page-wrapper .page-inner section.normal pre .css .hljs-class, .book .book-body .page-wrapper .page-inner section.normal code .css .hljs-class, .book .book-body .page-wrapper .page-inner section.normal pre .css .hljs-pseudo, .book .book-body .page-wrapper .page-inner section.normal code .css .hljs-pseudo { color: #c82829; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-number, .book .book-body .page-wrapper .page-inner section.normal code .hljs-number, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-preprocessor, .book .book-body .page-wrapper .page-inner section.normal code .hljs-preprocessor, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-pragma, .book .book-body .page-wrapper .page-inner section.normal code .hljs-pragma, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-built_in, .book .book-body .page-wrapper .page-inner section.normal code .hljs-built_in, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-literal, .book .book-body .page-wrapper .page-inner section.normal code .hljs-literal, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-params, .book .book-body .page-wrapper .page-inner section.normal code .hljs-params, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-constant, .book .book-body .page-wrapper .page-inner section.normal code .hljs-constant { color: #f5871f; } .book .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-class .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-class .hljs-title, .book .book-body .page-wrapper .page-inner section.normal pre .css .hljs-rules .hljs-attribute, .book .book-body .page-wrapper .page-inner section.normal code .css .hljs-rules .hljs-attribute { color: #eab700; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-string, .book .book-body .page-wrapper .page-inner section.normal code .hljs-string, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-value, .book .book-body .page-wrapper .page-inner section.normal code .hljs-value, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-inheritance, .book .book-body .page-wrapper .page-inner section.normal code .hljs-inheritance, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-header, .book .book-body .page-wrapper .page-inner section.normal code .hljs-header, .book .book-body .page-wrapper .page-inner section.normal pre .hljs-addition, .book .book-body .page-wrapper .page-inner section.normal code .hljs-addition, .book .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-symbol, .book .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-symbol, .book .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-cdata, .book .book-body .page-wrapper .page-inner section.normal code .xml .hljs-cdata { color: #718c00; } .book .book-body .page-wrapper .page-inner section.normal pre .css .hljs-hexcolor, .book .book-body .page-wrapper .page-inner section.normal code .css .hljs-hexcolor { color: #3e999f; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-function, .book .book-body .page-wrapper .page-inner section.normal code .hljs-function, .book .book-body .page-wrapper .page-inner section.normal pre .python .hljs-decorator, .book .book-body .page-wrapper .page-inner section.normal code .python .hljs-decorator, .book .book-body .page-wrapper .page-inner section.normal pre .python .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .python .hljs-title, .book .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-function .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-function .hljs-title, .book .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-title .hljs-keyword, .book .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-title .hljs-keyword, .book .book-body .page-wrapper .page-inner section.normal pre .perl .hljs-sub, .book .book-body .page-wrapper .page-inner section.normal code .perl .hljs-sub, .book .book-body .page-wrapper .page-inner section.normal pre .javascript .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .javascript .hljs-title, .book .book-body .page-wrapper .page-inner section.normal pre .coffeescript .hljs-title, .book .book-body .page-wrapper .page-inner section.normal code .coffeescript .hljs-title { color: #4271ae; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs-keyword, .book .book-body .page-wrapper .page-inner section.normal code .hljs-keyword, .book .book-body .page-wrapper .page-inner section.normal pre .javascript .hljs-function, .book .book-body .page-wrapper .page-inner section.normal code .javascript .hljs-function { color: #8959a8; } .book .book-body .page-wrapper .page-inner section.normal pre .hljs, .book .book-body .page-wrapper .page-inner section.normal code .hljs { display: block; background: white; color: #4d4d4c; padding: 0.5em; } .book .book-body .page-wrapper .page-inner section.normal pre .coffeescript .javascript, .book .book-body .page-wrapper .page-inner section.normal code .coffeescript .javascript, .book .book-body .page-wrapper .page-inner section.normal pre .javascript .xml, .book .book-body .page-wrapper .page-inner section.normal code .javascript .xml, .book .book-body .page-wrapper .page-inner section.normal pre .tex .hljs-formula, .book .book-body .page-wrapper .page-inner section.normal code .tex .hljs-formula, .book .book-body .page-wrapper .page-inner section.normal pre .xml .javascript, .book .book-body .page-wrapper .page-inner section.normal code .xml .javascript, .book .book-body .page-wrapper .page-inner section.normal pre .xml .vbscript, .book .book-body .page-wrapper .page-inner section.normal code .xml .vbscript, .book .book-body .page-wrapper .page-inner section.normal pre .xml .css, .book .book-body .page-wrapper .page-inner section.normal code .xml .css, .book .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-cdata, .book .book-body .page-wrapper .page-inner section.normal code .xml .hljs-cdata { opacity: 0.5; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code { /* Orginal Style from ethanschoonover.com/solarized (c) Jeremy Hull */ /* Solarized Green */ /* Solarized Cyan */ /* Solarized Blue */ /* Solarized Yellow */ /* Solarized Orange */ /* Solarized Red */ /* Solarized Violet */ } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs { display: block; padding: 0.5em; background: #fdf6e3; color: #657b83; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-comment, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-comment, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-template_comment, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-template_comment, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .diff .hljs-header, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .diff .hljs-header, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-doctype, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-doctype, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-pi, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-pi, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .lisp .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .lisp .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-javadoc, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-javadoc { color: #93a1a1; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-keyword, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-keyword, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-winutils, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-winutils, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .method, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .method, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-addition, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-addition, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-tag, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .css .hljs-tag, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-request, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-request, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-status, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-status, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .nginx .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .nginx .hljs-title { color: #859900; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-number, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-number, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-command, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-command, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-tag .hljs-value, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-tag .hljs-value, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-rules .hljs-value, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-rules .hljs-value, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-phpdoc, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-phpdoc, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .tex .hljs-formula, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .tex .hljs-formula, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-regexp, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-regexp, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-hexcolor, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-hexcolor, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-link_url, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-link_url { color: #2aa198; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-localvars, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-localvars, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-chunk, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-chunk, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-decorator, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-decorator, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-built_in, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-built_in, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-identifier, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-identifier, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .vhdl .hljs-literal, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .vhdl .hljs-literal, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-id, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-id, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-function, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .css .hljs-function { color: #268bd2; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-attribute, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-attribute, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-variable, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-variable, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .lisp .hljs-body, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .lisp .hljs-body, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .smalltalk .hljs-number, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .smalltalk .hljs-number, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-constant, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-constant, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-class .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-class .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-parent, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-parent, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .haskell .hljs-type, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .haskell .hljs-type, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-link_reference, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-link_reference { color: #b58900; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-preprocessor, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-preprocessor, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-preprocessor .hljs-keyword, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-preprocessor .hljs-keyword, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-pragma, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-pragma, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-shebang, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-shebang, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-symbol, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-symbol, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-symbol .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-symbol .hljs-string, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .diff .hljs-change, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .diff .hljs-change, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-special, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-special, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-attr_selector, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-attr_selector, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-subst, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-subst, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-cdata, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-cdata, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .clojure .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .clojure .hljs-title, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-pseudo, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .css .hljs-pseudo, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-header, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-header { color: #cb4b16; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-deletion, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-deletion, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-important, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-important { color: #dc322f; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .hljs-link_label, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .hljs-link_label { color: #6c71c4; } .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal pre .tex .hljs-formula, .book.color-theme-1 .book-body .page-wrapper .page-inner section.normal code .tex .hljs-formula { background: #eee8d5; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code { /* Tomorrow Night Bright Theme */ /* Original theme - https://github.com/chriskempson/tomorrow-theme */ /* http://jmblog.github.com/color-themes-for-google-code-highlightjs */ /* Tomorrow Comment */ /* Tomorrow Red */ /* Tomorrow Orange */ /* Tomorrow Yellow */ /* Tomorrow Green */ /* Tomorrow Aqua */ /* Tomorrow Blue */ /* Tomorrow Purple */ } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-comment, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-comment, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-title { color: #969896; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-variable, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-variable, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-attribute, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-attribute, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-tag, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-tag, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-regexp, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-regexp, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-deletion, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-deletion, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-constant, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-constant, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-tag .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .hljs-tag .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-pi, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .hljs-pi, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-doctype, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .hljs-doctype, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .html .hljs-doctype, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .html .hljs-doctype, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-id, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .css .hljs-id, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-class, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .css .hljs-class, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-pseudo, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .css .hljs-pseudo { color: #d54e53; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-number, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-number, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-preprocessor, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-preprocessor, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-pragma, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-pragma, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-built_in, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-built_in, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-literal, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-literal, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-params, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-params, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-constant, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-constant { color: #e78c45; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-class .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-class .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-rules .hljs-attribute, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .css .hljs-rules .hljs-attribute { color: #e7c547; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-string, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-string, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-value, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-value, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-inheritance, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-inheritance, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-header, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-header, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-addition, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-addition, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-symbol, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-symbol, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-cdata, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .hljs-cdata { color: #b9ca4a; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .css .hljs-hexcolor, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .css .hljs-hexcolor { color: #70c0b1; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-function, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-function, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .python .hljs-decorator, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .python .hljs-decorator, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .python .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .python .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-function .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-function .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .ruby .hljs-title .hljs-keyword, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .ruby .hljs-title .hljs-keyword, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .perl .hljs-sub, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .perl .hljs-sub, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .javascript .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .javascript .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .coffeescript .hljs-title, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .coffeescript .hljs-title { color: #7aa6da; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs-keyword, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs-keyword, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .javascript .hljs-function, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .javascript .hljs-function { color: #c397d8; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .hljs, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .hljs { display: block; background: black; color: #eaeaea; padding: 0.5em; } .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .coffeescript .javascript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .coffeescript .javascript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .javascript .xml, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .javascript .xml, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .tex .hljs-formula, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .tex .hljs-formula, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .javascript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .javascript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .vbscript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .vbscript, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .css, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .css, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal pre .xml .hljs-cdata, .book.color-theme-2 .book-body .page-wrapper .page-inner section.normal code .xml .hljs-cdata { opacity: 0.5; } ================================================ FILE: docs/gitbook/gitbook-plugin-lunr/search-lunr.js ================================================ require([ 'gitbook', 'jquery' ], function(gitbook, $) { // Define global search engine function LunrSearchEngine() { this.index = null; this.store = {}; this.name = 'LunrSearchEngine'; } // Initialize lunr by fetching the search index LunrSearchEngine.prototype.init = function() { var that = this; var d = $.Deferred(); $.getJSON(gitbook.state.basePath+'/search_index.json') .then(function(data) { // eslint-disable-next-line no-undef that.index = lunr.Index.load(data.index); that.store = data.store; d.resolve(); }); return d.promise(); }; // Search for a term and return results LunrSearchEngine.prototype.search = function(q, offset, length) { var that = this; var results = []; if (this.index) { results = $.map(this.index.search(q), function(result) { var doc = that.store[result.ref]; return { title: doc.title, url: doc.url, body: doc.summary || doc.body }; }); } return $.Deferred().resolve({ query: q, results: results.slice(0, length), count: results.length }).promise(); }; // Set gitbook research gitbook.events.bind('start', function(e, config) { var engine = gitbook.search.getEngine(); if (!engine) { gitbook.search.setEngine(LunrSearchEngine, config); } }); }); ================================================ FILE: docs/gitbook/gitbook-plugin-search/search-engine.js ================================================ require([ 'gitbook', 'jquery' ], function(gitbook, $) { // Global search objects var engine = null; var initialized = false; // Set a new search engine function setEngine(Engine, config) { initialized = false; engine = new Engine(config); init(config); } // Initialize search engine with config function init(config) { if (!engine) throw new Error('No engine set for research. Set an engine using gitbook.research.setEngine(Engine).'); return engine.init(config) .then(function() { initialized = true; gitbook.events.trigger('search.ready'); }); } // Launch search for query q function query(q, offset, length) { if (!initialized) throw new Error('Search has not been initialized'); return engine.search(q, offset, length); } // Get stats about search function getEngine() { return engine? engine.name : null; } function isInitialized() { return initialized; } // Initialize gitbook.search gitbook.search = { setEngine: setEngine, getEngine: getEngine, query: query, isInitialized: isInitialized }; }); ================================================ FILE: docs/gitbook/gitbook-plugin-search/search.css ================================================ /* This CSS only styled the search results section, not the search input It defines the basic interraction to hide content when displaying results, etc */ #book-search-results .search-results { display: none; } #book-search-results .search-results ul.search-results-list { list-style-type: none; padding-left: 0; } #book-search-results .search-results ul.search-results-list li { margin-bottom: 1.5rem; padding-bottom: 0.5rem; /* Highlight results */ } #book-search-results .search-results ul.search-results-list li p em { background-color: rgba(255, 220, 0, 0.4); font-style: normal; } #book-search-results .search-results .no-results { display: none; } #book-search-results.open .search-results { display: block; } #book-search-results.open .search-noresults { display: none; } #book-search-results.no-results .search-results .has-results { display: none; } #book-search-results.no-results .search-results .no-results { display: block; } ================================================ FILE: docs/gitbook/gitbook-plugin-search/search.js ================================================ require([ 'gitbook', 'jquery' ], function(gitbook, $) { var MAX_RESULTS = 15; var MAX_DESCRIPTION_SIZE = 500; var usePushState = (typeof history.pushState !== 'undefined'); // DOM Elements var $body = $('body'); var $bookSearchResults; var $searchInput; var $searchList; var $searchTitle; var $searchResultsCount; var $searchQuery; // Throttle search function throttle(fn, wait) { var timeout; return function() { var ctx = this, args = arguments; if (!timeout) { timeout = setTimeout(function() { timeout = null; fn.apply(ctx, args); }, wait); } }; } function displayResults(res) { $bookSearchResults.addClass('open'); var noResults = res.count == 0; $bookSearchResults.toggleClass('no-results', noResults); // Clear old results $searchList.empty(); // Display title for research $searchResultsCount.text(res.count); $searchQuery.text(res.query); // Create an
    • element for each result res.results.forEach(function(res) { var $li = $('
    • ', { 'class': 'search-results-item' }); var $title = $('

      '); var $link = $('', { 'href': gitbook.state.basePath + '/' + res.url, 'text': res.title }); var content = res.body.trim(); if (content.length > MAX_DESCRIPTION_SIZE) { content = content.slice(0, MAX_DESCRIPTION_SIZE).trim()+'...'; } var $content = $('

      ').html(content); $link.appendTo($title); $title.appendTo($li); $content.appendTo($li); $li.appendTo($searchList); }); } function launchSearch(q) { // Add class for loading $body.addClass('with-search'); $body.addClass('search-loading'); // Launch search query throttle(gitbook.search.query(q, 0, MAX_RESULTS) .then(function(results) { displayResults(results); }) .always(function() { $body.removeClass('search-loading'); }), 1000); } function closeSearch() { $body.removeClass('with-search'); $bookSearchResults.removeClass('open'); } function launchSearchFromQueryString() { var q = getParameterByName('q'); if (q && q.length > 0) { // Update search input $searchInput.val(q); // Launch search launchSearch(q); } } function bindSearch() { // Bind DOM $searchInput = $('#book-search-input input'); $bookSearchResults = $('#book-search-results'); $searchList = $bookSearchResults.find('.search-results-list'); $searchTitle = $bookSearchResults.find('.search-results-title'); $searchResultsCount = $searchTitle.find('.search-results-count'); $searchQuery = $searchTitle.find('.search-query'); // Launch query based on input content function handleUpdate() { var q = $searchInput.val(); if (q.length == 0) { closeSearch(); } else { launchSearch(q); } } // Detect true content change in search input // Workaround for IE < 9 var propertyChangeUnbound = false; $searchInput.on('propertychange', function(e) { if (e.originalEvent.propertyName == 'value') { handleUpdate(); } }); // HTML5 (IE9 & others) $searchInput.on('input', function(e) { // Unbind propertychange event for IE9+ if (!propertyChangeUnbound) { $(this).unbind('propertychange'); propertyChangeUnbound = true; } handleUpdate(); }); // Push to history on blur $searchInput.on('blur', function(e) { // Update history state if (usePushState) { var uri = updateQueryString('q', $(this).val()); history.pushState({ path: uri }, null, uri); } }); } gitbook.events.on('page.change', function() { bindSearch(); closeSearch(); // Launch search based on query parameter if (gitbook.search.isInitialized()) { launchSearchFromQueryString(); } }); gitbook.events.on('search.ready', function() { bindSearch(); // Launch search from query param at start launchSearchFromQueryString(); }); function getParameterByName(name) { var url = window.location.href; name = name.replace(/[\[\]]/g, '\\$&'); var regex = new RegExp('[?&]' + name + '(=([^&#]*)|&|#|$)', 'i'), results = regex.exec(url); if (!results) return null; if (!results[2]) return ''; return decodeURIComponent(results[2].replace(/\+/g, ' ')); } function updateQueryString(key, value) { value = encodeURIComponent(value); var url = window.location.href; var re = new RegExp('([?&])' + key + '=.*?(&|#|$)(.*)', 'gi'), hash; if (re.test(url)) { if (typeof value !== 'undefined' && value !== null) return url.replace(re, '$1' + key + '=' + value + '$2$3'); else { hash = url.split('#'); url = hash[0].replace(re, '$1$3').replace(/(&|\?)$/, ''); if (typeof hash[1] !== 'undefined' && hash[1] !== null) url += '#' + hash[1]; return url; } } else { if (typeof value !== 'undefined' && value !== null) { var separator = url.indexOf('?') !== -1 ? '&' : '?'; hash = url.split('#'); url = hash[0] + separator + key + '=' + value; if (typeof hash[1] !== 'undefined' && hash[1] !== null) url += '#' + hash[1]; return url; } else return url; } } }); ================================================ FILE: docs/gitbook/gitbook-plugin-sharing/buttons.js ================================================ require(['gitbook', 'jquery'], function(gitbook, $) { var SITES = { 'facebook': { 'label': 'Facebook', 'icon': 'fa fa-facebook', 'onClick': function(e) { e.preventDefault(); window.open('http://www.facebook.com/sharer/sharer.php?s=100&p[url]='+encodeURIComponent(location.href)); } }, 'twitter': { 'label': 'Twitter', 'icon': 'fa fa-twitter', 'onClick': function(e) { e.preventDefault(); window.open('http://twitter.com/home?status='+encodeURIComponent(document.title+' '+location.href)); } }, 'google': { 'label': 'Google+', 'icon': 'fa fa-google-plus', 'onClick': function(e) { e.preventDefault(); window.open('https://plus.google.com/share?url='+encodeURIComponent(location.href)); } }, 'weibo': { 'label': 'Weibo', 'icon': 'fa fa-weibo', 'onClick': function(e) { e.preventDefault(); window.open('http://service.weibo.com/share/share.php?content=utf-8&url='+encodeURIComponent(location.href)+'&title='+encodeURIComponent(document.title)); } }, 'instapaper': { 'label': 'Instapaper', 'icon': 'fa fa-instapaper', 'onClick': function(e) { e.preventDefault(); window.open('http://www.instapaper.com/text?u='+encodeURIComponent(location.href)); } }, 'vk': { 'label': 'VK', 'icon': 'fa fa-vk', 'onClick': function(e) { e.preventDefault(); window.open('http://vkontakte.ru/share.php?url='+encodeURIComponent(location.href)); } } }; gitbook.events.bind('start', function(e, config) { var opts = config.sharing; // Create dropdown menu var menu = $.map(opts.all, function(id) { var site = SITES[id]; return { text: site.label, onClick: site.onClick }; }); // Create main button with dropdown if (menu.length > 0) { gitbook.toolbar.createButton({ icon: 'fa fa-share-alt', label: 'Share', position: 'right', dropdown: [menu] }); } // Direct actions to share $.each(SITES, function(sideId, site) { if (!opts[sideId]) return; gitbook.toolbar.createButton({ icon: site.icon, label: site.text, position: 'right', onClick: site.onClick }); }); }); }); ================================================ FILE: docs/gitbook/gitbook.js ================================================ !function e(t,n,r){function o(s,a){if(!n[s]){if(!t[s]){var u="function"==typeof require&&require;if(!a&&u)return u(s,!0);if(i)return i(s,!0);var c=new Error("Cannot find module '"+s+"'");throw c.code="MODULE_NOT_FOUND",c}var l=n[s]={exports:{}};t[s][0].call(l.exports,function(e){var n=t[s][1][e];return o(n?n:e)},l,l.exports,e,t,n,r)}return n[s].exports}for(var i="function"==typeof require&&require,s=0;s0&&t-1 in e)}function o(e,t,n){return de.isFunction(t)?de.grep(e,function(e,r){return!!t.call(e,r,e)!==n}):t.nodeType?de.grep(e,function(e){return e===t!==n}):"string"!=typeof t?de.grep(e,function(e){return se.call(t,e)>-1!==n}):je.test(t)?de.filter(t,e,n):(t=de.filter(t,e),de.grep(e,function(e){return se.call(t,e)>-1!==n&&1===e.nodeType}))}function i(e,t){for(;(e=e[t])&&1!==e.nodeType;);return e}function s(e){var t={};return de.each(e.match(qe)||[],function(e,n){t[n]=!0}),t}function a(e){return e}function u(e){throw e}function c(e,t,n){var r;try{e&&de.isFunction(r=e.promise)?r.call(e).done(t).fail(n):e&&de.isFunction(r=e.then)?r.call(e,t,n):t.call(void 0,e)}catch(e){n.call(void 0,e)}}function l(){te.removeEventListener("DOMContentLoaded",l),e.removeEventListener("load",l),de.ready()}function f(){this.expando=de.expando+f.uid++}function p(e){return"true"===e||"false"!==e&&("null"===e?null:e===+e+""?+e:Ie.test(e)?JSON.parse(e):e)}function h(e,t,n){var r;if(void 0===n&&1===e.nodeType)if(r="data-"+t.replace(Pe,"-$&").toLowerCase(),n=e.getAttribute(r),"string"==typeof n){try{n=p(n)}catch(e){}Re.set(e,t,n)}else n=void 0;return n}function d(e,t,n,r){var o,i=1,s=20,a=r?function(){return r.cur()}:function(){return de.css(e,t,"")},u=a(),c=n&&n[3]||(de.cssNumber[t]?"":"px"),l=(de.cssNumber[t]||"px"!==c&&+u)&&$e.exec(de.css(e,t));if(l&&l[3]!==c){c=c||l[3],n=n||[],l=+u||1;do i=i||".5",l/=i,de.style(e,t,l+c);while(i!==(i=a()/u)&&1!==i&&--s)}return n&&(l=+l||+u||0,o=n[1]?l+(n[1]+1)*n[2]:+n[2],r&&(r.unit=c,r.start=l,r.end=o)),o}function g(e){var t,n=e.ownerDocument,r=e.nodeName,o=Ue[r];return o?o:(t=n.body.appendChild(n.createElement(r)),o=de.css(t,"display"),t.parentNode.removeChild(t),"none"===o&&(o="block"),Ue[r]=o,o)}function m(e,t){for(var n,r,o=[],i=0,s=e.length;i-1)o&&o.push(i);else if(c=de.contains(i.ownerDocument,i),s=v(f.appendChild(i),"script"),c&&y(s),n)for(l=0;i=s[l++];)Ve.test(i.type||"")&&n.push(i);return f}function b(){return!0}function w(){return!1}function T(){try{return te.activeElement}catch(e){}}function C(e,t,n,r,o,i){var s,a;if("object"==typeof t){"string"!=typeof n&&(r=r||n,n=void 0);for(a in t)C(e,a,n,r,t[a],i);return e}if(null==r&&null==o?(o=n,r=n=void 0):null==o&&("string"==typeof n?(o=r,r=void 0):(o=r,r=n,n=void 0)),o===!1)o=w;else if(!o)return e;return 1===i&&(s=o,o=function(e){return de().off(e),s.apply(this,arguments)},o.guid=s.guid||(s.guid=de.guid++)),e.each(function(){de.event.add(this,t,o,r,n)})}function j(e,t){return de.nodeName(e,"table")&&de.nodeName(11!==t.nodeType?t:t.firstChild,"tr")?e.getElementsByTagName("tbody")[0]||e:e}function k(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function E(e){var t=rt.exec(e.type);return t?e.type=t[1]:e.removeAttribute("type"),e}function S(e,t){var n,r,o,i,s,a,u,c;if(1===t.nodeType){if(Fe.hasData(e)&&(i=Fe.access(e),s=Fe.set(t,i),c=i.events)){delete s.handle,s.events={};for(o in c)for(n=0,r=c[o].length;n1&&"string"==typeof d&&!pe.checkClone&&nt.test(d))return e.each(function(n){var i=e.eq(n);g&&(t[0]=d.call(this,n,i.html())),A(i,t,r,o)});if(p&&(i=x(t,e[0].ownerDocument,!1,e,o),s=i.firstChild,1===i.childNodes.length&&(i=s),s||o)){for(a=de.map(v(i,"script"),k),u=a.length;f=0&&nC.cacheLength&&delete e[t.shift()],e[n+" "]=r}var t=[];return e}function r(e){return e[$]=!0,e}function o(e){var t=L.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function i(e,t){for(var n=e.split("|"),r=n.length;r--;)C.attrHandle[n[r]]=t}function s(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)for(;n=n.nextSibling;)if(n===t)return-1;return e?1:-1}function a(e){return function(t){var n=t.nodeName.toLowerCase();return"input"===n&&t.type===e}}function u(e){return function(t){var n=t.nodeName.toLowerCase();return("input"===n||"button"===n)&&t.type===e}}function c(e){return function(t){return"form"in t?t.parentNode&&t.disabled===!1?"label"in t?"label"in t.parentNode?t.parentNode.disabled===e:t.disabled===e:t.isDisabled===e||t.isDisabled!==!e&&je(t)===e:t.disabled===e:"label"in t&&t.disabled===e}}function l(e){return r(function(t){return t=+t,r(function(n,r){for(var o,i=e([],n.length,t),s=i.length;s--;)n[o=i[s]]&&(n[o]=!(r[o]=n[o]))})})}function f(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}function p(){}function h(e){for(var t=0,n=e.length,r="";t1?function(t,n,r){for(var o=e.length;o--;)if(!e[o](t,n,r))return!1;return!0}:e[0]}function m(e,n,r){for(var o=0,i=n.length;o-1&&(r[c]=!(s[c]=f))}}else x=v(x===s?x.splice(d,x.length):x),i?i(null,s,x,u):K.apply(s,x)})}function x(e){for(var t,n,r,o=e.length,i=C.relative[e[0].type],s=i||C.relative[" "],a=i?1:0,u=d(function(e){return e===t},s,!0),c=d(function(e){return ee(t,e)>-1},s,!0),l=[function(e,n,r){var o=!i&&(r||n!==A)||((t=n).nodeType?u(e,n,r):c(e,n,r));return t=null,o}];a1&&g(l),a>1&&h(e.slice(0,a-1).concat({value:" "===e[a-2].type?"*":""})).replace(ae,"$1"),n,a0,i=e.length>0,s=function(r,s,a,u,c){var l,f,p,h=0,d="0",g=r&&[],m=[],y=A,x=r||i&&C.find.TAG("*",c),b=B+=null==y?1:Math.random()||.1,w=x.length;for(c&&(A=s===L||s||c);d!==w&&null!=(l=x[d]);d++){if(i&&l){for(f=0,s||l.ownerDocument===L||(O(l),a=!F);p=e[f++];)if(p(l,s||L,a)){u.push(l);break}c&&(B=b)}o&&((l=!p&&l)&&h--,r&&g.push(l))}if(h+=d,o&&d!==h){for(f=0;p=n[f++];)p(g,m,s,a);if(r){if(h>0)for(;d--;)g[d]||m[d]||(m[d]=Q.call(u));m=v(m)}K.apply(u,m),c&&!r&&m.length>0&&h+n.length>1&&t.uniqueSort(u)}return c&&(B=b,A=y),g};return o?r(s):s}var w,T,C,j,k,E,S,N,A,q,D,O,L,H,F,R,I,P,M,$="sizzle"+1*new Date,W=e.document,B=0,_=0,U=n(),z=n(),X=n(),V=function(e,t){return e===t&&(D=!0),0},G={}.hasOwnProperty,Y=[],Q=Y.pop,J=Y.push,K=Y.push,Z=Y.slice,ee=function(e,t){for(var n=0,r=e.length;n+~]|"+ne+")"+ne+"*"),le=new RegExp("="+ne+"*([^\\]'\"]*?)"+ne+"*\\]","g"),fe=new RegExp(ie),pe=new RegExp("^"+re+"$"),he={ID:new RegExp("^#("+re+")"),CLASS:new RegExp("^\\.("+re+")"),TAG:new RegExp("^("+re+"|[*])"),ATTR:new RegExp("^"+oe),PSEUDO:new RegExp("^"+ie),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+ne+"*(even|odd|(([+-]|)(\\d*)n|)"+ne+"*(?:([+-]|)"+ne+"*(\\d+)|))"+ne+"*\\)|)","i"),bool:new RegExp("^(?:"+te+")$","i"),needsContext:new RegExp("^"+ne+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+ne+"*((?:-\\d)?\\d*)"+ne+"*\\)|)(?=[^-]|$)","i")},de=/^(?:input|select|textarea|button)$/i,ge=/^h\d$/i,me=/^[^{]+\{\s*\[native \w/,ve=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ye=/[+~]/,xe=new RegExp("\\\\([\\da-f]{1,6}"+ne+"?|("+ne+")|.)","ig"),be=function(e,t,n){var r="0x"+t-65536;return r!==r||n?t:r<0?String.fromCharCode(r+65536):String.fromCharCode(r>>10|55296,1023&r|56320)},we=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,Te=function(e,t){return t?"\0"===e?"�":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e},Ce=function(){O()},je=d(function(e){return e.disabled===!0&&("form"in e||"label"in e)},{dir:"parentNode",next:"legend"});try{K.apply(Y=Z.call(W.childNodes),W.childNodes),Y[W.childNodes.length].nodeType}catch(e){K={apply:Y.length?function(e,t){J.apply(e,Z.call(t))}:function(e,t){for(var n=e.length,r=0;e[n++]=t[r++];);e.length=n-1}}}T=t.support={},k=t.isXML=function(e){var t=e&&(e.ownerDocument||e).documentElement;return!!t&&"HTML"!==t.nodeName},O=t.setDocument=function(e){var t,n,r=e?e.ownerDocument||e:W;return r!==L&&9===r.nodeType&&r.documentElement?(L=r,H=L.documentElement,F=!k(L),W!==L&&(n=L.defaultView)&&n.top!==n&&(n.addEventListener?n.addEventListener("unload",Ce,!1):n.attachEvent&&n.attachEvent("onunload",Ce)),T.attributes=o(function(e){return e.className="i",!e.getAttribute("className")}),T.getElementsByTagName=o(function(e){return e.appendChild(L.createComment("")),!e.getElementsByTagName("*").length}),T.getElementsByClassName=me.test(L.getElementsByClassName),T.getById=o(function(e){return H.appendChild(e).id=$,!L.getElementsByName||!L.getElementsByName($).length}),T.getById?(C.filter.ID=function(e){var t=e.replace(xe,be);return function(e){return e.getAttribute("id")===t}},C.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&F){var n=t.getElementById(e);return n?[n]:[]}}):(C.filter.ID=function(e){var t=e.replace(xe,be);return function(e){var n="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return n&&n.value===t}},C.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&F){var n,r,o,i=t.getElementById(e);if(i){if(n=i.getAttributeNode("id"),n&&n.value===e)return[i];for(o=t.getElementsByName(e),r=0;i=o[r++];)if(n=i.getAttributeNode("id"),n&&n.value===e)return[i]}return[]}}),C.find.TAG=T.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):T.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],o=0,i=t.getElementsByTagName(e);if("*"===e){for(;n=i[o++];)1===n.nodeType&&r.push(n);return r}return i},C.find.CLASS=T.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&F)return t.getElementsByClassName(e)},I=[],R=[],(T.qsa=me.test(L.querySelectorAll))&&(o(function(e){H.appendChild(e).innerHTML="",e.querySelectorAll("[msallowcapture^='']").length&&R.push("[*^$]="+ne+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||R.push("\\["+ne+"*(?:value|"+te+")"),e.querySelectorAll("[id~="+$+"-]").length||R.push("~="),e.querySelectorAll(":checked").length||R.push(":checked"),e.querySelectorAll("a#"+$+"+*").length||R.push(".#.+[+~]")}),o(function(e){e.innerHTML="";var t=L.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&R.push("name"+ne+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&R.push(":enabled",":disabled"),H.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&R.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),R.push(",.*:")})),(T.matchesSelector=me.test(P=H.matches||H.webkitMatchesSelector||H.mozMatchesSelector||H.oMatchesSelector||H.msMatchesSelector))&&o(function(e){T.disconnectedMatch=P.call(e,"*"),P.call(e,"[s!='']:x"),I.push("!=",ie)}),R=R.length&&new RegExp(R.join("|")),I=I.length&&new RegExp(I.join("|")),t=me.test(H.compareDocumentPosition),M=t||me.test(H.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)for(;t=t.parentNode;)if(t===e)return!0;return!1},V=t?function(e,t){if(e===t)return D=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n?n:(n=(e.ownerDocument||e)===(t.ownerDocument||t)?e.compareDocumentPosition(t):1,1&n||!T.sortDetached&&t.compareDocumentPosition(e)===n?e===L||e.ownerDocument===W&&M(W,e)?-1:t===L||t.ownerDocument===W&&M(W,t)?1:q?ee(q,e)-ee(q,t):0:4&n?-1:1)}:function(e,t){if(e===t)return D=!0,0;var n,r=0,o=e.parentNode,i=t.parentNode,a=[e],u=[t];if(!o||!i)return e===L?-1:t===L?1:o?-1:i?1:q?ee(q,e)-ee(q,t):0;if(o===i)return s(e,t);for(n=e;n=n.parentNode;)a.unshift(n);for(n=t;n=n.parentNode;)u.unshift(n);for(;a[r]===u[r];)r++;return r?s(a[r],u[r]):a[r]===W?-1:u[r]===W?1:0},L):L},t.matches=function(e,n){return t(e,null,null,n)},t.matchesSelector=function(e,n){if((e.ownerDocument||e)!==L&&O(e),n=n.replace(le,"='$1']"),T.matchesSelector&&F&&!X[n+" "]&&(!I||!I.test(n))&&(!R||!R.test(n)))try{var r=P.call(e,n);if(r||T.disconnectedMatch||e.document&&11!==e.document.nodeType)return r}catch(e){}return t(n,L,null,[e]).length>0},t.contains=function(e,t){return(e.ownerDocument||e)!==L&&O(e),M(e,t)},t.attr=function(e,t){(e.ownerDocument||e)!==L&&O(e);var n=C.attrHandle[t.toLowerCase()],r=n&&G.call(C.attrHandle,t.toLowerCase())?n(e,t,!F):void 0;return void 0!==r?r:T.attributes||!F?e.getAttribute(t):(r=e.getAttributeNode(t))&&r.specified?r.value:null},t.escape=function(e){return(e+"").replace(we,Te)},t.error=function(e){throw new Error("Syntax error, unrecognized expression: "+e)},t.uniqueSort=function(e){var t,n=[],r=0,o=0;if(D=!T.detectDuplicates,q=!T.sortStable&&e.slice(0),e.sort(V),D){for(;t=e[o++];)t===e[o]&&(r=n.push(o));for(;r--;)e.splice(n[r],1)}return q=null,e},j=t.getText=function(e){var t,n="",r=0,o=e.nodeType;if(o){if(1===o||9===o||11===o){if("string"==typeof e.textContent)return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=j(e)}else if(3===o||4===o)return e.nodeValue}else for(;t=e[r++];)n+=j(t);return n},C=t.selectors={cacheLength:50,createPseudo:r,match:he,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(xe,be),e[3]=(e[3]||e[4]||e[5]||"").replace(xe,be),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||t.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&t.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return he.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&fe.test(n)&&(t=E(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(xe,be).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=U[e+" "];return t||(t=new RegExp("(^|"+ne+")"+e+"("+ne+"|$)"))&&U(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(e,n,r){return function(o){var i=t.attr(o,e);return null==i?"!="===n:!n||(i+="","="===n?i===r:"!="===n?i!==r:"^="===n?r&&0===i.indexOf(r):"*="===n?r&&i.indexOf(r)>-1:"$="===n?r&&i.slice(-r.length)===r:"~="===n?(" "+i.replace(se," ")+" ").indexOf(r)>-1:"|="===n&&(i===r||i.slice(0,r.length+1)===r+"-"))}},CHILD:function(e,t,n,r,o){var i="nth"!==e.slice(0,3),s="last"!==e.slice(-4),a="of-type"===t;return 1===r&&0===o?function(e){return!!e.parentNode}:function(t,n,u){var c,l,f,p,h,d,g=i!==s?"nextSibling":"previousSibling",m=t.parentNode,v=a&&t.nodeName.toLowerCase(),y=!u&&!a,x=!1;if(m){if(i){for(;g;){for(p=t;p=p[g];)if(a?p.nodeName.toLowerCase()===v:1===p.nodeType)return!1;d=g="only"===e&&!d&&"nextSibling"}return!0}if(d=[s?m.firstChild:m.lastChild],s&&y){for(p=m,f=p[$]||(p[$]={}),l=f[p.uniqueID]||(f[p.uniqueID]={}),c=l[e]||[],h=c[0]===B&&c[1],x=h&&c[2],p=h&&m.childNodes[h];p=++h&&p&&p[g]||(x=h=0)||d.pop();)if(1===p.nodeType&&++x&&p===t){l[e]=[B,h,x];break}}else if(y&&(p=t,f=p[$]||(p[$]={}),l=f[p.uniqueID]||(f[p.uniqueID]={}),c=l[e]||[],h=c[0]===B&&c[1],x=h),x===!1)for(;(p=++h&&p&&p[g]||(x=h=0)||d.pop())&&((a?p.nodeName.toLowerCase()!==v:1!==p.nodeType)||!++x||(y&&(f=p[$]||(p[$]={}),l=f[p.uniqueID]||(f[p.uniqueID]={}),l[e]=[B,x]),p!==t)););return x-=o,x===r||x%r===0&&x/r>=0}}},PSEUDO:function(e,n){var o,i=C.pseudos[e]||C.setFilters[e.toLowerCase()]||t.error("unsupported pseudo: "+e);return i[$]?i(n):i.length>1?(o=[e,e,"",n],C.setFilters.hasOwnProperty(e.toLowerCase())?r(function(e,t){for(var r,o=i(e,n),s=o.length;s--;)r=ee(e,o[s]),e[r]=!(t[r]=o[s])}):function(e){return i(e,0,o)}):i}},pseudos:{not:r(function(e){var t=[],n=[],o=S(e.replace(ae,"$1"));return o[$]?r(function(e,t,n,r){for(var i,s=o(e,null,r,[]),a=e.length;a--;)(i=s[a])&&(e[a]=!(t[a]=i))}):function(e,r,i){return t[0]=e,o(t,null,i,n),t[0]=null,!n.pop()}}),has:r(function(e){return function(n){ return t(e,n).length>0}}),contains:r(function(e){return e=e.replace(xe,be),function(t){return(t.textContent||t.innerText||j(t)).indexOf(e)>-1}}),lang:r(function(e){return pe.test(e||"")||t.error("unsupported lang: "+e),e=e.replace(xe,be).toLowerCase(),function(t){var n;do if(n=F?t.lang:t.getAttribute("xml:lang")||t.getAttribute("lang"))return n=n.toLowerCase(),n===e||0===n.indexOf(e+"-");while((t=t.parentNode)&&1===t.nodeType);return!1}}),target:function(t){var n=e.location&&e.location.hash;return n&&n.slice(1)===t.id},root:function(e){return e===H},focus:function(e){return e===L.activeElement&&(!L.hasFocus||L.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},enabled:c(!1),disabled:c(!0),checked:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&!!e.checked||"option"===t&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,e.selected===!0},empty:function(e){for(e=e.firstChild;e;e=e.nextSibling)if(e.nodeType<6)return!1;return!0},parent:function(e){return!C.pseudos.empty(e)},header:function(e){return ge.test(e.nodeName)},input:function(e){return de.test(e.nodeName)},button:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&"button"===e.type||"button"===t},text:function(e){var t;return"input"===e.nodeName.toLowerCase()&&"text"===e.type&&(null==(t=e.getAttribute("type"))||"text"===t.toLowerCase())},first:l(function(){return[0]}),last:l(function(e,t){return[t-1]}),eq:l(function(e,t,n){return[n<0?n+t:n]}),even:l(function(e,t){for(var n=0;n=0;)e.push(r);return e}),gt:l(function(e,t,n){for(var r=n<0?n+t:n;++r2&&"ID"===(s=i[0]).type&&9===t.nodeType&&F&&C.relative[i[1].type]){if(t=(C.find.ID(s.matches[0].replace(xe,be),t)||[])[0],!t)return n;c&&(t=t.parentNode),e=e.slice(i.shift().value.length)}for(o=he.needsContext.test(e)?0:i.length;o--&&(s=i[o],!C.relative[a=s.type]);)if((u=C.find[a])&&(r=u(s.matches[0].replace(xe,be),ye.test(i[0].type)&&f(t.parentNode)||t))){if(i.splice(o,1),e=r.length&&h(i),!e)return K.apply(n,r),n;break}}return(c||S(e,l))(r,t,!F,n,!t||ye.test(e)&&f(t.parentNode)||t),n},T.sortStable=$.split("").sort(V).join("")===$,T.detectDuplicates=!!D,O(),T.sortDetached=o(function(e){return 1&e.compareDocumentPosition(L.createElement("fieldset"))}),o(function(e){return e.innerHTML="","#"===e.firstChild.getAttribute("href")})||i("type|href|height|width",function(e,t,n){if(!n)return e.getAttribute(t,"type"===t.toLowerCase()?1:2)}),T.attributes&&o(function(e){return e.innerHTML="",e.firstChild.setAttribute("value",""),""===e.firstChild.getAttribute("value")})||i("value",function(e,t,n){if(!n&&"input"===e.nodeName.toLowerCase())return e.defaultValue}),o(function(e){return null==e.getAttribute("disabled")})||i(te,function(e,t,n){var r;if(!n)return e[t]===!0?t.toLowerCase():(r=e.getAttributeNode(t))&&r.specified?r.value:null}),t}(e);de.find=xe,de.expr=xe.selectors,de.expr[":"]=de.expr.pseudos,de.uniqueSort=de.unique=xe.uniqueSort,de.text=xe.getText,de.isXMLDoc=xe.isXML,de.contains=xe.contains,de.escapeSelector=xe.escape;var be=function(e,t,n){for(var r=[],o=void 0!==n;(e=e[t])&&9!==e.nodeType;)if(1===e.nodeType){if(o&&de(e).is(n))break;r.push(e)}return r},we=function(e,t){for(var n=[];e;e=e.nextSibling)1===e.nodeType&&e!==t&&n.push(e);return n},Te=de.expr.match.needsContext,Ce=/^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i,je=/^.[^:#\[\.,]*$/;de.filter=function(e,t,n){var r=t[0];return n&&(e=":not("+e+")"),1===t.length&&1===r.nodeType?de.find.matchesSelector(r,e)?[r]:[]:de.find.matches(e,de.grep(t,function(e){return 1===e.nodeType}))},de.fn.extend({find:function(e){var t,n,r=this.length,o=this;if("string"!=typeof e)return this.pushStack(de(e).filter(function(){for(t=0;t1?de.uniqueSort(n):n},filter:function(e){return this.pushStack(o(this,e||[],!1))},not:function(e){return this.pushStack(o(this,e||[],!0))},is:function(e){return!!o(this,"string"==typeof e&&Te.test(e)?de(e):e||[],!1).length}});var ke,Ee=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,Se=de.fn.init=function(e,t,n){var r,o;if(!e)return this;if(n=n||ke,"string"==typeof e){if(r="<"===e[0]&&">"===e[e.length-1]&&e.length>=3?[null,e,null]:Ee.exec(e),!r||!r[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(r[1]){if(t=t instanceof de?t[0]:t,de.merge(this,de.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:te,!0)),Ce.test(r[1])&&de.isPlainObject(t))for(r in t)de.isFunction(this[r])?this[r](t[r]):this.attr(r,t[r]);return this}return o=te.getElementById(r[2]),o&&(this[0]=o,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):de.isFunction(e)?void 0!==n.ready?n.ready(e):e(de):de.makeArray(e,this)};Se.prototype=de.fn,ke=de(te);var Ne=/^(?:parents|prev(?:Until|All))/,Ae={children:!0,contents:!0,next:!0,prev:!0};de.fn.extend({has:function(e){var t=de(e,this),n=t.length;return this.filter(function(){for(var e=0;e-1:1===n.nodeType&&de.find.matchesSelector(n,e))){i.push(n);break}return this.pushStack(i.length>1?de.uniqueSort(i):i)},index:function(e){return e?"string"==typeof e?se.call(de(e),this[0]):se.call(this,e.jquery?e[0]:e):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(e,t){return this.pushStack(de.uniqueSort(de.merge(this.get(),de(e,t))))},addBack:function(e){return this.add(null==e?this.prevObject:this.prevObject.filter(e))}}),de.each({parent:function(e){var t=e.parentNode;return t&&11!==t.nodeType?t:null},parents:function(e){return be(e,"parentNode")},parentsUntil:function(e,t,n){return be(e,"parentNode",n)},next:function(e){return i(e,"nextSibling")},prev:function(e){return i(e,"previousSibling")},nextAll:function(e){return be(e,"nextSibling")},prevAll:function(e){return be(e,"previousSibling")},nextUntil:function(e,t,n){return be(e,"nextSibling",n)},prevUntil:function(e,t,n){return be(e,"previousSibling",n)},siblings:function(e){return we((e.parentNode||{}).firstChild,e)},children:function(e){return we(e.firstChild)},contents:function(e){return e.contentDocument||de.merge([],e.childNodes)}},function(e,t){de.fn[e]=function(n,r){var o=de.map(this,t,n);return"Until"!==e.slice(-5)&&(r=n),r&&"string"==typeof r&&(o=de.filter(r,o)),this.length>1&&(Ae[e]||de.uniqueSort(o),Ne.test(e)&&o.reverse()),this.pushStack(o)}});var qe=/[^\x20\t\r\n\f]+/g;de.Callbacks=function(e){e="string"==typeof e?s(e):de.extend({},e);var t,n,r,o,i=[],a=[],u=-1,c=function(){for(o=e.once,r=t=!0;a.length;u=-1)for(n=a.shift();++u-1;)i.splice(n,1),n<=u&&u--}),this},has:function(e){return e?de.inArray(e,i)>-1:i.length>0},empty:function(){return i&&(i=[]),this},disable:function(){return o=a=[],i=n="",this},disabled:function(){return!i},lock:function(){return o=a=[],n||t||(i=n=""),this},locked:function(){return!!o},fireWith:function(e,n){return o||(n=n||[],n=[e,n.slice?n.slice():n],a.push(n),t||c()),this},fire:function(){return l.fireWith(this,arguments),this},fired:function(){return!!r}};return l},de.extend({Deferred:function(t){var n=[["notify","progress",de.Callbacks("memory"),de.Callbacks("memory"),2],["resolve","done",de.Callbacks("once memory"),de.Callbacks("once memory"),0,"resolved"],["reject","fail",de.Callbacks("once memory"),de.Callbacks("once memory"),1,"rejected"]],r="pending",o={state:function(){return r},always:function(){return i.done(arguments).fail(arguments),this},catch:function(e){return o.then(null,e)},pipe:function(){var e=arguments;return de.Deferred(function(t){de.each(n,function(n,r){var o=de.isFunction(e[r[4]])&&e[r[4]];i[r[1]](function(){var e=o&&o.apply(this,arguments);e&&de.isFunction(e.promise)?e.promise().progress(t.notify).done(t.resolve).fail(t.reject):t[r[0]+"With"](this,o?[e]:arguments)})}),e=null}).promise()},then:function(t,r,o){function i(t,n,r,o){return function(){var c=this,l=arguments,f=function(){var e,f;if(!(t=s&&(r!==u&&(c=void 0,l=[e]),n.rejectWith(c,l))}};t?p():(de.Deferred.getStackHook&&(p.stackTrace=de.Deferred.getStackHook()),e.setTimeout(p))}}var s=0;return de.Deferred(function(e){n[0][3].add(i(0,e,de.isFunction(o)?o:a,e.notifyWith)),n[1][3].add(i(0,e,de.isFunction(t)?t:a)),n[2][3].add(i(0,e,de.isFunction(r)?r:u))}).promise()},promise:function(e){return null!=e?de.extend(e,o):o}},i={};return de.each(n,function(e,t){var s=t[2],a=t[5];o[t[1]]=s.add,a&&s.add(function(){r=a},n[3-e][2].disable,n[0][2].lock),s.add(t[3].fire),i[t[0]]=function(){return i[t[0]+"With"](this===i?void 0:this,arguments),this},i[t[0]+"With"]=s.fireWith}),o.promise(i),t&&t.call(i,i),i},when:function(e){var t=arguments.length,n=t,r=Array(n),o=re.call(arguments),i=de.Deferred(),s=function(e){return function(n){r[e]=this,o[e]=arguments.length>1?re.call(arguments):n,--t||i.resolveWith(r,o)}};if(t<=1&&(c(e,i.done(s(n)).resolve,i.reject),"pending"===i.state()||de.isFunction(o[n]&&o[n].then)))return i.then();for(;n--;)c(o[n],s(n),i.reject);return i.promise()}});var De=/^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;de.Deferred.exceptionHook=function(t,n){e.console&&e.console.warn&&t&&De.test(t.name)&&e.console.warn("jQuery.Deferred exception: "+t.message,t.stack,n)},de.readyException=function(t){e.setTimeout(function(){throw t})};var Oe=de.Deferred();de.fn.ready=function(e){return Oe.then(e).catch(function(e){de.readyException(e)}),this},de.extend({isReady:!1,readyWait:1,holdReady:function(e){e?de.readyWait++:de.ready(!0)},ready:function(e){(e===!0?--de.readyWait:de.isReady)||(de.isReady=!0,e!==!0&&--de.readyWait>0||Oe.resolveWith(te,[de]))}}),de.ready.then=Oe.then,"complete"===te.readyState||"loading"!==te.readyState&&!te.documentElement.doScroll?e.setTimeout(de.ready):(te.addEventListener("DOMContentLoaded",l),e.addEventListener("load",l));var Le=function(e,t,n,r,o,i,s){var a=0,u=e.length,c=null==n;if("object"===de.type(n)){o=!0;for(a in n)Le(e,t,a,n[a],!0,i,s)}else if(void 0!==r&&(o=!0,de.isFunction(r)||(s=!0),c&&(s?(t.call(e,r),t=null):(c=t,t=function(e,t,n){return c.call(de(e),n)})),t))for(;a1,null,!0)},removeData:function(e){return this.each(function(){Re.remove(this,e)})}}),de.extend({queue:function(e,t,n){var r;if(e)return t=(t||"fx")+"queue",r=Fe.get(e,t),n&&(!r||de.isArray(n)?r=Fe.access(e,t,de.makeArray(n)):r.push(n)),r||[]},dequeue:function(e,t){t=t||"fx";var n=de.queue(e,t),r=n.length,o=n.shift(),i=de._queueHooks(e,t),s=function(){de.dequeue(e,t)};"inprogress"===o&&(o=n.shift(),r--),o&&("fx"===t&&n.unshift("inprogress"),delete i.stop,o.call(e,s,i)),!r&&i&&i.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return Fe.get(e,n)||Fe.access(e,n,{empty:de.Callbacks("once memory").add(function(){Fe.remove(e,[t+"queue",n])})})}}),de.fn.extend({queue:function(e,t){var n=2;return"string"!=typeof e&&(t=e,e="fx",n--),arguments.length\x20\t\r\n\f]+)/i,Ve=/^$|\/(?:java|ecma)script/i,Ge={option:[1,""],thead:[1,"","
      "],col:[2,"","
      "],tr:[2,"","
      "],td:[3,"","
      "],_default:[0,"",""]};Ge.optgroup=Ge.option,Ge.tbody=Ge.tfoot=Ge.colgroup=Ge.caption=Ge.thead,Ge.th=Ge.td;var Ye=/<|&#?\w+;/;!function(){var e=te.createDocumentFragment(),t=e.appendChild(te.createElement("div")),n=te.createElement("input");n.setAttribute("type","radio"),n.setAttribute("checked","checked"),n.setAttribute("name","t"),t.appendChild(n),pe.checkClone=t.cloneNode(!0).cloneNode(!0).lastChild.checked,t.innerHTML="",pe.noCloneChecked=!!t.cloneNode(!0).lastChild.defaultValue}();var Qe=te.documentElement,Je=/^key/,Ke=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,Ze=/^([^.]*)(?:\.(.+)|)/;de.event={global:{},add:function(e,t,n,r,o){var i,s,a,u,c,l,f,p,h,d,g,m=Fe.get(e);if(m)for(n.handler&&(i=n,n=i.handler,o=i.selector),o&&de.find.matchesSelector(Qe,o),n.guid||(n.guid=de.guid++),(u=m.events)||(u=m.events={}),(s=m.handle)||(s=m.handle=function(t){return"undefined"!=typeof de&&de.event.triggered!==t.type?de.event.dispatch.apply(e,arguments):void 0}),t=(t||"").match(qe)||[""],c=t.length;c--;)a=Ze.exec(t[c])||[],h=g=a[1],d=(a[2]||"").split(".").sort(),h&&(f=de.event.special[h]||{},h=(o?f.delegateType:f.bindType)||h,f=de.event.special[h]||{},l=de.extend({type:h,origType:g,data:r,handler:n,guid:n.guid,selector:o,needsContext:o&&de.expr.match.needsContext.test(o),namespace:d.join(".")},i),(p=u[h])||(p=u[h]=[],p.delegateCount=0,f.setup&&f.setup.call(e,r,d,s)!==!1||e.addEventListener&&e.addEventListener(h,s)),f.add&&(f.add.call(e,l),l.handler.guid||(l.handler.guid=n.guid)),o?p.splice(p.delegateCount++,0,l):p.push(l),de.event.global[h]=!0)},remove:function(e,t,n,r,o){var i,s,a,u,c,l,f,p,h,d,g,m=Fe.hasData(e)&&Fe.get(e);if(m&&(u=m.events)){for(t=(t||"").match(qe)||[""],c=t.length;c--;)if(a=Ze.exec(t[c])||[],h=g=a[1],d=(a[2]||"").split(".").sort(),h){for(f=de.event.special[h]||{},h=(r?f.delegateType:f.bindType)||h,p=u[h]||[],a=a[2]&&new RegExp("(^|\\.)"+d.join("\\.(?:.*\\.|)")+"(\\.|$)"),s=i=p.length;i--;)l=p[i],!o&&g!==l.origType||n&&n.guid!==l.guid||a&&!a.test(l.namespace)||r&&r!==l.selector&&("**"!==r||!l.selector)||(p.splice(i,1),l.selector&&p.delegateCount--,f.remove&&f.remove.call(e,l));s&&!p.length&&(f.teardown&&f.teardown.call(e,d,m.handle)!==!1||de.removeEvent(e,h,m.handle),delete u[h])}else for(h in u)de.event.remove(e,h+t[c],n,r,!0);de.isEmptyObject(u)&&Fe.remove(e,"handle events")}},dispatch:function(e){var t,n,r,o,i,s,a=de.event.fix(e),u=new Array(arguments.length),c=(Fe.get(this,"events")||{})[a.type]||[],l=de.event.special[a.type]||{};for(u[0]=a,t=1;t=1))for(;c!==this;c=c.parentNode||this)if(1===c.nodeType&&("click"!==e.type||c.disabled!==!0)){for(i=[],s={},n=0;n-1:de.find(o,this,null,[c]).length),s[o]&&i.push(r);i.length&&a.push({elem:c,handlers:i})}return c=this,u\x20\t\r\n\f]*)[^>]*)\/>/gi,tt=/\s*$/g;de.extend({htmlPrefilter:function(e){return e.replace(et,"<$1>")},clone:function(e,t,n){var r,o,i,s,a=e.cloneNode(!0),u=de.contains(e.ownerDocument,e);if(!(pe.noCloneChecked||1!==e.nodeType&&11!==e.nodeType||de.isXMLDoc(e)))for(s=v(a),i=v(e),r=0,o=i.length;r0&&y(s,!u&&v(e,"script")),a},cleanData:function(e){for(var t,n,r,o=de.event.special,i=0;void 0!==(n=e[i]);i++)if(He(n)){if(t=n[Fe.expando]){if(t.events)for(r in t.events)o[r]?de.event.remove(n,r):de.removeEvent(n,r,t.handle);n[Fe.expando]=void 0}n[Re.expando]&&(n[Re.expando]=void 0)}}}),de.fn.extend({detach:function(e){return q(this,e,!0)},remove:function(e){return q(this,e)},text:function(e){return Le(this,function(e){return void 0===e?de.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=e)})},null,e,arguments.length)},append:function(){return A(this,arguments,function(e){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var t=j(this,e);t.appendChild(e)}})},prepend:function(){return A(this,arguments,function(e){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var t=j(this,e);t.insertBefore(e,t.firstChild)}})},before:function(){return A(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this)})},after:function(){return A(this,arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this.nextSibling)})},empty:function(){for(var e,t=0;null!=(e=this[t]);t++)1===e.nodeType&&(de.cleanData(v(e,!1)),e.textContent="");return this},clone:function(e,t){return e=null!=e&&e,t=null==t?e:t,this.map(function(){return de.clone(this,e,t)})},html:function(e){return Le(this,function(e){var t=this[0]||{},n=0,r=this.length;if(void 0===e&&1===t.nodeType)return t.innerHTML;if("string"==typeof e&&!tt.test(e)&&!Ge[(Xe.exec(e)||["",""])[1].toLowerCase()]){e=de.htmlPrefilter(e);try{for(;n1)}}),de.Tween=I,I.prototype={constructor:I,init:function(e,t,n,r,o,i){this.elem=e,this.prop=n,this.easing=o||de.easing._default,this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=i||(de.cssNumber[n]?"":"px")},cur:function(){var e=I.propHooks[this.prop];return e&&e.get?e.get(this):I.propHooks._default.get(this)},run:function(e){var t,n=I.propHooks[this.prop];return this.options.duration?this.pos=t=de.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):this.pos=t=e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),n&&n.set?n.set(this):I.propHooks._default.set(this),this}},I.prototype.init.prototype=I.prototype,I.propHooks={_default:{get:function(e){var t;return 1!==e.elem.nodeType||null!=e.elem[e.prop]&&null==e.elem.style[e.prop]?e.elem[e.prop]:(t=de.css(e.elem,e.prop,""),t&&"auto"!==t?t:0)},set:function(e){de.fx.step[e.prop]?de.fx.step[e.prop](e):1!==e.elem.nodeType||null==e.elem.style[de.cssProps[e.prop]]&&!de.cssHooks[e.prop]?e.elem[e.prop]=e.now:de.style(e.elem,e.prop,e.now+e.unit)}}},I.propHooks.scrollTop=I.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},de.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2},_default:"swing"},de.fx=I.prototype.init,de.fx.step={};var ht,dt,gt=/^(?:toggle|show|hide)$/,mt=/queueHooks$/;de.Animation=de.extend(U,{tweeners:{"*":[function(e,t){var n=this.createTween(e,t);return d(n.elem,e,$e.exec(t),n),n}]},tweener:function(e,t){de.isFunction(e)?(t=e,e=["*"]):e=e.match(qe);for(var n,r=0,o=e.length;r1)},removeAttr:function(e){return this.each(function(){de.removeAttr(this,e)})}}),de.extend({attr:function(e,t,n){var r,o,i=e.nodeType;if(3!==i&&8!==i&&2!==i)return"undefined"==typeof e.getAttribute?de.prop(e,t,n):(1===i&&de.isXMLDoc(e)||(o=de.attrHooks[t.toLowerCase()]||(de.expr.match.bool.test(t)?vt:void 0)),void 0!==n?null===n?void de.removeAttr(e,t):o&&"set"in o&&void 0!==(r=o.set(e,n,t))?r:(e.setAttribute(t,n+""),n):o&&"get"in o&&null!==(r=o.get(e,t))?r:(r=de.find.attr(e,t),null==r?void 0:r))},attrHooks:{type:{set:function(e,t){if(!pe.radioValue&&"radio"===t&&de.nodeName(e,"input")){var n=e.value;return e.setAttribute("type",t),n&&(e.value=n),t}}}},removeAttr:function(e,t){var n,r=0,o=t&&t.match(qe);if(o&&1===e.nodeType)for(;n=o[r++];)e.removeAttribute(n)}}),vt={set:function(e,t,n){return t===!1?de.removeAttr(e,n):e.setAttribute(n,n),n}},de.each(de.expr.match.bool.source.match(/\w+/g),function(e,t){var n=yt[t]||de.find.attr;yt[t]=function(e,t,r){var o,i,s=t.toLowerCase();return r||(i=yt[s],yt[s]=o,o=null!=n(e,t,r)?s:null,yt[s]=i),o}});var xt=/^(?:input|select|textarea|button)$/i,bt=/^(?:a|area)$/i;de.fn.extend({prop:function(e,t){return Le(this,de.prop,e,t,arguments.length>1)},removeProp:function(e){return this.each(function(){delete this[de.propFix[e]||e]})}}),de.extend({prop:function(e,t,n){var r,o,i=e.nodeType;if(3!==i&&8!==i&&2!==i)return 1===i&&de.isXMLDoc(e)||(t=de.propFix[t]||t,o=de.propHooks[t]),void 0!==n?o&&"set"in o&&void 0!==(r=o.set(e,n,t))?r:e[t]=n:o&&"get"in o&&null!==(r=o.get(e,t))?r:e[t]},propHooks:{tabIndex:{get:function(e){var t=de.find.attr(e,"tabindex");return t?parseInt(t,10):xt.test(e.nodeName)||bt.test(e.nodeName)&&e.href?0:-1}}},propFix:{for:"htmlFor",class:"className"}}),pe.optSelected||(de.propHooks.selected={get:function(e){var t=e.parentNode;return t&&t.parentNode&&t.parentNode.selectedIndex,null},set:function(e){var t=e.parentNode;t&&(t.selectedIndex,t.parentNode&&t.parentNode.selectedIndex)}}),de.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){de.propFix[this.toLowerCase()]=this}),de.fn.extend({addClass:function(e){var t,n,r,o,i,s,a,u=0;if(de.isFunction(e))return this.each(function(t){de(this).addClass(e.call(this,t,X(this)))});if("string"==typeof e&&e)for(t=e.match(qe)||[];n=this[u++];)if(o=X(n),r=1===n.nodeType&&" "+z(o)+" "){for(s=0;i=t[s++];)r.indexOf(" "+i+" ")<0&&(r+=i+" ");a=z(r),o!==a&&n.setAttribute("class",a)}return this},removeClass:function(e){var t,n,r,o,i,s,a,u=0;if(de.isFunction(e))return this.each(function(t){de(this).removeClass(e.call(this,t,X(this)))});if(!arguments.length)return this.attr("class","");if("string"==typeof e&&e)for(t=e.match(qe)||[];n=this[u++];)if(o=X(n),r=1===n.nodeType&&" "+z(o)+" "){for(s=0;i=t[s++];)for(;r.indexOf(" "+i+" ")>-1;)r=r.replace(" "+i+" "," ");a=z(r),o!==a&&n.setAttribute("class",a)}return this},toggleClass:function(e,t){var n=typeof e;return"boolean"==typeof t&&"string"===n?t?this.addClass(e):this.removeClass(e):de.isFunction(e)?this.each(function(n){de(this).toggleClass(e.call(this,n,X(this),t),t)}):this.each(function(){var t,r,o,i;if("string"===n)for(r=0,o=de(this),i=e.match(qe)||[];t=i[r++];)o.hasClass(t)?o.removeClass(t):o.addClass(t);else void 0!==e&&"boolean"!==n||(t=X(this),t&&Fe.set(this,"__className__",t),this.setAttribute&&this.setAttribute("class",t||e===!1?"":Fe.get(this,"__className__")||""))})},hasClass:function(e){var t,n,r=0;for(t=" "+e+" ";n=this[r++];)if(1===n.nodeType&&(" "+z(X(n))+" ").indexOf(t)>-1)return!0;return!1}});var wt=/\r/g;de.fn.extend({val:function(e){var t,n,r,o=this[0];{if(arguments.length)return r=de.isFunction(e),this.each(function(n){var o;1===this.nodeType&&(o=r?e.call(this,n,de(this).val()):e,null==o?o="":"number"==typeof o?o+="":de.isArray(o)&&(o=de.map(o,function(e){return null==e?"":e+""})),t=de.valHooks[this.type]||de.valHooks[this.nodeName.toLowerCase()],t&&"set"in t&&void 0!==t.set(this,o,"value")||(this.value=o))});if(o)return t=de.valHooks[o.type]||de.valHooks[o.nodeName.toLowerCase()],t&&"get"in t&&void 0!==(n=t.get(o,"value"))?n:(n=o.value,"string"==typeof n?n.replace(wt,""):null==n?"":n)}}}),de.extend({valHooks:{option:{get:function(e){var t=de.find.attr(e,"value");return null!=t?t:z(de.text(e))}},select:{get:function(e){var t,n,r,o=e.options,i=e.selectedIndex,s="select-one"===e.type,a=s?null:[],u=s?i+1:o.length;for(r=i<0?u:s?i:0;r-1)&&(n=!0);return n||(e.selectedIndex=-1),i}}}}),de.each(["radio","checkbox"],function(){de.valHooks[this]={set:function(e,t){if(de.isArray(t))return e.checked=de.inArray(de(e).val(),t)>-1}},pe.checkOn||(de.valHooks[this].get=function(e){return null===e.getAttribute("value")?"on":e.value})});var Tt=/^(?:focusinfocus|focusoutblur)$/;de.extend(de.event,{trigger:function(t,n,r,o){var i,s,a,u,c,l,f,p=[r||te],h=ce.call(t,"type")?t.type:t,d=ce.call(t,"namespace")?t.namespace.split("."):[];if(s=a=r=r||te,3!==r.nodeType&&8!==r.nodeType&&!Tt.test(h+de.event.triggered)&&(h.indexOf(".")>-1&&(d=h.split("."),h=d.shift(),d.sort()),c=h.indexOf(":")<0&&"on"+h,t=t[de.expando]?t:new de.Event(h,"object"==typeof t&&t),t.isTrigger=o?2:3,t.namespace=d.join("."),t.rnamespace=t.namespace?new RegExp("(^|\\.)"+d.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,t.result=void 0,t.target||(t.target=r),n=null==n?[t]:de.makeArray(n,[t]),f=de.event.special[h]||{},o||!f.trigger||f.trigger.apply(r,n)!==!1)){if(!o&&!f.noBubble&&!de.isWindow(r)){for(u=f.delegateType||h,Tt.test(u+h)||(s=s.parentNode);s;s=s.parentNode)p.push(s),a=s;a===(r.ownerDocument||te)&&p.push(a.defaultView||a.parentWindow||e)}for(i=0;(s=p[i++])&&!t.isPropagationStopped();)t.type=i>1?u:f.bindType||h,l=(Fe.get(s,"events")||{})[t.type]&&Fe.get(s,"handle"),l&&l.apply(s,n),l=c&&s[c],l&&l.apply&&He(s)&&(t.result=l.apply(s,n),t.result===!1&&t.preventDefault());return t.type=h,o||t.isDefaultPrevented()||f._default&&f._default.apply(p.pop(),n)!==!1||!He(r)||c&&de.isFunction(r[h])&&!de.isWindow(r)&&(a=r[c],a&&(r[c]=null),de.event.triggered=h,r[h](),de.event.triggered=void 0,a&&(r[c]=a)),t.result}},simulate:function(e,t,n){var r=de.extend(new de.Event,n,{type:e,isSimulated:!0});de.event.trigger(r,null,t)}}),de.fn.extend({trigger:function(e,t){return this.each(function(){de.event.trigger(e,t,this)})},triggerHandler:function(e,t){var n=this[0];if(n)return de.event.trigger(e,t,n,!0)}}),de.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,t){de.fn[t]=function(e,n){return arguments.length>0?this.on(t,null,e,n):this.trigger(t)}}),de.fn.extend({hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),pe.focusin="onfocusin"in e,pe.focusin||de.each({focus:"focusin",blur:"focusout"},function(e,t){var n=function(e){de.event.simulate(t,e.target,de.event.fix(e))};de.event.special[t]={setup:function(){var r=this.ownerDocument||this,o=Fe.access(r,t);o||r.addEventListener(e,n,!0),Fe.access(r,t,(o||0)+1)},teardown:function(){var r=this.ownerDocument||this,o=Fe.access(r,t)-1;o?Fe.access(r,t,o):(r.removeEventListener(e,n,!0),Fe.remove(r,t))}}});var Ct=e.location,jt=de.now(),kt=/\?/;de.parseXML=function(t){var n;if(!t||"string"!=typeof t)return null;try{n=(new e.DOMParser).parseFromString(t,"text/xml")}catch(e){n=void 0}return n&&!n.getElementsByTagName("parsererror").length||de.error("Invalid XML: "+t),n};var Et=/\[\]$/,St=/\r?\n/g,Nt=/^(?:submit|button|image|reset|file)$/i,At=/^(?:input|select|textarea|keygen)/i;de.param=function(e,t){var n,r=[],o=function(e,t){var n=de.isFunction(t)?t():t;r[r.length]=encodeURIComponent(e)+"="+encodeURIComponent(null==n?"":n)};if(de.isArray(e)||e.jquery&&!de.isPlainObject(e))de.each(e,function(){o(this.name,this.value)});else for(n in e)V(n,e[n],t,o);return r.join("&")},de.fn.extend({serialize:function(){return de.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var e=de.prop(this,"elements");return e?de.makeArray(e):this}).filter(function(){var e=this.type;return this.name&&!de(this).is(":disabled")&&At.test(this.nodeName)&&!Nt.test(e)&&(this.checked||!ze.test(e))}).map(function(e,t){var n=de(this).val();return null==n?null:de.isArray(n)?de.map(n,function(e){return{name:t.name,value:e.replace(St,"\r\n")}}):{name:t.name,value:n.replace(St,"\r\n")}}).get()}});var qt=/%20/g,Dt=/#.*$/,Ot=/([?&])_=[^&]*/,Lt=/^(.*?):[ \t]*([^\r\n]*)$/gm,Ht=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Ft=/^(?:GET|HEAD)$/,Rt=/^\/\//,It={},Pt={},Mt="*/".concat("*"),$t=te.createElement("a");$t.href=Ct.href,de.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:Ct.href,type:"GET",isLocal:Ht.test(Ct.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Mt,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":JSON.parse,"text xml":de.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(e,t){return t?Q(Q(e,de.ajaxSettings),t):Q(de.ajaxSettings,e)},ajaxPrefilter:G(It),ajaxTransport:G(Pt),ajax:function(t,n){function r(t,n,r,a){var c,p,h,b,w,T=n;l||(l=!0,u&&e.clearTimeout(u),o=void 0,s=a||"",C.readyState=t>0?4:0,c=t>=200&&t<300||304===t,r&&(b=J(d,C,r)),b=K(d,b,C,c),c?(d.ifModified&&(w=C.getResponseHeader("Last-Modified"),w&&(de.lastModified[i]=w),w=C.getResponseHeader("etag"),w&&(de.etag[i]=w)),204===t||"HEAD"===d.type?T="nocontent":304===t?T="notmodified":(T=b.state,p=b.data,h=b.error,c=!h)):(h=T,!t&&T||(T="error",t<0&&(t=0))),C.status=t,C.statusText=(n||T)+"",c?v.resolveWith(g,[p,T,C]):v.rejectWith(g,[C,T,h]),C.statusCode(x),x=void 0,f&&m.trigger(c?"ajaxSuccess":"ajaxError",[C,d,c?p:h]),y.fireWith(g,[C,T]),f&&(m.trigger("ajaxComplete",[C,d]),--de.active||de.event.trigger("ajaxStop")))}"object"==typeof t&&(n=t,t=void 0),n=n||{};var o,i,s,a,u,c,l,f,p,h,d=de.ajaxSetup({},n),g=d.context||d,m=d.context&&(g.nodeType||g.jquery)?de(g):de.event,v=de.Deferred(),y=de.Callbacks("once memory"),x=d.statusCode||{},b={},w={},T="canceled",C={readyState:0,getResponseHeader:function(e){var t;if(l){if(!a)for(a={};t=Lt.exec(s);)a[t[1].toLowerCase()]=t[2];t=a[e.toLowerCase()]}return null==t?null:t},getAllResponseHeaders:function(){return l?s:null},setRequestHeader:function(e,t){return null==l&&(e=w[e.toLowerCase()]=w[e.toLowerCase()]||e,b[e]=t),this},overrideMimeType:function(e){return null==l&&(d.mimeType=e),this},statusCode:function(e){var t;if(e)if(l)C.always(e[C.status]);else for(t in e)x[t]=[x[t],e[t]];return this},abort:function(e){var t=e||T;return o&&o.abort(t),r(0,t),this}};if(v.promise(C),d.url=((t||d.url||Ct.href)+"").replace(Rt,Ct.protocol+"//"),d.type=n.method||n.type||d.method||d.type,d.dataTypes=(d.dataType||"*").toLowerCase().match(qe)||[""],null==d.crossDomain){c=te.createElement("a");try{c.href=d.url,c.href=c.href,d.crossDomain=$t.protocol+"//"+$t.host!=c.protocol+"//"+c.host}catch(e){d.crossDomain=!0}}if(d.data&&d.processData&&"string"!=typeof d.data&&(d.data=de.param(d.data,d.traditional)),Y(It,d,n,C),l)return C;f=de.event&&d.global,f&&0===de.active++&&de.event.trigger("ajaxStart"),d.type=d.type.toUpperCase(),d.hasContent=!Ft.test(d.type),i=d.url.replace(Dt,""),d.hasContent?d.data&&d.processData&&0===(d.contentType||"").indexOf("application/x-www-form-urlencoded")&&(d.data=d.data.replace(qt,"+")):(h=d.url.slice(i.length),d.data&&(i+=(kt.test(i)?"&":"?")+d.data,delete d.data),d.cache===!1&&(i=i.replace(Ot,"$1"),h=(kt.test(i)?"&":"?")+"_="+jt++ +h),d.url=i+h),d.ifModified&&(de.lastModified[i]&&C.setRequestHeader("If-Modified-Since",de.lastModified[i]),de.etag[i]&&C.setRequestHeader("If-None-Match",de.etag[i])),(d.data&&d.hasContent&&d.contentType!==!1||n.contentType)&&C.setRequestHeader("Content-Type",d.contentType),C.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+("*"!==d.dataTypes[0]?", "+Mt+"; q=0.01":""):d.accepts["*"]);for(p in d.headers)C.setRequestHeader(p,d.headers[p]);if(d.beforeSend&&(d.beforeSend.call(g,C,d)===!1||l))return C.abort();if(T="abort",y.add(d.complete),C.done(d.success),C.fail(d.error),o=Y(Pt,d,n,C)){if(C.readyState=1,f&&m.trigger("ajaxSend",[C,d]),l)return C;d.async&&d.timeout>0&&(u=e.setTimeout(function(){C.abort("timeout")},d.timeout));try{l=!1,o.send(b,r)}catch(e){if(l)throw e;r(-1,e)}}else r(-1,"No Transport");return C},getJSON:function(e,t,n){return de.get(e,t,n,"json")},getScript:function(e,t){return de.get(e,void 0,t,"script")}}),de.each(["get","post"],function(e,t){de[t]=function(e,n,r,o){return de.isFunction(n)&&(o=o||r,r=n,n=void 0),de.ajax(de.extend({url:e,type:t,dataType:o,data:n,success:r},de.isPlainObject(e)&&e))}}),de._evalUrl=function(e){return de.ajax({url:e,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,throws:!0})},de.fn.extend({wrapAll:function(e){var t;return this[0]&&(de.isFunction(e)&&(e=e.call(this[0])),t=de(e,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&t.insertBefore(this[0]),t.map(function(){for(var e=this;e.firstElementChild;)e=e.firstElementChild;return e}).append(this)),this},wrapInner:function(e){return de.isFunction(e)?this.each(function(t){de(this).wrapInner(e.call(this,t))}):this.each(function(){var t=de(this),n=t.contents();n.length?n.wrapAll(e):t.append(e)})},wrap:function(e){var t=de.isFunction(e);return this.each(function(n){de(this).wrapAll(t?e.call(this,n):e)})},unwrap:function(e){return this.parent(e).not("body").each(function(){de(this).replaceWith(this.childNodes)}),this}}),de.expr.pseudos.hidden=function(e){return!de.expr.pseudos.visible(e)},de.expr.pseudos.visible=function(e){return!!(e.offsetWidth||e.offsetHeight||e.getClientRects().length)},de.ajaxSettings.xhr=function(){try{return new e.XMLHttpRequest}catch(e){}};var Wt={0:200,1223:204},Bt=de.ajaxSettings.xhr();pe.cors=!!Bt&&"withCredentials"in Bt,pe.ajax=Bt=!!Bt,de.ajaxTransport(function(t){var n,r;if(pe.cors||Bt&&!t.crossDomain)return{send:function(o,i){var s,a=t.xhr();if(a.open(t.type,t.url,t.async,t.username,t.password),t.xhrFields)for(s in t.xhrFields)a[s]=t.xhrFields[s];t.mimeType&&a.overrideMimeType&&a.overrideMimeType(t.mimeType),t.crossDomain||o["X-Requested-With"]||(o["X-Requested-With"]="XMLHttpRequest");for(s in o)a.setRequestHeader(s,o[s]);n=function(e){return function(){n&&(n=r=a.onload=a.onerror=a.onabort=a.onreadystatechange=null,"abort"===e?a.abort():"error"===e?"number"!=typeof a.status?i(0,"error"):i(a.status,a.statusText):i(Wt[a.status]||a.status,a.statusText,"text"!==(a.responseType||"text")||"string"!=typeof a.responseText?{binary:a.response}:{text:a.responseText},a.getAllResponseHeaders()))}},a.onload=n(),r=a.onerror=n("error"),void 0!==a.onabort?a.onabort=r:a.onreadystatechange=function(){4===a.readyState&&e.setTimeout(function(){n&&r()})},n=n("abort");try{a.send(t.hasContent&&t.data||null)}catch(e){if(n)throw e}},abort:function(){n&&n()}}}),de.ajaxPrefilter(function(e){e.crossDomain&&(e.contents.script=!1)}),de.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(e){return de.globalEval(e),e}}}),de.ajaxPrefilter("script",function(e){void 0===e.cache&&(e.cache=!1),e.crossDomain&&(e.type="GET")}),de.ajaxTransport("script",function(e){if(e.crossDomain){var t,n;return{send:function(r,o){t=de("

      ================================================ FILE: docs/search_index.json ================================================ {"index":{"version":"0.5.12","fields":[{"name":"title","boost":10},{"name":"keywords","boost":15},{"name":"body","boost":1}],"ref":"url","documentStore":{"store":{"./":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"turn","\"user","\"x","#gener","#global","#launch","#local","&","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","+=",",",".",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","/^win/.test(process.platform);","0)","10.3","10000,","1080","127.0.0.1:8001","127.0.0.1:8001,","127.0.0.1:8001/index.html","1,write","2,","200,","3,","3.x:","4,","404","404;","443;","5","5000);","5s","80,","8001","8001,","8002","=","===","=>",">","[switchyomega]","_re","_req","_req:","_res:","a:","abil","abliti","activ","add","address","advanc","advance.","agent","agent\":","agent'","agent')","agent':","agent']","agent,","alive',","all,","alway","android","anyprox","anyproxi","anyproxy',","anyproxy,","anyproxy.","anyproxy.io,","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","append","appendix:how","apt","as:","at',","attack","attack.","automat","base","be","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","below,","below.","below:","besid","bodi","body,","body:","bonou","browser","browser.","by']","by:","by:anyproxi","ca","ca,","ca.","call","card,","caus","cert","certdir","certdir);","certif","certificate.","certificates,","certificates.","certifyc","chang","chart","choos","chrome","class:","cli,","cli.","click","clicking,","client","client,","close","co","co.","code","code,","code.","collect","command","common","compat","config","config.","configur","connect","connection':","connection:","connections.","console,","console.error('error","console.log('th","const","content","content,","control':","couldn't","course.","creat","credenti","curl","curl:","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","dangerouslyignoreunauthorized:true,","data","data\"","data'","data',","deal","dealed.","deall","debian","decrypt","default","default.","defin","delay","demo","design","detect","determin","differ","directli","disabl","dn","doubl","download","driven","due","dure","e.g.","easist","edit","emit","enabl","enable:","encrypt","end","entir","error","error)","error);","errors,","event:","example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例","except","exec","exec('open","exec('start","exist","expect","extens","extent","face","fail","failure,","fals","false,","false;","faq","feature,","features:","few","field","file","file,","file.","file:","find","finish","first","flow","follow","follows.","forc","forceproxyhttp","forceproxyhttps:","forward","found","frequent","full","fulli","function","function()","function.","g","gener","get","github:","give","global","go,","gui","guid","h","hack","hacked!","hacked!';","handl","handling.","happen","header","header,","header:","headers:","headersar","help","here","here,","here.","hood,","host","host:","hostname:","http","http.rquest,","http/1.1","http/http","http://....j","http://127.0.0.1:8001","http://127.0.0.1:8002","http://127.0.0.1:8002,","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://github.com/alibaba/anyproxi","https://httpbin.org/us","https://nodejs.org/api/http.html#http_http_request_options_callback","https://sample.com/rule.j","i.e.","ignor","import","in.","includ","influenc","info","info,","insid","instal","installing.","instruct","intercept","interest","interfac","interface,","introduct","invok","io","ios/android","issu","iswin","it,","it.","javascript.","js","kb/s,","keep","keypath)","kind","know","know.com')","known","launch","launching,","legaci","line","line,","list","load","local","localrespons","locat","log","logic","login","longer.","luanch","make","man","manag","manual","menu","menu,","messag","method","method,","method:","middl","mind","modifi","modul","module,","module.","module.export","myrulepkg","nativ","need","network.","new","newli","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","node.j","nodej","nodejs.","nothing,","now","npm","null","null;","number","object","object.","object.assign({},","offer","on","on\"","on,","onconnecterror","onconnecterror(requestdetail,","one.","onerror","onerror(requestdetail,","onlin","oppo","option","option,","optional,","osx","otherwire,","otherwise,","own.","page","param","part","pass","password","path.","path:","phones.","place","pleas","point","popular,","port","port:","post","privat","process","promis","promise((resolve,","propmt","protocol","protocol\":","protocol,","protocol:","protocol,i.e.","provid","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","qr","quest","r15.","read","readi","receiv","record","redirect","ref:","refer","reject)","rejectunauthor","replac","request","request,","request.","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requesting,","requestopt","requestoptions:","requests,","requests.","requir","require('anyproxy');","require('child_process').exec;","require('http').request","require('myrulemodule'),","require('path').dirname(keypath);","require()","required,","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","right","risk.","root","rootca","rootca',","rootca.crt","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","run","safari","safe","same","sampl","sample.j","sample:","save","scan","scenario,","sd","second","seconds.","secur","security.","see","self","send","sent","server","server,","server.","set","settimeout(()","settings.","sever","side","signed.","silent","silent:","similar","simpl","simplifi","site","site(s)","situation.","skip","someth","source.","specifi","ssl","start","statu","statuscod","statuscode:","step","storage,","stuff:","such","sudo","summari","summary,","summary:","support","sure","system","system,","take","target","tell","test","test:","text","text/plain\"","that.","this:","throttl","throttle:","time","timeout","tip","to!","toward","traffic","tri","true","true,","true;","trust","turn","type","type':","type:","typic","ubuntu","ui,","unauthor","under","unknown","unlimit","unsecur","untrust","url","url:","us","use,","user","user.","users,","usual","valid","valu","var","via","view","visit","want","warning:","way","web","webinterfac","webinterface:","webport","webport:","websocket","websocket,","whether","wifi","window","without","work","world'","write","written","ws","wsintercept","wsintercept:","wss","x","yieldabl","yieldable,","you'r","{","{boolean}","{number}","{object}","{string}","}","})","});","},","};","中文文档",":"]},"length":1},"tokenStore":{"root":{"0":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}},"1":{"0":{"0":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"8":{"0":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}},"docs":{},".":{"3":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"docs":{}}},"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},",":{"docs":{},"访":{"docs":{},"问":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},",":{"docs":{},"编":{"docs":{},"写":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"w":{"docs":{},"r":{"docs":{},"i":{"docs":{},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"2":{"0":{"0":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"3":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{},"x":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"docs":{}},"4":{"3":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"0":{"docs":{},"的":{"docs":{},"主":{"docs":{},"要":{"docs":{},"变":{"docs":{},"化":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},",":{"docs":{},"欢":{"docs":{},"迎":{"docs":{},"提":{"docs":{},"供":{"docs":{},"反":{"docs":{},"馈":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"docs":{}}},"5":{"0":{"0":{"0":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"docs":{}},"docs":{}},"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"2":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}},"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},"\"":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"/":{"7":{"docs":{},".":{"4":{"3":{"docs":{},".":{"0":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"docs":{}}},"docs":{}},"docs":{}}},"docs":{}}}}}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"#":{"docs":{},"全":{"docs":{},"局":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"地":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"生":{"docs":{},"成":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"生":{"docs":{},"成":{"docs":{},"后":{"docs":{},"需":{"docs":{},"要":{"docs":{},"手":{"docs":{},"动":{"docs":{},"信":{"docs":{},"任":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"l":{"docs":{},"a":{"docs":{},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"&":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"&":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"8":{"0":{"0":{"1":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"8":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},".":{"docs":{},".":{"docs":{},".":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"/":{"docs":{},"t":{"docs":{},"o":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"o":{"docs":{},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{},"/":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"a":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"/":{"0":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}}},"docs":{}}},"docs":{}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"l":{"docs":{},"i":{"docs":{},"c":{"docs":{},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"/":{"docs":{},"j":{"docs":{},"s":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"o":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"i":{"docs":{},"o":{"docs":{},"/":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{},"?":{"docs":{},"s":{"docs":{},"h":{"docs":{},"o":{"docs":{},"w":{"docs":{},"_":{"docs":{},"e":{"docs":{},"n":{"docs":{},"v":{"docs":{},"=":{"1":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"m":{"docs":{},"a":{"docs":{},"g":{"docs":{},"e":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"f":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"k":{"docs":{},"e":{"docs":{},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"m":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"e":{"docs":{},"x":{"docs":{},"t":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}},"h":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"{":{"docs":{},"\"":{"docs":{},"h":{"docs":{},"e":{"docs":{},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"(":{"docs":{},"!":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},".":{"docs":{},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"e":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},".":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},"o":{"docs":{},"f":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"/":{"docs":{},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"*":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.007178106774338268}},"}":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"+":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},".":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"/":{"docs":{},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{},"/":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"/":{"docs":{},"*":{"docs":{"./":{"ref":"./","tf":0.00762673844773441}}},"/":{"docs":{"./":{"ref":"./","tf":0.006280843427545985}},"w":{"docs":{},"h":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"这":{"docs":{},"里":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"同":{"docs":{},"时":{"docs":{},"加":{"docs":{},"上":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"^":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{},"/":{"docs":{},".":{"docs":{},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"(":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},".":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"t":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"m":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"=":{"docs":{"./":{"ref":"./","tf":0.020188425302826378}},"=":{"docs":{},"=":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}},">":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},">":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"=":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"关":{"docs":{},"于":{"docs":{},"本":{"docs":{},"机":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"信":{"docs":{},"任":{"docs":{},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"通":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"q":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"l":{"docs":{},"i":{"docs":{},"v":{"docs":{},"e":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"w":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.03230148048452221}}},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"c":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"r":{"docs":{},"(":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"(":{"docs":{},"(":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"f":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},"f":{"docs":{},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"e":{"docs":{},"x":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{},"s":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"s":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"(":{"docs":{},"'":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},".":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"使":{"docs":{},"用":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},"加":{"docs":{},"载":{"docs":{},"本":{"docs":{},"地":{"docs":{},"规":{"docs":{},"则":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"在":{"docs":{},"参":{"docs":{},"数":{"docs":{},"里":{"docs":{},"传":{"docs":{},"入":{"docs":{},"一":{"docs":{},"个":{"docs":{},"本":{"docs":{},"地":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{},"路":{"docs":{},"径":{"docs":{},",":{"docs":{},"或":{"docs":{},"是":{"docs":{},"某":{"docs":{},"个":{"docs":{},"全":{"docs":{},"局":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"作":{"docs":{},"为":{"docs":{},"一":{"docs":{},"个":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{},",":{"docs":{},"整":{"docs":{},"合":{"docs":{},"进":{"docs":{},"其":{"docs":{},"他":{"docs":{},"工":{"docs":{},"具":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"送":{"docs":{},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"在":{"docs":{},"与":{"docs":{},"目":{"docs":{},"标":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"建":{"docs":{},"立":{"docs":{},"连":{"docs":{},"接":{"docs":{},"的":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"这":{"docs":{},"个":{"docs":{},"方":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"了":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"的":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"j":{"docs":{},"s":{"docs":{},"编":{"docs":{},"写":{"docs":{},"自":{"docs":{},"己":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"来":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"网":{"docs":{},"络":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"并":{"docs":{},"带":{"docs":{},"上":{"docs":{},"参":{"docs":{},"数":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"开":{"docs":{},"放":{"docs":{},"式":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"新":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"或":{"docs":{},"返":{"docs":{},"回":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"对":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},",":{"docs":{},"如":{"docs":{},"需":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"配":{"docs":{},"置":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"x":{"docs":{},":":{"docs":{},"h":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"t":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"t":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},")":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"提":{"docs":{},"前":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"b":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"l":{"docs":{},"i":{"docs":{},"t":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"c":{"docs":{},"t":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"d":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"a":{"docs":{},"n":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"t":{"docs":{},"o":{"docs":{},"m":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"b":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"d":{"docs":{},"y":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},"三":{"docs":{},"个":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},")":{"docs":{},",":{"docs":{},"甚":{"docs":{},"至":{"docs":{},"是":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},"n":{"docs":{},"o":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"y":{"docs":{},"'":{"docs":{},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"a":{"docs":{},"s":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"o":{"docs":{},"w":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.01031852848811126}},"方":{"docs":{},"法":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},",":{"docs":{},"按":{"docs":{},"提":{"docs":{},"示":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"即":{"docs":{},"可":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}}}},"r":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"f":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"i":{"docs":{},"c":{"docs":{},"a":{"docs":{},"t":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"y":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"l":{"docs":{},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"i":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"d":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"(":{"docs":{},"'":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.008524001794526694}}}},"t":{"docs":{},"r":{"docs":{},"o":{"docs":{},"l":{"docs":{},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"f":{"docs":{},"i":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"l":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"l":{"docs":{},"d":{"docs":{},"n":{"docs":{},"'":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"s":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"w":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"h":{"docs":{},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"o":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"d":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"'":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"e":{"docs":{},"l":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"e":{"docs":{},"d":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"b":{"docs":{},"i":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"c":{"docs":{},"r":{"docs":{},"y":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"f":{"docs":{},"a":{"docs":{},"u":{"docs":{},"l":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"m":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"m":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"u":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"w":{"docs":{},"n":{"docs":{},"l":{"docs":{},"o":{"docs":{},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}},"i":{"docs":{},"f":{"docs":{},"f":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"l":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"s":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"r":{"docs":{},"i":{"docs":{},"v":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"u":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"g":{"docs":{},"l":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"c":{"docs":{},"r":{"docs":{},"y":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.006729475100942127}},")":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"v":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"'":{"docs":{},"o":{"docs":{},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"p":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"(":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"c":{"docs":{},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},".":{"docs":{},"g":{"docs":{},"o":{"docs":{},"o":{"docs":{},"g":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"/":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"/":{"docs":{},"p":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"k":{"docs":{},"g":{"docs":{},"c":{"docs":{},"e":{"docs":{},"m":{"docs":{},"l":{"docs":{},"o":{"docs":{},"k":{"docs":{},"b":{"docs":{},"a":{"docs":{},"d":{"docs":{},"o":{"docs":{},"h":{"docs":{},"g":{"docs":{},"k":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"j":{"docs":{},"o":{"docs":{},"m":{"docs":{},"c":{"docs":{},"l":{"docs":{},"g":{"docs":{},"j":{"docs":{},"g":{"docs":{},"i":{"docs":{},"f":{"docs":{},")":{"docs":{},"为":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},".":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"a":{"docs":{},"s":{"docs":{},"i":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"d":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"m":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"q":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"u":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.006280843427545985}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"i":{"docs":{},"s":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"r":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"l":{"docs":{},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"e":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"l":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},"主":{"docs":{},"页":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"g":{"docs":{},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"a":{"docs":{},"l":{"docs":{},"i":{"docs":{},"b":{"docs":{},"a":{"docs":{},"b":{"docs":{},"a":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"l":{"docs":{},"o":{"docs":{},"b":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"o":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"d":{"docs":{},"!":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"'":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"n":{"docs":{},"d":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"p":{"docs":{},"p":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},")":{"docs":{},",":{"docs":{},"请":{"docs":{},"求":{"docs":{},"体":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},")":{"docs":{},"、":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"l":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.014804845222072678}},"/":{"1":{"docs":{},".":{"1":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"docs":{}}},"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}},"2":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"/":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}}}}}}}}}}}}}}},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},":":{"8":{"0":{"0":{"8":{"docs":{},"/":{"docs":{},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"e":{"docs":{},"x":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"g":{"docs":{},"i":{"docs":{},"t":{"docs":{},"h":{"docs":{},"u":{"docs":{},"b":{"docs":{},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"/":{"docs":{},"a":{"docs":{},"l":{"docs":{},"i":{"docs":{},"b":{"docs":{},"a":{"docs":{},"b":{"docs":{},"a":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"a":{"docs":{},"p":{"docs":{},"i":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"#":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"_":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"连":{"docs":{},"接":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"出":{"docs":{},"错":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}},"的":{"docs":{},"代":{"docs":{},"理":{"docs":{},"模":{"docs":{},"式":{"docs":{},"中":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"的":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"是":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},".":{"docs":{},"r":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}},"n":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.008075370121130552}},"l":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"u":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"按":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"启":{"docs":{},"动":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"`":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"后":{"docs":{},"才":{"docs":{},"会":{"docs":{},"从":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"看":{"docs":{},"到":{"docs":{},"相":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"f":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"o":{"docs":{},"d":{"docs":{},"u":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":10.001345895020188}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"c":{"docs":{},"l":{"docs":{},"u":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}},"f":{"docs":{},"l":{"docs":{},"u":{"docs":{},"e":{"docs":{},"n":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"v":{"docs":{},"o":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},".":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"m":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"n":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},".":{"docs":{},"c":{"docs":{},"o":{"docs":{},"m":{"docs":{},"'":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"b":{"docs":{},"/":{"docs":{},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"l":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"c":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"a":{"docs":{},"u":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"u":{"docs":{},"a":{"docs":{},"n":{"docs":{},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"n":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"s":{"docs":{},"a":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"d":{"docs":{},"d":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},")":{"docs":{},",":{"docs":{},"用":{"docs":{},"户":{"docs":{},"必":{"docs":{},"须":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"才":{"docs":{},"能":{"docs":{},"进":{"docs":{},"行":{"docs":{},"后":{"docs":{},"续":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.012113055181695828}}}}},"u":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"e":{"docs":{},"x":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"p":{"docs":{},"k":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"a":{"docs":{},"k":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"a":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"u":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"n":{"docs":{},"e":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},".":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},".":{"docs":{},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"s":{"docs":{},"[":{"docs":{},"'":{"docs":{},"u":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"r":{"docs":{},"e":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},"e":{"docs":{},".":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},"[":{"docs":{},"'":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}}}}}},"l":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"e":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}},"t":{"docs":{},"w":{"docs":{},"o":{"docs":{},"r":{"docs":{},"k":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"t":{"docs":{},"h":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"p":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},";":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}},"m":{"docs":{},"b":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"a":{"docs":{},"s":{"docs":{},"s":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"(":{"docs":{},"{":{"docs":{},"}":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}}}}}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"n":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"i":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"p":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"a":{"docs":{},"l":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},"x":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"f":{"docs":{},"f":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{},"w":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}}},"w":{"docs":{},"n":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"g":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"r":{"docs":{},"a":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"w":{"docs":{},"o":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"p":{"docs":{},"u":{"docs":{},"l":{"docs":{},"a":{"docs":{},"r":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"i":{"docs":{},"v":{"docs":{},"a":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}}}}},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"\"":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},",":{"docs":{},"i":{"docs":{},".":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.016150740242261104}}},"y":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"c":{"docs":{},"l":{"docs":{},"o":{"docs":{},"s":{"docs":{},"e":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"o":{"docs":{},"n":{"docs":{},"(":{"docs":{},"'":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"y":{"docs":{},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{},"(":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"p":{"docs":{},"m":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"v":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"h":{"docs":{},"o":{"docs":{},"n":{"docs":{},"e":{"docs":{},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"l":{"docs":{},"a":{"docs":{},"c":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"e":{"docs":{},"a":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}},"q":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"1":{"5":{"docs":{},")":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"docs":{}},"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"f":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.02332884701659937}},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}},".":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},";":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}}}}}}}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}},".":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"i":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"'":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}}}},"c":{"docs":{},"h":{"docs":{},"i":{"docs":{},"l":{"docs":{},"d":{"docs":{},"_":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"c":{"docs":{},"e":{"docs":{},"s":{"docs":{},"s":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"e":{"docs":{},"x":{"docs":{},"e":{"docs":{},"c":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}},"m":{"docs":{},"y":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"'":{"docs":{},")":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"d":{"docs":{},"i":{"docs":{},"r":{"docs":{},"n":{"docs":{},"a":{"docs":{},"m":{"docs":{},"e":{"docs":{},"(":{"docs":{},"k":{"docs":{},"e":{"docs":{},"y":{"docs":{},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"v":{"docs":{},"e":{"docs":{},"(":{"docs":{},"{":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.014804845222072678}},"e":{"docs":{},"\"":{"docs":{},"}":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.007178106774338268}}},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},")":{"docs":{},";":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},";":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}}}}}}}}}}}}}}},"t":{"docs":{},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.016150740242261104}}}}}},"c":{"docs":{},"e":{"docs":{},"i":{"docs":{},"v":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005832211754149843}},"c":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"'":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.021085688649618663}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"_":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"/":{"docs":{},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{},"e":{"docs":{},"_":{"docs":{},"m":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{},"f":{"docs":{},"y":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"p":{"docs":{},"a":{"docs":{},"t":{"docs":{},"h":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"o":{"docs":{},"c":{"docs":{},"o":{"docs":{},"l":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"d":{"docs":{},"a":{"docs":{},"t":{"docs":{},"a":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"h":{"docs":{},"e":{"docs":{},"a":{"docs":{},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"_":{"docs":{},"l":{"docs":{},"o":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"接":{"docs":{},"口":{"docs":{},"文":{"docs":{},"档":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"n":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"i":{"docs":{},"g":{"docs":{},"h":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{},"k":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"s":{"docs":{},"a":{"docs":{},"m":{"docs":{},"p":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00493494840735756}},"e":{"docs":{},".":{"docs":{},"j":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},":":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},"v":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"f":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"t":{"docs":{},"i":{"docs":{},"m":{"docs":{},"e":{"docs":{},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{},"(":{"docs":{},"(":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}},"n":{"docs":{},"g":{"docs":{},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"i":{"docs":{},"t":{"docs":{},"y":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"l":{"docs":{},"f":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"n":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.004486316733961417}}},"t":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"r":{"docs":{},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.008075370121130552}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"v":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"i":{"docs":{},"l":{"docs":{},"e":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"(":{"docs":{},"s":{"docs":{},")":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"a":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"g":{"docs":{},"n":{"docs":{},"e":{"docs":{},"d":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"m":{"docs":{},"i":{"docs":{},"l":{"docs":{},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"p":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"s":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"t":{"docs":{},"a":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}}}}}},"e":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"o":{"docs":{},"r":{"docs":{},"a":{"docs":{},"g":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"u":{"docs":{},"f":{"docs":{},"f":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"u":{"docs":{},"d":{"docs":{},"o":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"y":{"docs":{},"(":{"docs":{},")":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"c":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"p":{"docs":{},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"r":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"c":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"e":{"docs":{},"n":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{},"o":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}},"d":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"k":{"docs":{},"i":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"u":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"p":{"docs":{},"e":{"docs":{},"c":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"t":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},":":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}},"x":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"/":{"docs":{},"p":{"docs":{},"l":{"docs":{},"a":{"docs":{},"i":{"docs":{},"n":{"docs":{},"\"":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"t":{"docs":{},"t":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"a":{"docs":{},"t":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"i":{"docs":{},"s":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"o":{"docs":{},"w":{"docs":{},"a":{"docs":{},"r":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0026917900403768506}}}}}},"!":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},",":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},";":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"或":{"docs":{},"者":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{},"e":{"docs":{},",":{"docs":{},"表":{"docs":{},"示":{"docs":{},"是":{"docs":{},"否":{"docs":{},"需":{"docs":{},"要":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.005383580080753701}},")":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"a":{"docs":{},"f":{"docs":{},"f":{"docs":{},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"y":{"docs":{},"p":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"'":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"i":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"a":{"docs":{},"k":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"g":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}}}}}}},"i":{"docs":{},"m":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"p":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"r":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"u":{"docs":{},"i":{"docs":{},"中":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"d":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}},"k":{"docs":{},"n":{"docs":{},"o":{"docs":{},"w":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"l":{"docs":{},"i":{"docs":{},"m":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"s":{"docs":{},"e":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.009421265141318977}},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"s":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"u":{"docs":{},"a":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}},"b":{"docs":{},"u":{"docs":{},"n":{"docs":{},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"v":{"docs":{},"a":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"l":{"docs":{},"i":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"u":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}},"i":{"docs":{},"a":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"e":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"s":{"docs":{},"i":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.001794526693584567}}}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{"./":{"ref":"./","tf":0.0031404217137729925}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"f":{"docs":{},"a":{"docs":{},"c":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"e":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"重":{"docs":{},"构":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}},"i":{"docs":{},"n":{"docs":{},"d":{"docs":{},"o":{"docs":{},"w":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"s":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"f":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"u":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"o":{"docs":{},"r":{"docs":{},"l":{"docs":{},"d":{"docs":{},"'":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"k":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},"i":{"docs":{},"n":{"docs":{},"t":{"docs":{},"e":{"docs":{},"r":{"docs":{},"c":{"docs":{},"e":{"docs":{},"p":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"a":{"docs":{},"n":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"r":{"docs":{},"n":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}},"y":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}},"h":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}},"r":{"docs":{},"i":{"docs":{},"t":{"docs":{},"e":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}},"t":{"docs":{},"e":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}},"x":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}}}}}}}}}}},"o":{"docs":{},"u":{"docs":{},"'":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}},"{":{"docs":{"./":{"ref":"./","tf":0.03947958725886048}},"b":{"docs":{},"o":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"a":{"docs":{},"n":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0022431583669807087}}}}}}}}}},"n":{"docs":{},"u":{"docs":{},"m":{"docs":{},"b":{"docs":{},"e":{"docs":{},"r":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}},"o":{"docs":{},"b":{"docs":{},"j":{"docs":{},"e":{"docs":{},"c":{"docs":{},"t":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.003589053387169134}}}}}}}}},"s":{"docs":{},"t":{"docs":{},"r":{"docs":{},"i":{"docs":{},"n":{"docs":{},"g":{"docs":{},"}":{"docs":{"./":{"ref":"./","tf":0.0013458950201884253}}}}}}}}}},"|":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"}":{"docs":{"./":{"ref":"./","tf":0.013458950201884253}},")":{"docs":{"./":{"ref":"./","tf":0.0008972633467922835}},";":{"docs":{"./":{"ref":"./","tf":0.004037685060565276}}}},",":{"docs":{"./":{"ref":"./","tf":0.008972633467922835}}},";":{"docs":{"./":{"ref":"./","tf":0.01256168685509197}}}},"下":{"docs":{},"载":{"docs":{},"后":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"可":{"docs":{},"以":{"docs":{},"直":{"docs":{},"接":{"docs":{},"单":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"并":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"是":{"docs":{},"最":{"docs":{},"简":{"docs":{},"单":{"docs":{},"的":{"docs":{},",":{"docs":{},"直":{"docs":{},"接":{"docs":{},"安":{"docs":{},"装":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"不":{"docs":{},"做":{"docs":{},"任":{"docs":{},"何":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{},"此":{"docs":{},"时":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"返":{"docs":{},"回":{"docs":{},"一":{"docs":{},"个":{"docs":{},"默":{"docs":{},"认":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"返":{"docs":{},"回":{"docs":{},"n":{"docs":{},"u":{"docs":{},"l":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}},"同":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},"类":{"docs":{},"型":{"docs":{},"不":{"docs":{},"尽":{"docs":{},"相":{"docs":{},"同":{"docs":{},",":{"docs":{},"大":{"docs":{},"多":{"docs":{},"支":{"docs":{},"持":{"docs":{},"安":{"docs":{},"装":{"docs":{},"拓":{"docs":{},"展":{"docs":{},"名":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"中":{"docs":{},"把":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"开":{"docs":{},"关":{"docs":{},"打":{"docs":{},"开":{"docs":{},",":{"docs":{},"否":{"docs":{},"则":{"docs":{},"s":{"docs":{},"a":{"docs":{},"f":{"docs":{},"a":{"docs":{},"r":{"docs":{},"i":{"docs":{},"将":{"docs":{},"报":{"docs":{},"错":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"文":{"docs":{},"文":{"docs":{},"档":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"什":{"docs":{},"么":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"能":{"docs":{},"进":{"docs":{},"入":{"docs":{},"处":{"docs":{},"理":{"docs":{},"函":{"docs":{},"数":{"docs":{},"?":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"主":{"docs":{},"要":{"docs":{},"特":{"docs":{},"性":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},":":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"从":{"docs":{},"s":{"docs":{},"d":{"docs":{},"卡":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"存":{"docs":{},"储":{"docs":{},"设":{"docs":{},"备":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"找":{"docs":{},"到":{"docs":{},"你":{"docs":{},"下":{"docs":{},"载":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"启":{"docs":{},"动":{"docs":{},"完":{"docs":{},"成":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"都":{"docs":{},"在":{"docs":{},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"以":{"docs":{},"c":{"docs":{},"h":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"的":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"t":{"docs":{},"c":{"docs":{},"h":{"docs":{},"y":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"插":{"docs":{},"件":{"docs":{},"为":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"下":{"docs":{},"任":{"docs":{},"意":{"docs":{},"一":{"docs":{},"项":{"docs":{},"都":{"docs":{},"能":{"docs":{},"用":{"docs":{},"来":{"docs":{},"改":{"docs":{},"变":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"处":{"docs":{},"理":{"docs":{},"特":{"docs":{},"性":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"几":{"docs":{},"种":{"docs":{},"返":{"docs":{},"回":{"docs":{},"都":{"docs":{},"是":{"docs":{},"合":{"docs":{},"法":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}}}}}}}}}}}}},"作":{"docs":{},"为":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"模":{"docs":{},"块":{"docs":{},"使":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"全":{"docs":{},"局":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"你":{"docs":{},"可":{"docs":{},"以":{"docs":{},"通":{"docs":{},"过":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"使":{"docs":{},"用":{"docs":{},"n":{"docs":{},"p":{"docs":{},"m":{"docs":{},"包":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"举":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"在":{"docs":{},"线":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"路":{"docs":{},"径":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"修":{"docs":{},"改":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"协":{"docs":{},"议":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},",":{"docs":{},"如":{"docs":{},"强":{"docs":{},"制":{"docs":{},"改":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"目":{"docs":{},"标":{"docs":{},"地":{"docs":{},"址":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"内":{"docs":{},"容":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"头":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"关":{"docs":{},"闭":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"其":{"docs":{},"他":{"docs":{},"命":{"docs":{},"令":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"创":{"docs":{},"建":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"加":{"docs":{},"密":{"docs":{},"与":{"docs":{},"凭":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"即":{"docs":{},"将":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"供":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"i":{"docs":{},"r":{"docs":{},"e":{"docs":{},"(":{"docs":{},"'":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"'":{"docs":{},")":{"docs":{},".":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"作":{"docs":{},"为":{"docs":{},"使":{"docs":{},"用":{"docs":{},"。":{"docs":{},"详":{"docs":{},"见":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{},"/":{"docs":{},"a":{"docs":{},"p":{"docs":{},"i":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},".":{"docs":{},"h":{"docs":{},"t":{"docs":{},"m":{"docs":{},"l":{"docs":{},"#":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"_":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"_":{"docs":{},"o":{"docs":{},"p":{"docs":{},"t":{"docs":{},"i":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"_":{"docs":{},"c":{"docs":{},"a":{"docs":{},"l":{"docs":{},"l":{"docs":{},"b":{"docs":{},"a":{"docs":{},"c":{"docs":{},"k":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"原":{"docs":{},"始":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{},"对":{"docs":{},"象":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"双":{"docs":{},"击":{"docs":{},"打":{"docs":{},"开":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"响":{"docs":{},"应":{"docs":{},"前":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"请":{"docs":{},"求":{"docs":{},"前":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"只":{"docs":{},"有":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"才":{"docs":{},"会":{"docs":{},"尝":{"docs":{},"试":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},"、":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"。":{"docs":{},"否":{"docs":{},"则":{"docs":{},"只":{"docs":{},"做":{"docs":{},"数":{"docs":{},"据":{"docs":{},"流":{"docs":{},"转":{"docs":{},"发":{"docs":{},",":{"docs":{},"无":{"docs":{},"法":{"docs":{},"看":{"docs":{},"到":{"docs":{},"明":{"docs":{},"文":{"docs":{},"数":{"docs":{},"据":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"函":{"docs":{},"数":{"docs":{},",":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"普":{"docs":{},"通":{"docs":{},"的":{"docs":{},"字":{"docs":{},"符":{"docs":{},"串":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"修":{"docs":{},"改":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"请":{"docs":{},"求":{"docs":{},"头":{"docs":{},"(":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"的":{"docs":{},"内":{"docs":{},"容":{"docs":{},"包":{"docs":{},"括":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"(":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"同":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"中":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"加":{"docs":{},"载":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"后":{"docs":{},"将":{"docs":{},"终":{"docs":{},"端":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}}}},"命":{"docs":{},"令":{"docs":{},"行":{"docs":{},"启":{"docs":{},"动":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"时":{"docs":{},"配":{"docs":{},"置":{"docs":{},"`":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"端":{"docs":{},"口":{"docs":{},"号":{"8":{"0":{"0":{"1":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"在":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},"j":{"docs":{},"s":{"docs":{},"代":{"docs":{},"码":{"docs":{},"中":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"w":{"docs":{},"i":{"docs":{},"f":{"docs":{},"i":{"docs":{},"高":{"docs":{},"级":{"docs":{},"设":{"docs":{},"置":{"docs":{},"中":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"即":{"docs":{},"可":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"构":{"docs":{},"造":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"实":{"docs":{},"例":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},",":{"docs":{},"传":{"docs":{},"入":{"docs":{},"参":{"docs":{},"数":{"docs":{},"d":{"docs":{},"a":{"docs":{},"n":{"docs":{},"g":{"docs":{},"e":{"docs":{},"r":{"docs":{},"o":{"docs":{},"u":{"docs":{},"s":{"docs":{},"l":{"docs":{},"y":{"docs":{},"i":{"docs":{},"g":{"docs":{},"n":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"u":{"docs":{},"n":{"docs":{},"a":{"docs":{},"u":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"r":{"docs":{},"i":{"docs":{},"z":{"docs":{},"e":{"docs":{},"d":{"docs":{},":":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"中":{"docs":{},"发":{"docs":{},"生":{"docs":{},"错":{"docs":{},"误":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"调":{"docs":{},"用":{"docs":{},"o":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"r":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"并":{"docs":{},"提":{"docs":{},"供":{"docs":{},"对":{"docs":{},"应":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"基":{"docs":{},"于":{"docs":{},"n":{"docs":{},"o":{"docs":{},"d":{"docs":{},"e":{"docs":{},".":{"docs":{},"j":{"docs":{},"s":{"docs":{},",":{"docs":{},"开":{"docs":{},"放":{"docs":{},"二":{"docs":{},"次":{"docs":{},"开":{"docs":{},"发":{"docs":{},"能":{"docs":{},"力":{"docs":{},",":{"docs":{},"允":{"docs":{},"许":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"请":{"docs":{},"求":{"docs":{},"处":{"docs":{},"理":{"docs":{},"逻":{"docs":{},"辑":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"图":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"多":{"docs":{},"数":{"docs":{},"场":{"docs":{},"景":{"docs":{},"下":{"docs":{},",":{"docs":{},"错":{"docs":{},"误":{"docs":{},"会":{"docs":{},"在":{"docs":{},"请":{"docs":{},"求":{"docs":{},"目":{"docs":{},"标":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"的":{"docs":{},"时":{"docs":{},"候":{"docs":{},"发":{"docs":{},"生":{"docs":{},",":{"docs":{},"比":{"docs":{},"如":{"docs":{},"d":{"docs":{},"n":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},"失":{"docs":{},"败":{"docs":{},"、":{"docs":{},"请":{"docs":{},"求":{"docs":{},"超":{"docs":{},"时":{"docs":{},"等":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"如":{"docs":{},"下":{"docs":{},"几":{"docs":{},"种":{"docs":{},"方":{"docs":{},"案":{"docs":{},"都":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"来":{"docs":{},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"何":{"docs":{},"引":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"果":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"返":{"docs":{},"回":{"docs":{},"了":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},",":{"docs":{},"则":{"docs":{},"立":{"docs":{},"即":{"docs":{},"把":{"docs":{},"此":{"docs":{},"响":{"docs":{},"应":{"docs":{},"返":{"docs":{},"回":{"docs":{},"到":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"(":{"docs":{},"而":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{},"真":{"docs":{},"正":{"docs":{},"的":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},")":{"docs":{},",":{"docs":{},"流":{"docs":{},"程":{"docs":{},"结":{"docs":{},"束":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{},"了":{"docs":{},"全":{"docs":{},"局":{"docs":{},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"则":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"略":{"docs":{},"过":{"docs":{},"这":{"docs":{},"个":{"docs":{},"调":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"要":{"docs":{},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"解":{"docs":{},"析":{"docs":{},",":{"docs":{},"请":{"docs":{},"在":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"启":{"docs":{},"动":{"docs":{},"前":{"docs":{},"自":{"docs":{},"行":{"docs":{},"调":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},".":{"docs":{},"u":{"docs":{},"t":{"docs":{},"i":{"docs":{},"l":{"docs":{},"s":{"docs":{},".":{"docs":{},"c":{"docs":{},"e":{"docs":{},"r":{"docs":{},"t":{"docs":{},"m":{"docs":{},"g":{"docs":{},"r":{"docs":{},"相":{"docs":{},"关":{"docs":{},"方":{"docs":{},"法":{"docs":{},"生":{"docs":{},"成":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"并":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},"安":{"docs":{},"装":{"docs":{},"。":{"docs":{},"或":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"使":{"docs":{},"用":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"安":{"docs":{},"全":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"性":{"docs":{},"与":{"docs":{},"位":{"docs":{},"置":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"c":{"docs":{},"a":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"对":{"docs":{},"于":{"docs":{},"d":{"docs":{},"e":{"docs":{},"b":{"docs":{},"i":{"docs":{},"a":{"docs":{},"n":{"docs":{},"或":{"docs":{},"者":{"docs":{},"u":{"docs":{},"b":{"docs":{},"u":{"docs":{},"n":{"docs":{},"t":{"docs":{},"u":{"docs":{},"系":{"docs":{},"统":{"docs":{},",":{"docs":{},"在":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"之":{"docs":{},"前":{"docs":{},",":{"docs":{},"可":{"docs":{},"能":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"安":{"docs":{},"装":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"发":{"docs":{},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"入":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"当":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"请":{"docs":{},"求":{"docs":{},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"时":{"docs":{},",":{"docs":{},"具":{"docs":{},"体":{"docs":{},"处":{"docs":{},"理":{"docs":{},"过":{"docs":{},"程":{"docs":{},"是":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"收":{"docs":{},"到":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"时":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"可":{"docs":{},"以":{"docs":{},"替":{"docs":{},"换":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"对":{"docs":{},"请":{"docs":{},"求":{"docs":{},"做":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"启":{"docs":{},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"代":{"docs":{},"理":{"docs":{},"时":{"docs":{},",":{"docs":{},"w":{"docs":{},"s":{"docs":{},"s":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"不":{"docs":{},"会":{"docs":{},"被":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"记":{"docs":{},"录":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"开":{"docs":{},"启":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"访":{"docs":{},"问":{"docs":{},"特":{"docs":{},"定":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"站":{"docs":{},"点":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"会":{"docs":{},"提":{"docs":{},"示":{"docs":{},"该":{"docs":{},"站":{"docs":{},"点":{"docs":{},"不":{"docs":{},"是":{"docs":{},"一":{"docs":{},"个":{"docs":{},"安":{"docs":{},"全":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"这":{"docs":{},"通":{"docs":{},"常":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"设":{"docs":{},"置":{"docs":{},"不":{"docs":{},"能":{"docs":{},"被":{"docs":{},"正":{"docs":{},"确":{"docs":{},"识":{"docs":{},"别":{"docs":{},"导":{"docs":{},"致":{"docs":{},"的":{"docs":{},"(":{"docs":{},"比":{"docs":{},"如":{"docs":{},",":{"docs":{},"站":{"docs":{},"点":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"是":{"docs":{},"自":{"docs":{},"签":{"docs":{},"发":{"docs":{},"的":{"docs":{},")":{"docs":{},"。":{"docs":{},"如":{"docs":{},"果":{"docs":{},"您":{"docs":{},"信":{"docs":{},"任":{"docs":{},"该":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"可":{"docs":{},"以":{"docs":{},"用":{"docs":{},"以":{"docs":{},"下":{"docs":{},"方":{"docs":{},"式":{"docs":{},"来":{"docs":{},"继":{"docs":{},"续":{"docs":{},"访":{"docs":{},"问":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},",":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"端":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"快":{"docs":{},"速":{"docs":{},"开":{"docs":{},"始":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"我":{"docs":{},"们":{"docs":{},"自":{"docs":{},"然":{"docs":{},"也":{"docs":{},"可":{"docs":{},"以":{"docs":{},"借":{"docs":{},"助":{"docs":{},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"实":{"docs":{},"现":{"docs":{},"这":{"docs":{},"个":{"docs":{},"效":{"docs":{},"果":{"docs":{},",":{"docs":{},"而":{"docs":{},"且":{"docs":{},"我":{"docs":{},"们":{"docs":{},"还":{"docs":{},"可":{"docs":{},"以":{"docs":{},"控":{"docs":{},"制":{"docs":{},"到":{"docs":{},"只":{"docs":{},"允":{"docs":{},"许":{"docs":{},"指":{"docs":{},"定":{"docs":{},"网":{"docs":{},"址":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"对":{"docs":{},"不":{"docs":{},"在":{"docs":{},"列":{"docs":{},"表":{"docs":{},"的":{"docs":{},"网":{"docs":{},"址":{"docs":{},",":{"docs":{},"进":{"docs":{},"行":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"强":{"docs":{},"验":{"docs":{},"证":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"b":{"docs":{},"i":{"docs":{},"n":{"docs":{},".":{"docs":{},"o":{"docs":{},"r":{"docs":{},"g":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"找":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"刚":{"docs":{},"导":{"docs":{},"入":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"证":{"docs":{},"书":{"docs":{},",":{"docs":{},"配":{"docs":{},"置":{"docs":{},"为":{"docs":{},"信":{"docs":{},"任":{"docs":{},"(":{"docs":{},"a":{"docs":{},"l":{"docs":{},"w":{"docs":{},"a":{"docs":{},"y":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"把":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}},"响":{"docs":{},"应":{"docs":{},"信":{"docs":{},"息":{"docs":{},"返":{"docs":{},"回":{"docs":{},"给":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"所":{"docs":{},"有":{"docs":{},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"用":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"协":{"docs":{},"议":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"拦":{"docs":{},"截":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"对":{"docs":{},"内":{"docs":{},"容":{"docs":{},"做":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"发":{"docs":{},"送":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"并":{"docs":{},"修":{"docs":{},"改":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"正":{"docs":{},"在":{"docs":{},"发":{"docs":{},"送":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"提":{"docs":{},"供":{"docs":{},"g":{"docs":{},"u":{"docs":{},"i":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"用":{"docs":{},"以":{"docs":{},"观":{"docs":{},"察":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"了":{"docs":{},"多":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"可":{"docs":{},"在":{"docs":{},"下":{"docs":{},"载":{"docs":{},"安":{"docs":{},"装":{"docs":{},"时":{"docs":{},"选":{"docs":{},"择":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"支":{"docs":{},"持":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"的":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"收":{"docs":{},"集":{"docs":{},"请":{"docs":{},"求":{"docs":{},"所":{"docs":{},"有":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"m":{"docs":{},"e":{"docs":{},"t":{"docs":{},"h":{"docs":{},"o":{"docs":{},"d":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"改":{"docs":{},"成":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"并":{"docs":{},"发":{"docs":{},"送":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"文":{"docs":{},"件":{"docs":{},"(":{"docs":{},"已":{"docs":{},"知":{"docs":{},"如":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"时":{"docs":{},",":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"t":{"docs":{},"a":{"docs":{},"i":{"docs":{},"l":{"docs":{},"参":{"docs":{},"数":{"docs":{},"内":{"docs":{},"容":{"docs":{},"大":{"docs":{},"致":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"否":{"docs":{},"启":{"docs":{},"用":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"版":{"docs":{},"界":{"docs":{},"面":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"处":{"docs":{},"理":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}},"屏":{"docs":{},"蔽":{"docs":{},"所":{"docs":{},"有":{"docs":{},"c":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"o":{"docs":{},"l":{"docs":{},"e":{"docs":{},"输":{"docs":{},"出":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"开":{"docs":{},"启":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"s":{"docs":{},"o":{"docs":{},"c":{"docs":{},"k":{"docs":{},"e":{"docs":{},"t":{"docs":{},"代":{"docs":{},"理":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"强":{"docs":{},"制":{"docs":{},"拦":{"docs":{},"截":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},",":{"docs":{},"忽":{"docs":{},"略":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"请":{"docs":{},"求":{"docs":{},"中":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"错":{"docs":{},"误":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"f":{"docs":{},"a":{"docs":{},"l":{"docs":{},"s":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}},"必":{"docs":{},"选":{"docs":{},"字":{"docs":{},"段":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"本":{"docs":{},"文":{"docs":{},"档":{"docs":{},"的":{"docs":{},"适":{"docs":{},"用":{"docs":{},"范":{"docs":{},"围":{"docs":{},"是":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"质":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"来":{"docs":{},"加":{"docs":{},"载":{"docs":{},"模":{"docs":{},"块":{"docs":{},"并":{"docs":{},"体":{"docs":{},"验":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"忽":{"docs":{},"略":{"docs":{},"证":{"docs":{},"书":{"docs":{},"认":{"docs":{},"证":{"docs":{},"的":{"docs":{},"错":{"docs":{},"误":{"docs":{},"。":{"docs":{},"需":{"docs":{},"要":{"docs":{},"注":{"docs":{},"意":{"docs":{},"的":{"docs":{},"是":{"docs":{},",":{"docs":{},"该":{"docs":{},"参":{"docs":{},"数":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"生":{"docs":{},"效":{"docs":{},"的":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"你":{"docs":{},"在":{"docs":{},"此":{"docs":{},"期":{"docs":{},"间":{"docs":{},"访":{"docs":{},"问":{"docs":{},"了":{"docs":{},"其":{"docs":{},"他":{"docs":{},"未":{"docs":{},"知":{"docs":{},"的":{"docs":{},"网":{"docs":{},"站":{"docs":{},",":{"docs":{},"他":{"docs":{},"们":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"也":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{},",":{"docs":{},"这":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"带":{"docs":{},"来":{"docs":{},"安":{"docs":{},"全":{"docs":{},"隐":{"docs":{},"患":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"查":{"docs":{},"看":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"校":{"docs":{},"验":{"docs":{},"系":{"docs":{},"统":{"docs":{},"内":{"docs":{},"是":{"docs":{},"否":{"docs":{},"存":{"docs":{},"在":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"根":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"样":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"根":{"docs":{},"据":{"docs":{},"请":{"docs":{},"求":{"docs":{},"参":{"docs":{},"数":{"docs":{},",":{"docs":{},"向":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"发":{"docs":{},"出":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"接":{"docs":{},"收":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"响":{"docs":{},"应":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"介":{"docs":{},"绍":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"此":{"docs":{},"处":{"docs":{},"无":{"docs":{},"法":{"docs":{},"控":{"docs":{},"制":{"docs":{},"向":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"无":{"docs":{},"需":{"docs":{},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"注":{"docs":{},"意":{"docs":{},":":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"引":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"前":{"docs":{},",":{"docs":{},"请":{"docs":{},"务":{"docs":{},"必":{"docs":{},"确":{"docs":{},"保":{"docs":{},"文":{"docs":{},"件":{"docs":{},"来":{"docs":{},"源":{"docs":{},"可":{"docs":{},"靠":{"docs":{},",":{"docs":{},"以":{"docs":{},"免":{"docs":{},"发":{"docs":{},"生":{"docs":{},"安":{"docs":{},"全":{"docs":{},"问":{"docs":{},"题":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{},"规":{"docs":{},"则":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}}}}},"点":{"docs":{},"击":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"然":{"docs":{},"后":{"docs":{},",":{"docs":{},"安":{"docs":{},"装":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"生":{"docs":{},"成":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"r":{"docs":{},"o":{"docs":{},"o":{"docs":{},"t":{"docs":{},"c":{"docs":{},"a":{"docs":{},",":{"docs":{},"完":{"docs":{},"成":{"docs":{},"后":{"docs":{},"请":{"docs":{},"引":{"docs":{},"导":{"docs":{},"用":{"docs":{},"户":{"docs":{},"信":{"docs":{},"任":{"docs":{},".":{"docs":{},"c":{"docs":{},"r":{"docs":{},"t":{"docs":{},"文":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"并":{"docs":{},"解":{"docs":{},"析":{"docs":{},"所":{"docs":{},"有":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"用":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{},"发":{"docs":{},"请":{"docs":{},"求":{"docs":{},"测":{"docs":{},"试":{"docs":{},"的":{"docs":{},"方":{"docs":{},"法":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"测":{"docs":{},"试":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"测":{"docs":{},"试":{"docs":{},":":{"docs":{},"配":{"docs":{},"置":{"docs":{},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"为":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"会":{"docs":{},"被":{"docs":{},"解":{"docs":{},"析":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"p":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},"数":{"docs":{},"据":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}},"u":{"docs":{},"s":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"证":{"docs":{},"书":{"docs":{},"文":{"docs":{},"件":{"docs":{},",":{"docs":{},"少":{"docs":{},"部":{"docs":{},"分":{"docs":{},"仅":{"docs":{},"支":{"docs":{},"持":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"请":{"docs":{},"求":{"docs":{},"全":{"docs":{},"部":{"docs":{},"改":{"docs":{},"到":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},",":{"docs":{},"使":{"docs":{},"用":{"docs":{},"本":{"docs":{},"地":{"docs":{},"数":{"docs":{},"据":{"docs":{},"代":{"docs":{},"替":{"docs":{},"服":{"docs":{},"务":{"docs":{},"端":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{},"测":{"docs":{},"试":{"docs":{},"信":{"docs":{},"息":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"docs":{}}}}}}}}}}}}},"头":{"docs":{},"里":{"docs":{},"加":{"docs":{},"上":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"最":{"docs":{},"后":{"docs":{},"追":{"docs":{},"加":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"签":{"docs":{},"名":{"docs":{},",":{"docs":{},"并":{"docs":{},"延":{"docs":{},"迟":{"5":{"docs":{},"秒":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"docs":{}}}}}}}}}}}}}}}}}}}},"状":{"docs":{},"态":{"docs":{},"码":{"docs":{},"都":{"docs":{},"改":{"docs":{},"成":{"4":{"0":{"4":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"直":{"docs":{},"接":{"docs":{},"请":{"docs":{},"求":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"客":{"docs":{},"户":{"docs":{},"端":{"docs":{},",":{"docs":{},"不":{"docs":{},"再":{"docs":{},"发":{"docs":{},"起":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"中":{"docs":{},"s":{"docs":{},"t":{"docs":{},"a":{"docs":{},"t":{"docs":{},"u":{"docs":{},"s":{"docs":{},"c":{"docs":{},"o":{"docs":{},"d":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}},"相":{"docs":{},"比":{"3":{"docs":{},".":{"docs":{},"x":{"docs":{},"版":{"docs":{},"本":{"docs":{},",":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},"docs":{}}},"确":{"docs":{},"认":{"docs":{},"将":{"docs":{},"证":{"docs":{},"书":{"docs":{},"添":{"docs":{},"加":{"docs":{},"到":{"docs":{},"l":{"docs":{},"o":{"docs":{},"g":{"docs":{},"i":{"docs":{},"n":{"docs":{},"或":{"docs":{},"s":{"docs":{},"y":{"docs":{},"s":{"docs":{},"t":{"docs":{},"e":{"docs":{},"m":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"示":{"docs":{},"例":{"docs":{"./":{"ref":"./","tf":0.0029476787030213707}}}},"简":{"docs":{},"介":{"docs":{"./":{"ref":"./","tf":10}}},"化":{"docs":{},"了":{"docs":{},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"内":{"docs":{},"的":{"docs":{},"接":{"docs":{},"口":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"管":{"docs":{},"理":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"系":{"docs":{},"统":{"docs":{},"的":{"docs":{},"全":{"docs":{},"局":{"docs":{},"代":{"docs":{},"理":{"docs":{},"配":{"docs":{},"置":{"docs":{},",":{"docs":{},"方":{"docs":{},"法":{"docs":{},"调":{"docs":{},"用":{"docs":{},"时":{"docs":{},"可":{"docs":{},"能":{"docs":{},"会":{"docs":{},"弹":{"docs":{},"出":{"docs":{},"密":{"docs":{},"码":{"docs":{},"框":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"类":{"docs":{},"似":{"docs":{},"这":{"docs":{},"种":{"docs":{},"报":{"docs":{},"错":{"docs":{},"都":{"docs":{},"是":{"docs":{},"因":{"docs":{},"为":{"docs":{},"系":{"docs":{},"统":{"docs":{},"没":{"docs":{},"有":{"docs":{},"信":{"docs":{},"任":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"生":{"docs":{},"成":{"docs":{},"的":{"docs":{},"c":{"docs":{},"a":{"docs":{},"所":{"docs":{},"造":{"docs":{},"成":{"docs":{},"的":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"经":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"后":{"docs":{},",":{"docs":{},"期":{"docs":{},"望":{"docs":{},"的":{"docs":{},"返":{"docs":{},"回":{"docs":{},"如":{"docs":{},"下":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"被":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"后":{"docs":{},"的":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"处":{"docs":{},"理":{"docs":{},"流":{"docs":{},"程":{"docs":{},"同":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"一":{"docs":{},"致":{"docs":{},"。":{"docs":{},"未":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"会":{"docs":{},"再":{"docs":{},"进":{"docs":{},"入":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"做":{"docs":{},"处":{"docs":{},"理":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"规":{"docs":{},"则":{"docs":{},"文":{"docs":{},"件":{"docs":{},"中":{"docs":{},",":{"docs":{},"除":{"docs":{},"了":{"docs":{},"s":{"docs":{},"u":{"docs":{},"m":{"docs":{},"m":{"docs":{},"a":{"docs":{},"r":{"docs":{},"y":{"docs":{},",":{"docs":{},"都":{"docs":{},"是":{"docs":{},"由":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"内":{"docs":{},"提":{"docs":{},"供":{"docs":{},"`":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"`":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"返":{"docs":{},"回":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"(":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},")":{"docs":{},"全":{"docs":{},"面":{"docs":{},"支":{"docs":{},"持":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"和":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"模":{"docs":{},"块":{"docs":{},"应":{"docs":{},"该":{"docs":{},"符":{"docs":{},"合":{"docs":{},"c":{"docs":{},"m":{"docs":{},"d":{"docs":{},"规":{"docs":{},"范":{"docs":{},",":{"docs":{},"一":{"docs":{},"个":{"docs":{},"典":{"docs":{},"型":{"docs":{},"的":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"代":{"docs":{},"码":{"docs":{},"结":{"docs":{},"构":{"docs":{},"如":{"docs":{},"下":{"docs":{},"。":{"docs":{},"模":{"docs":{},"块":{"docs":{},"中":{"docs":{},"所":{"docs":{},"有":{"docs":{},"方":{"docs":{},"法":{"docs":{},"都":{"docs":{},"是":{"docs":{},"可":{"docs":{},"选":{"docs":{},"的":{"docs":{},",":{"docs":{},"只":{"docs":{},"需":{"docs":{},"实":{"docs":{},"现":{"docs":{},"业":{"docs":{},"务":{"docs":{},"感":{"docs":{},"兴":{"docs":{},"趣":{"docs":{},"的":{"docs":{},"部":{"docs":{},"分":{"docs":{},"即":{"docs":{},"可":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"是":{"docs":{},"用":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"的":{"docs":{},"介":{"docs":{},"绍":{"docs":{},"文":{"docs":{},"案":{"docs":{},",":{"docs":{},"用":{"docs":{},"于":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},"提":{"docs":{},"示":{"docs":{},"用":{"docs":{},"户":{"docs":{},",":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}},"能":{"docs":{},"力":{"docs":{},"范":{"docs":{},"围":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"解":{"docs":{},"析":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"请":{"docs":{},"求":{"docs":{},"的":{"docs":{},"原":{"docs":{},"理":{"docs":{},"是":{"docs":{},"中":{"docs":{},"间":{"docs":{},"人":{"docs":{},"攻":{"docs":{},"击":{"docs":{},"(":{"docs":{},"m":{"docs":{},"a":{"docs":{},"n":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}},"警":{"docs":{},"告":{"docs":{},":":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{},"和":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"全":{"docs":{},"息":{"docs":{},"息":{"docs":{},"相":{"docs":{},"关":{"docs":{},",":{"docs":{},"建":{"docs":{},"议":{"docs":{},"亲":{"docs":{},"自":{"docs":{},"生":{"docs":{},"成":{"docs":{},",":{"docs":{},"并":{"docs":{},"妥":{"docs":{},"善":{"docs":{},"保":{"docs":{},"管":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"设":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.002210759027266028}},"属":{"docs":{},"性":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"访":{"docs":{},"问":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},":":{"docs":{},"/":{"docs":{},"/":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"2":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{}}}}}}}}}},"证":{"docs":{},"书":{"docs":{},"下":{"docs":{},"载":{"docs":{},"到":{"docs":{},"指":{"docs":{},"定":{"docs":{},"目":{"docs":{},"录":{"docs":{},"后":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"从":{"docs":{},"其":{"docs":{},"他":{"docs":{},"入":{"docs":{},"口":{"docs":{},"进":{"docs":{},"行":{"docs":{},"安":{"docs":{},"装":{"docs":{},",":{"docs":{},"包":{"docs":{},"括":{"docs":{},":":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"请":{"docs":{},"求":{"docs":{},"b":{"docs":{},"o":{"docs":{},"d":{"docs":{},"i":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}},"使":{"docs":{},"用":{"docs":{},"的":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"或":{"docs":{},"者":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"出":{"docs":{},"错":{"docs":{},"的":{"docs":{},"事":{"docs":{},"件":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"的":{"docs":{},"原":{"docs":{},"始":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{"./":{"ref":"./","tf":0.0014738393515106854}}}}}}}}}}}},"目":{"docs":{},"标":{"docs":{},"的":{"docs":{},"h":{"docs":{},"o":{"docs":{},"s":{"docs":{},"t":{"docs":{},",":{"docs":{},"受":{"docs":{},"制":{"docs":{},"于":{"docs":{},"协":{"docs":{},"议":{"docs":{},",":{"docs":{},"这":{"docs":{},"里":{"docs":{},"无":{"docs":{},"法":{"docs":{},"获":{"docs":{},"取":{"docs":{},"完":{"docs":{},"整":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"调":{"docs":{},"用":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"b":{"docs":{},"e":{"docs":{},"f":{"docs":{},"o":{"docs":{},"r":{"docs":{},"e":{"docs":{},"d":{"docs":{},"e":{"docs":{},"a":{"docs":{},"l":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"s":{"docs":{},"r":{"docs":{},"e":{"docs":{},"q":{"docs":{},"u":{"docs":{},"e":{"docs":{},"s":{"docs":{},"t":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"如":{"docs":{},"果":{"docs":{},"返":{"docs":{},"回":{"docs":{},"t":{"docs":{},"r":{"docs":{},"u":{"docs":{},"e":{"docs":{},",":{"docs":{},"会":{"docs":{},"明":{"docs":{},"文":{"docs":{},"解":{"docs":{},"析":{"docs":{},"这":{"docs":{},"个":{"docs":{},"请":{"docs":{},"求":{"docs":{},",":{"docs":{},"其":{"docs":{},"他":{"docs":{},"请":{"docs":{},"求":{"docs":{},"不":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"s":{"docs":{},"e":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"e":{"docs":{},"s":{"docs":{},"p":{"docs":{},"o":{"docs":{},"n":{"docs":{},"s":{"docs":{},"e":{"docs":{},"方":{"docs":{},"法":{"docs":{},",":{"docs":{},"由":{"docs":{},"模":{"docs":{},"块":{"docs":{},"对":{"docs":{},"响":{"docs":{},"应":{"docs":{},"内":{"docs":{},"容":{"docs":{},"进":{"docs":{},"行":{"docs":{},"处":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"运":{"docs":{},"行":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}},"返":{"docs":{},"回":{"docs":{},"值":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"错":{"docs":{},"误":{"docs":{},"页":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}},"这":{"docs":{},"里":{"docs":{},"提":{"docs":{},"供":{"docs":{},"一":{"docs":{},"些":{"docs":{},"样":{"docs":{},"例":{"docs":{},",":{"docs":{},"来":{"docs":{},"讲":{"docs":{},"解":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},"的":{"docs":{},"常":{"docs":{},"见":{"docs":{},"用":{"docs":{},"法":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}},"通":{"docs":{},"过":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{},"请":{"docs":{},"求":{"docs":{},":":{"docs":{},"c":{"docs":{},"u":{"docs":{},"r":{"docs":{},"l":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"参":{"docs":{},"数":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}},"自":{"docs":{},"定":{"docs":{},"义":{"docs":{},"的":{"docs":{},"r":{"docs":{},"u":{"docs":{},"l":{"docs":{},"e":{"docs":{},"来":{"docs":{},"修":{"docs":{},"改":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"这":{"docs":{},"种":{"docs":{},"方":{"docs":{},"式":{"docs":{},"初":{"docs":{},"始":{"docs":{},"化":{"docs":{},"的":{"docs":{},"a":{"docs":{},"n":{"docs":{},"y":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"x":{"docs":{},"y":{"docs":{},",":{"docs":{},"其":{"docs":{},"配":{"docs":{},"置":{"docs":{},"也":{"docs":{},"是":{"docs":{},"全":{"docs":{},"局":{"docs":{},"性":{"docs":{},"的":{"docs":{},",":{"docs":{},"所":{"docs":{},"有":{"docs":{},"网":{"docs":{},"站":{"docs":{},"的":{"docs":{},"证":{"docs":{},"书":{"docs":{},"问":{"docs":{},"题":{"docs":{},"都":{"docs":{},"会":{"docs":{},"被":{"docs":{},"忽":{"docs":{},"略":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"配":{"docs":{},"置":{"1":{"2":{"7":{"docs":{},".":{"0":{"docs":{},".":{"0":{"docs":{},".":{"1":{"docs":{},":":{"8":{"0":{"0":{"1":{"docs":{},"为":{"docs":{},"全":{"docs":{},"局":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{},"服":{"docs":{},"务":{"docs":{},"器":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}}},"docs":{}},"docs":{}},"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"/":{"docs":{},"a":{"docs":{},"n":{"docs":{},"d":{"docs":{},"r":{"docs":{},"o":{"docs":{},"i":{"docs":{},"d":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"o":{"docs":{},"s":{"docs":{},"x":{"docs":{},"系":{"docs":{},"统":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}},"启":{"docs":{},"动":{"docs":{},"端":{"docs":{},"口":{"docs":{},",":{"docs":{},"如":{"1":{"0":{"8":{"0":{"docs":{},"端":{"docs":{},"口":{"docs":{},"启":{"docs":{},"动":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}},"docs":{}},"docs":{}},"docs":{}},"docs":{}}}}}}},"浏":{"docs":{},"览":{"docs":{},"器":{"docs":{},"h":{"docs":{},"t":{"docs":{},"t":{"docs":{},"p":{"docs":{},"代":{"docs":{},"理":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"附":{"docs":{},"录":{"docs":{},":":{"docs":{},"如":{"docs":{},"何":{"docs":{},"信":{"docs":{},"任":{"docs":{},"c":{"docs":{},"a":{"docs":{},"证":{"docs":{},"书":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}},"限":{"docs":{},"速":{"docs":{},"值":{"docs":{},",":{"docs":{},"单":{"docs":{},"位":{"docs":{},"k":{"docs":{},"b":{"docs":{},"/":{"docs":{},"s":{"docs":{},",":{"docs":{},"默":{"docs":{},"认":{"docs":{},"不":{"docs":{},"限":{"docs":{},"速":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"除":{"docs":{},"了":{"docs":{},"上":{"docs":{},"述":{"docs":{},"证":{"docs":{},"书":{"docs":{},"安":{"docs":{},"装":{"docs":{},"过":{"docs":{},"程":{"docs":{},",":{"docs":{},"还":{"docs":{},"需":{"docs":{},"要":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}},"需":{"docs":{},"要":{"docs":{},"编":{"docs":{},"写":{"docs":{},"一":{"docs":{},"个":{"docs":{},"规":{"docs":{},"则":{"docs":{},"模":{"docs":{},"块":{"docs":{},",":{"docs":{},"在":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}},"首":{"docs":{},"先":{"docs":{},"和":{"docs":{},"i":{"docs":{},"o":{"docs":{},"s":{"docs":{},"类":{"docs":{},"似":{"docs":{},",":{"docs":{},"需":{"docs":{},"要":{"docs":{},"先":{"docs":{},"扫":{"docs":{},"描":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"二":{"docs":{},"维":{"docs":{},"码":{"docs":{},"进":{"docs":{},"行":{"docs":{},"下":{"docs":{},"载":{"docs":{},"。":{"docs":{},"然":{"docs":{},"后":{"docs":{},"不":{"docs":{},"同":{"docs":{},"的":{"docs":{},"安":{"docs":{},"卓":{"docs":{},"系":{"docs":{},"统":{"docs":{},"安":{"docs":{},"装":{"docs":{},"证":{"docs":{},"书":{"docs":{},"的":{"docs":{},"方":{"docs":{},"式":{"docs":{},"可":{"docs":{},"能":{"docs":{},"有":{"docs":{},"所":{"docs":{},"不":{"docs":{},"同":{"docs":{},",":{"docs":{},"但":{"docs":{},"是":{"docs":{},"安":{"docs":{},"装":{"docs":{},"的":{"docs":{},"步":{"docs":{},"骤":{"docs":{},"是":{"docs":{},"类":{"docs":{},"似":{"docs":{},"的":{"docs":{},",":{"docs":{},"我":{"docs":{},"们":{"docs":{},"列":{"docs":{},"举":{"docs":{},"了":{"docs":{},"几":{"docs":{},"种":{"docs":{},"类":{"docs":{},"型":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},"驱":{"docs":{},"动":{"docs":{},"的":{"docs":{},",":{"docs":{},"函":{"docs":{},"数":{"docs":{},"需":{"docs":{},"要":{"docs":{},"满":{"docs":{},"足":{"docs":{},"y":{"docs":{},"i":{"docs":{},"e":{"docs":{},"l":{"docs":{},"d":{"docs":{},"a":{"docs":{},"b":{"docs":{},"l":{"docs":{},"e":{"docs":{},"。":{"docs":{},"可":{"docs":{},"以":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"方":{"docs":{},"法":{"docs":{},"或":{"docs":{},"是":{"docs":{},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}},"返":{"docs":{},"回":{"docs":{},"p":{"docs":{},"r":{"docs":{},"o":{"docs":{},"m":{"docs":{},"i":{"docs":{},"s":{"docs":{},"e":{"docs":{},"或":{"docs":{},"使":{"docs":{},"用":{"docs":{},"g":{"docs":{},"e":{"docs":{},"n":{"docs":{},"e":{"docs":{},"r":{"docs":{},"a":{"docs":{},"t":{"docs":{},"o":{"docs":{},"r":{"docs":{},"函":{"docs":{},"数":{"docs":{},"。":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}}},",":{"docs":{},"w":{"docs":{},"e":{"docs":{},"b":{"docs":{},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"所":{"docs":{},"有":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}}}},"界":{"docs":{},"面":{"docs":{},"上":{"docs":{},"能":{"docs":{},"看":{"docs":{},"到":{"docs":{},"刚":{"docs":{},"才":{"docs":{},"的":{"docs":{},"请":{"docs":{},"求":{"docs":{},"信":{"docs":{},"息":{"docs":{"./":{"ref":"./","tf":0.0007369196757553427}}}}}}}}}}}}}}}},",":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}},"[":{"docs":{},"s":{"docs":{},"w":{"docs":{},"i":{"docs":{},"t":{"docs":{},"c":{"docs":{},"h":{"docs":{},"y":{"docs":{},"o":{"docs":{},"m":{"docs":{},"e":{"docs":{},"g":{"docs":{},"a":{"docs":{},"]":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}}}}}},"j":{"docs":{},"a":{"docs":{},"v":{"docs":{},"a":{"docs":{},"s":{"docs":{},"c":{"docs":{},"r":{"docs":{},"i":{"docs":{},"p":{"docs":{},"t":{"docs":{},".":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}}}}}}}}}},"s":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},":":{"docs":{"./":{"ref":"./","tf":0.00044863167339614175}}}},"length":1376},"corpusTokens":["\"content","\"curl/7.43.0\"","\"data\":","\"https\"","\"i","\"thi","\"turn","\"user","\"x","#gener","#global","#launch","#local","#全局包","#启动anyproxy,并解析所有https请求","#本地包","#生成rootca证书,生成后需要手动信任","&","&&","'","'...'","'...',","'/',","'/index.html';","'/redirect/to/another/path';","'127.0.0.1'","'8001');","'8008';","';","'a","'anyproxy';","'anyproxy.io',","'anyproxy/0.0.0';","'application/json'","'cach","'close',","'content","'get',","'get';","'hello","'http',","'http';","'http://anyproxy.io/',","'http://httpbin.org/get?show_env=1'","'http://httpbin.org/us","'https'","'https',","'i","'image/gif',","'keep","'mi","'origin","'proxi","'rule","'text/html'","'thi","'user","'{\"hello\":","(!anyproxy.utils.certmgr.ifrootcafileexists())","(!error)","()","(e)","(iswin)","(requestdetail.url","(requestdetail.url.indexof('http://httpbin.org')","(requestdetail.url.indexof('http://httpbin.org/post')","(requestdetail.url.indexof('http://httpbin.org/us","(requestdetail.url.indexof('https://httpbin.org/us","(requestdetail.url.indexof('https://th","*/","*/}","*beforedealhttpsrequest(requestdetail)","*beforesendrequest(requestdetail)","*beforesendresponse(requestdetail,","*onconnecterror(requestdetail,","*onerror(requestdetail,","*true*","+=",",",".",".',","...","...,","./myrulepkg/","./rule.j",".cer",".crt","/*","//","//when","//这里也可以同时加上requestopt","/^win/.test(process.platform);","0)","1.","10.3","10.3信任ca证书","10000,","1080","127.0.0.1:8001","127.0.0.1:8001,","127.0.0.1:8001/index.html","127.0.0.1:8001,访问","1,write","1,编写规则","2,","2.","200,","3,","3.x:","4,","4.0的主要变化:","4.0,欢迎提供反馈","404","404;","443;","5","5000);","5s","80,","8001","8001,","8002","=","===","=>",">",">=",">关于本机",">证书信任设置",">通用","[switchyomega]","_re","_req","_req:","_res:","a:","abil","abliti","activ","add","address","advanc","advance.","agent","agent\":","agent'","agent')","agent':","agent']","agent,","alive',","all,","alway","android","anyprox","anyproxi","anyproxy',","anyproxy,","anyproxy.","anyproxy.io","anyproxy.io,","anyproxy.proxycore(options);","anyproxy.proxyserv","anyproxy.proxyserver(options)","anyproxy.proxyserver(options);","anyproxy.utils.certmgr","anyproxy.utils.certmgr.generaterootca((error,","anyproxy.utils.certmgr.generaterootca(callback)","anyproxy.utils.certmgr.ifrootcafileexists()","anyproxy.utils.systemproxymgr","anyproxy.utils.systemproxymgr.disableglobalproxy();","anyproxy.utils.systemproxymgr.enableglobalproxy('127.0.0.1',","anyproxyin","anyproxyins.start();","anyproxy使用require()加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包","anyproxy可以作为一个npm模块使用,整合进其他工具。","anyproxy向客户端发送请求前,会调用beforesendresponse,并带上参数requestdetail","anyproxy向服务端发送请求前,会调用beforesendrequest,并带上参数requestdetail","anyproxy在与目标https服务器建立连接的过程中,如果发生错误,anyproxy会调用这个方法","anyproxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。","anyproxy收到https请求时,会调用beforedealhttpsrequest,并带上参数requestdetail","anyproxy是一个开放式的http代理服务器。","anyproxy调用规则模块beforesendrequest方法,由模块做处理,返回新的请求参数,或返回响应内容","anyproxy默认不对https请求做处理,如需看到明文信息,需要配置ca证书","append","appendix:how","apt","as:","at',","attack","attack.","attack),需要客户端提前信任anyproxy生成的ca","automat","base","be","befor","beforedealhttpsrequest","beforedealhttpsrequest(requestdetail)","beforesendrequest","beforesendrequest(requestdetail)","beforesendrespons","beforesendresponse(requestdetail,","below,","below.","below:","besid","bodi","body,","body:","body三个字段","body等","body),甚至是请求的目标地址等","bonou","browser","browser.","by']","by:","by:anyproxi","ca","ca,","ca.","call","card,","caus","ca方法。","ca,按提示扫描二维码即可安装","cert","certdir","certdir);","certif","certificate.","certificates,","certificates.","certifyc","chang","chart","choos","chrome","class:","cli,","cli.","click","clicking,","client","client,","close","co","co.","code","code)、响应头(respons","code,","code.","collect","command","common","compat","config","config.","configur","connect","connection':","connection:","connections.","console,","console.error('error","console.log('th","const","content","content,","control':","couldn't","course.","creat","credenti","curl","curl:","custom","cwd:","d","dangerouslyignoreunauthor","dangerouslyignoreunauthorized:","dangerouslyignoreunauthorized:true,","data","data\"","data'","data',","deal","dealed.","deall","debian","decrypt","default","default.","defin","delay","demo","design","detect","determin","differ","directli","disabl","dn","doc","doubl","download","driven","due","dure","e.g.","easist","edit","emit","enabl","enable:","encrypt","end","english","entir","error","error)","error);","errors,","event:","example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例","except","exec","exec('open","exec('start","exist","expect","extens","extent","face","fail","failure,","fals","false,","false;","faq","feature,","features:","few","field","file","file,","file.","file:","find","finish","first","flow","follow","follows.","forc","forceproxyhttp","forceproxyhttps:","forward","found","frequent","full","fulli","function","function()","function.","g","gener","get","github:","github主页:https://github.com/alibaba/anyproxi","give","global","go,","gui","guid","h","hack","hacked!","hacked!';","handl","handling.","happen","header","header),请求体(request","header,","header:","headers:","headersar","header)、响应内容等","help","here","here,","here.","hood,","host","host:","hostname:","http","http.rquest,","http/1.1","http/http","http://....j","http://127.0.0.1:8001","http://127.0.0.1:8002","http://127.0.0.1:8002,","http://httpbin.org","http://httpbin.org/","http://httpbin.org/post","http://httpbin.org/us","http://localhost:8008/index.html","httpbin.org","httpbin.org,","https://github.com/alibaba/anyproxi","https://httpbin.org/us","https://nodejs.org/api/http.html#http_http_request_options_callback","https://sample.com/rule.j","https连接服务器出错","http代理配置","http的代理模式中,这里的request是connect请求","i.e.","ignor","import","in.","includ","influenc","info","info,","insid","instal","installing.","instruct","intercept","intercept`参数,按npm模块启动时配置`forceproxyhttps`参数,所有https请求都会被替换证书并解析","intercept后才会从界面上看到相应内容。","interest","interfac","interface,","introduct","invok","io","ios/android","ios系统信任ca证书","issu","iswin","it,","it.","javascript.","js","kb/s,","keep","keypath)","kind","know","know.com')","known","launch","launching,","legaci","line","line,","list","load","local","localrespons","locat","log","logic","login","longer.","luanch","make","man","manag","manual","menu","menu,","messag","method","method,","method:","middl","middle),用户必须信任anyproxy生成的ca证书,才能进行后续流程","mind","modifi","modul","module,","module.","module.export","myrulepkg","nativ","need","network.","new","newli","newopt","newoption.path","newoption.port","newrequestopt","newrequestoptions.headers['us","newrequestoptions.hostnam","newrequestoptions.method","newrequestoptions.path","newrequestoptions.port","newrequestoptions.rejectunauthor","newrespons","newresponse.bodi","newresponse.header['x","newresponse.statuscod","node.j","nodej","nodejs.","nothing,","now","npm","null","null;","number","object","object.","object.assign({},","offer","on","on\"","on,","onconnecterror","onconnecterror(requestdetail,","one.","onerror","onerror(requestdetail,","onlin","oppo","option","option,","optional,","osx","osx系统信任ca证书","otherwire,","otherwise,","over","own.","page","param","part","pass","password","path.","path:","phones.","place","pleas","point","popular,","port","port:","post","privat","process","promis","promise((resolve,","propmt","protocol","protocol\":","protocol,","protocol:","protocol,i.e.","provid","proxi","proxy.close();","proxy.on('error',","proxy.on('ready',","proxy.start();","proxyserv","proxyserver.close();","proxyserver.on('error',","proxyserver.on('ready',","proxyserver.start();","q:","qr","quest","r15.","r15),anyproxi","read","readi","receiv","record","redirect","ref:","refer","reject)","rejectunauthor","replac","request","request,","request.","requestdata","requestdata:","requestdetail","requestdetail.protocol","requestdetail.requestoptions);","requestdetail.requestoptions;","requestdetail;","requesting,","requestopt","requestoptions:","requests,","requests.","requir","require('anyproxy');","require('child_process').exec;","require('http').request","require('myrulemodule'),","require('path').dirname(keypath);","require()","required,","resolve({","respons","response\"}'","response',","response:","responsedetail","responsedetail)","responsedetail.response);","responsedetail.response;","return","right","risk.","root","rootca","rootca',","rootca.crt","rule","rule:","rule_sample/sample_modify_request_data.j","rule_sample/sample_modify_request_header.j","rule_sample/sample_modify_request_path.j","rule_sample/sample_modify_request_protocol.j","rule_sample/sample_modify_response_data.j","rule_sample/sample_modify_response_header.j","rule_sample/sample_modify_response_statuscode.j","rule_sample/sample_use_local_response.j","rule接口文档","rule样例","rule模块","run","safari","safe","same","sampl","sample.j","sample:","save","scan","scenario,","sd","second","seconds.","secur","security.","see","self","send","sent","server","server,","server.","set","settimeout(()","settings.","sever","side","signed.","silent","silent:","similar","simpl","simplifi","site","site(s)","situation.","skip","someth","source.","specifi","ssl","start","statu","statuscod","statuscode:","step","storage,","string","stuff:","such","sudo","summari","summary():","summary,","summary:","summary:str","support","sure","system","system,","take","target","tell","test","test:","text","text/plain\"","that.","this:","throttl","throttle:","time","timeout","tip","to!","toward","traffic","tri","true","true,","true;","true或者false,表示是否需要anyproxy替换证书并解析http","trust","trust)","turn","type","type':","type:","typic","ubuntu","ui,","ui中的","unauthor","under","unknown","unlimit","unsecur","untrust","url","url:","us","use,","user","user.","users,","usual","valid","valu","var","via","view","visit","want","warning:","way","web","webinterfac","webinterface:","webport","webport:","websocket","websocket,","web版界面端口号,默认8002","web版界面配置","web版界面重构","whether","wifi","window","windows系统信任ca证书","without","work","world'","write","written","ws","wsintercept","wsintercept:","wss","x","yieldabl","yieldable,","you'r","{","{boolean}","{number}","{object}","{string}","|","}","})","});","},","};","下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可","不做任何处理。此时anyproxy会返回一个默认的错误页。","不做任何处理,返回null","不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为","不开启websocket代理","中把anyproxy证书的开关打开,否则safari将报错。","中文文档","为","为什么https请求不能进入处理函数?","主要特性包括:","举例","举例,请求","举例:请求","从sd卡安装证书。找到你下载的证书文件,进行安装","从存储设备安装。找到你下载的证书文件,进行安装","代理http","代理websocket","代理服务器发生错误","代理服务器启动完成","代理服务器都在wifi设置中配置","以chrome的switchyomega插件为例","以下任意一项都能用来改变https的处理特性:","以下几种返回都是合法的","作为npm模块使用","作为全局模块","你可以通过","使用npm包","使用举例","使用在线地址","使用本地数据","使用本地路径anyproxi","使用示例","修改发送到","修改请求bodi","修改请求协议","修改请求协议,如强制改用https发起请求","修改请求参数","修改请求头","修改请求数据","修改请求的目标地址","修改返回内容并延迟","修改返回头","修改返回状态码","修改返回的内容","修改返回的状态码","关闭代理服务器","关闭全局代理服务器","其他命令","创建代理服务器","加密与凭据","即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback","原始的服务端返回对象","双击打开rootca.crt","发送响应前处理","发送请求前拦截处理","只有返回true时,anyproxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。","可以是一个函数,也可以是一个普通的字符串","可修改内容包括请求头(request","可修改的内容包括http状态码(statu","同beforedealhttpsrequest中的参数","同beforesendrequest中的参数","启动","启动anyproxy,加载规则","启动代理服务器","启动后将终端http代理服务器配置为127.0.0.1:8001即可","命令行启动anyproxy时配置`","命令行启动anyproxy,默认端口号8001","命令行直接启动","在","在nodejs代码中启动","在wifi高级设置中,配置http代理即可","在构造anyproxy实例的时候,传入参数dangerouslyignoreunauthorized:true,","在请求处理过程中发生错误时,anyproxy会调用onerror方法,并提供对应的错误信息","基于node.js,开放二次开发能力,允许自定义请求处理逻辑","处理流程","处理流程图如下","多数场景下,错误会在请求目标服务器的时候发生,比如dns解析失败、请求超时等","如下几种方案都可以用来引用规则模块:","如下:","如何引用","如果beforesendrequest返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。","如果配置了全局解析https的参数,则anyproxy会略过这个调用","如要启用https解析,请在代理服务器启动前自行调用anyproxy.utils.certmgr相关方法生成证书,并引导用户信任安装。或引导用户使用anyproxi","安全","安全性与位置信息","安卓系统信任ca证书","安装","安装ca:","对于debian或者ubuntu系统,在安装anyproxy之前,可能还需要安装","开发示例","引入","当http请求经过代理服务器时,具体处理过程是:","当代理服务器收到https请求时,anyproxy可以替换证书,对请求做明文解析。","当启用https代理时,wss也会被代理,但是不会被anyproxy记录。需要开启","当访问特定的https站点,anyproxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问:","必选,代理服务器端口","快速开始","我们自然也可以借助自定义的rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。","所有http://httpbin.org","找到刚刚导入的anyproxy证书,配置为信任(alway","把","把响应信息返回给客户端","把所有发送到","把用http协议请求的","拦截https请求,对内容做修改","拦截发送到","拦截并修改服务端响应","拦截并修改正在发送的请求","提供gui界面,用以观察请求","提供了多种类型的证书文件,可在下载安装时选择。","提示","支持https的解析","收集请求所有请求参数,包括method,","改成https并发送","文件(已知如","时,requestdetail参数内容大致如下","时,responsedetail参数内容大致如下","是否启用web版界面,默认fals","是否处理https请求","是否屏蔽所有console输出,默认fals","是否开启websocket代理,默认fals","是否强制拦截所有的https,忽略规则模块的返回,默认fals","是否忽略请求中的证书错误,默认fals","是必选字段","服务端的返回信息,包括statuscod","本文档的适用范围是anyproxi","本质是中间人攻击(man","来加载模块并体验","来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。","查看请求信息","校验系统内是否存在anyproxy的根证书","样例","根据请求参数,向服务端发出请求,接收服务端响应。","模块介绍","此处无法控制向客户端的返回信息,无需返回值。","注意:http","注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题","测试规则","浏览器访问http://127.0.0.1:8002","点击web","然后,安装anyproxi","生成anyproxy的rootca,完成后请引导用户信任.crt文件","生成证书并解析所有https请求","用curl发请求测试的方法如下","用curl测试","用浏览器测试:配置浏览器http代理为","的https请求会被解析","的post数据","的user","的证书文件,少部分仅支持","的请求全部改到","的请求,使用本地数据代替服务端返回","的返回值里加上测试信息,并延迟5秒返回","的返回头里加上","的返回最后追加anyproxy的签名,并延迟5秒","的返回状态码都改成404","直接请求服务器:curl","直接返回客户端,不再发起请求,其中statuscod","相比3.x版本,anyproxi","确认将证书添加到login或system","示例","简介","简化了规则文件内的接口","管理anyproxy的证书","管理系统的全局代理配置,方法调用时可能会弹出密码框","类似这种报错都是因为系统没有信任anyproxy生成的ca所造成的","经过代理服务器后,期望的返回如下","自定义规则模块","被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。","规则文件中,除了summary,都是由","规则文件内提供`beforedealhttpsrequest`方法,返回","规则文件(rule)全面支持promise和gener","规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。","规则模块是用","规则模块的介绍文案,用于anyproxy提示用户,","规则模块的能力范围包括:","解析https请求的原理是中间人攻击(man","警告:ca证书和系统安全息息相关,建议亲自生成,并妥善保管","设置","设置属性","访问http://127.0.0.1:8002","证书下载到指定目录后,需要从其他入口进行安装,包括:","证书配置","请求bodi","请求url","请求使用的协议,http或者http","请求出错的事件","请求的原始request","请求目标的host,受制于协议,这里无法获取完整url","调用规则模块beforedealhttpsrequest方法,如果返回true,会明文解析这个请求,其他请求不处理","调用规则模块beforesendresponse方法,由模块对响应内容进行处理","运行","返回值","返回自定义错误页","这里提供一些样例,来讲解规则模块的常见用法","通过代理服务器请求:curl","通过启动参数","通过自定义的rule来修改","通过这种方式初始化的anyproxy,其配置也是全局性的,所有网站的证书问题都会被忽略","配置127.0.0.1:8001为全局http代理服务器","配置ios/android系统代理","配置osx系统代理","配置启动端口,如1080端口启动","配置浏览器http代理","附录:如何信任ca证书","限速值,单位kb/s,默认不限速","除了上述证书安装过程,还需要在","需要编写一个规则模块,在","首先和ios类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。","驱动的,函数需要满足yieldable。可以使用generator方法或是返回promise。","驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。",",web界面上能看到所有的请求信息",",界面上能看到刚才的请求信息",":"],"pipeline":["stopWordFilter","stemmer"]},"store":{"./":{"url":"./","title":"Introduction","keywords":"","body":"AnyProxy\nAnyProxy is a fully configurable http/https proxy in NodeJS.\nRef: 中文文档\nGithub:\n\nhttps://github.com/alibaba/anyproxy\n\nFeatures:\n\nOffer you the ablity to handle http traffic by invoking a js module\nIntercept https\nGUI webinterface\n\nChange Logs since 3.x:\n\nSupport Promise and Generator in rule module\nSimplified interface in rule module\nA newly designed web interface\n\n\nGetting Start\ninstall\nTo Debian and Ubuntu users, you may need to install nodejs-legacy at the same time\nsudo apt-get install nodejs-legacy\n\nThen install the AnyProxy\nnpm install -g anyproxy\n\nlaunch\n\nstart AnyProxy in command line, with default port 8001\n\nanyproxy\n\n\nnow you can use http proxy server by 127.0.0.1:8001\nvisit http://127.0.0.1:8002 to see the http requests\n\noptions\n\nspecify the port of http proxy\n\nanyproxy --port 1080\n\nUse AnyProxy as an npm module\nAnyProxy can be used as an npm module\n\nTo enable https feature, please guide users to use anyproxy-ca in cli. Or use methods under AnyProxy.utils.certMgr to generate certificates.\n\n\ninstall\n\nnpm i anyproxy --save\n\n\nsample\n\nconst AnyProxy = require('anyproxy');\nconst options = {\n port: 8001,\n rule: require('myRuleModule'),\n webInterface: {\n enable: true,\n webPort: 8002\n },\n throttle: 10000,\n forceProxyHttps: false,\n wsIntercept: false,\n silent: false\n};\nconst proxyServer = new AnyProxy.ProxyServer(options);\n\nproxyServer.on('ready', () => { /* */ });\nproxyServer.on('error', (e) => { /* */ });\nproxyServer.start();\n\n//when finished\nproxyServer.close();\n\n\nClass: AnyProxy.proxyServer\n\ncreate a proxy server\nconst proxy = new AnyProxy.proxyServer(options)\n\n\noptions\n\nport {number} required, port number of proxy server\nrule {object} your rule module\nthrottle {number} throttle in kb/s, unlimited for default\nforceProxyHttps {boolean} in force intercept all https request, default to false\nsilent {boolean} if keep silent in console, false for default false\ndangerouslyIgnoreUnauthorized {boolean} if ignore certificate error in request, default to false\nwsIntercept {boolean} whether to intercept websocket, default to false\nwebInterface {object} config for web interface\nenable {boolean} if enable web interface, default to false\nwebPort {number} port number for web interface\n\n\n\n\nEvent: ready\n\nemit when proxy server is ready\nsample\n\nproxy.on('ready', function() { })\n\n\nEvent: error\n\nemit when error happened inside proxy server\nsample\n\nproxy.on('error', function() { })\n\n\nMethod: start\n\nstart proxy server\nsample\n\nproxy.start();\n\n\nMethod: close\n\nclose proxy server\nsample\n\nproxy.close();\n\n\n\n\nAnyProxy.utils.systemProxyMgr\n\nmanage the system proxy config. sudo password may be required\nsample\n\n// set 127.0.0.1:8001 as system http server\nAnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001');\n\n// disable global proxy server\nAnyProxy.utils.systemProxyMgr.disableGlobalProxy();\n\n\nAnyProxy.utils.certMgr\n\nManage certificates of AnyProxy\nAnyProxy.utils.certMgr.ifRootCAFileExists()\ndetect if AnyProx rootCA exists\n\n\nAnyProxy.utils.certMgr.generateRootCA(callback)\ngenerate a rootCA\n\n\nSample\n\n const AnyProxy = require('anyproxy');\n const exec = require('child_process').exec;\n\n if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) {\n AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => {\n // let users to trust this CA before using proxy\n if (!error) {\n const certDir = require('path').dirname(keyPath);\n console.log('The cert is generated at', certDir);\n const isWin = /^win/.test(process.platform);\n if (isWin) {\n exec('start .', { cwd: certDir });\n } else {\n exec('open .', { cwd: certDir });\n }\n } else {\n console.error('error when generating rootCA', error);\n }\n });\n }\n\n\n\nProxy Https\n\nAnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate.\n\n\nUnder the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network.\n\n\ngenerate certifycates and intercept\n\nanyproxy-ca #generate root CA. manually trust it after that.\nanyproxy --intercept #launch anyproxy and intercept all https traffic\n\n\nAppendix:how to trust CA\n\nProxy WebSocket\nanyproxy --ws-intercept\n\n\nThe wss requests will be handled automatically when the HTTPS intercept is turned on, but AnyProxy will not record the data by default. You need to specify the --ws-intercept to tell AnyProxy to record it.\n\nRule Introduction\nAnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests.\n\nMake sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk.\n\nRule module could do the following stuff:\n\nintercept and modify the request which is being sent\neditable fields include request header, body, target address\n\n\nintercept and modify the response from server\neditable fields include response status code, header, body\n\n\nintercept https requests, modify request and response\n\nsample\n\nTarget\n\nwrite a rule module to append some text to the response of GET http://httpbin.org/user-agent, and delay the response for 5 seconds\n\n\nStep 1,Write the rule file, save as sample.js\n// file: sample.js\nmodule.exports = {\n summary: 'a rule to hack response',\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '- AnyProxy Hacked!';\n\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\n\nStep 2, start AnyProxy and load the rule file\n\nrun anyproxy --rule sample.js\n\n\nStep 3, test\n\nuse curl\ncurl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n\n\nuse browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent\n\nthe expected response from proxy is\n\n\n{\n \"user-agent\": \"curl/7.43.0\"\n}\n- AnyProxy Hacked!\n\nStep 4, view the request log\n\nvisit http://127.0.0.1:8002, the request just sent should be listed here\n\n\n\nhow does it work\n\nThe flow chart is as follows\n\n\n\nWhen got an http request, the entire process of proxy server is\n\nAnyProxy collects all the quest info, include method, header, body\nAnyProxy calls beforeSendRequest of the rule module. Rule module deal the request, return new request param or response content\nIf beforeSendRequest returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here.\nSend request to target server, collect response\nCall beforeSendResponse of the rule module. Rule module deal the response data\nSend response to client\n\n\nWhen AnyProxy get https request, it could replace the certificate and decrypt the request data\n\nAnyProxy calls beforeDealHttpsRequest of the rule module\nIf the function returns true, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed.\n\n\n\nhow to load rule module\n\nuse local file\nanyproxy --rule ./rule.js\n\n\nuse an online rule file\nanyproxy --rule https://sample.com/rule.js\n\n\nuse an npm module\n\nAnyProxy uses require() to load rule module. You could either load a local npm module or a global-installed one.\n\nanyproxy --rule ./myRulePkg/ #local module\nnpm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module\n\n\n\nRule module interface\nA typical rule module is as follows. All the functions are optional, just write the part you are interested in.\nmodule.exports = {\n // introduction\n summary: 'my customized rule for AnyProxy',\n // intercept before send request to server\n *beforeSendRequest(requestDetail) { /* ... */ },\n // deal response before send to client\n *beforeSendResponse(requestDetail, responseDetail) { /* ... */ },\n // if deal https request\n *beforeDealHttpsRequest(requestDetail) { /* ... */ },\n // error happened when dealing requests\n *onError(requestDetail, error) { /* ... */ },\n // error happened when connect to https server\n *onConnectError(requestDetail, error) { /* ... */ }\n};\n\n\nAll functions in your rule file, except summary, are all driven by co . They should be yieldable, i.e. return a promise or be a generator function.\n\nsummary\nsummary\n\nIntroduction of this rule file. AnyProxy will read this field and give some tip to user.\n\nbeforeSendRequest\nbeforeSendRequest(requestDetail)\n\nBefore sending request to server, AnyProxy will call beforeSendRequest with param requestDetail\nrequestDetail\nprotocol {string} the protocol to use, http or https\nrequestOptions {object} the options of the request-to-go, a param of require('http').request . ref: https://nodejs.org/api/http.html#http_http_request_options_callback\nrequestData {object} request body\nurl {string} request url\n_req {object} the native node.js request object\n\n\ne.g. When requesting anyproxy.io, requestDetail is something like the following\n{\n protocol: 'http',\n url: 'http://anyproxy.io/',\n requestOptions: {\n hostname: 'anyproxy.io',\n port: 80,\n path: '/',\n method: 'GET',\n headers: {\n Host: 'anyproxy.io',\n 'Proxy-Connection': 'keep-alive',\n 'User-Agent': '...'\n }\n },\n requestData: '...',\n _req: { /* ... */}\n}\n\n\nAny of these return values are valid\n\ndo nothing, and return null\n\nreturn null;\n\n\nmodify the request protocol,i.e. force use https\n\nreturn {\n protocol: 'https'\n};\n\n\nmodify request param\n\nvar newOption = Object.assign({}, requestDetail.requestOptions);\nnewOption.path = '/redirect/to/another/path';\nreturn {\n requestOptions: newOption\n};\n\n\nmodify request body\n\nreturn {\n requestData: 'my new request data'\n // requestOptions can also be used here\n};\n\n\ngive response to the client, not sending request any longer. statusCode headersare required is this situation.\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nbeforeSendResponse\nbeforeSendResponse(requestDetail, responseDetail)\n\nBefore sending response to client, AnyProxy will call beforeSendResponse with param requestDetail responseDetail\nrequestDetail is the same param as in beforeSendRequest\nresponseDetail\nresponse {object} the response from server, includes statusCode header body\n_res {object} the native node.js response object\n\n\ne.g. When requesting anyproxy.io, responseDetail is something like the following\n{\n response: {\n statusCode: 200,\n header: {\n 'Content-Type': 'image/gif',\n Connection: 'close',\n 'Cache-Control': '...'\n },\n body: '...'\n },\n _res: { /* ... */ }\n}\n\n\nAny of these return values are valid\n\ndo nothing, and return null\n\nreturn null;\n\n\nmodify the response status code\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.statusCode = 404;\nreturn {\n response: newResponse\n};\n\n\nmodify the response content\n\nvar newResponse = Object.assign({}, responseDetail.response);\nnewResponse.body += '--from anyproxy--';\nreturn {\n response: newResponse\n};\n\n\n\nbeforeDealHttpsRequest\nbeforeDealHttpsRequest(requestDetail)\n\nWhen receiving https request, AnyProxy will call beforeDealHttpsRequest with param requestDetail\nIf configed with forceProxyHttps in launching, AnyProxy will skip calling this method\nOnly by returning true, AnyProxy will try to replace the certificate and intercept the https request.\nrequestDetail\nhost {string} the target host to request. Due to the request protocol, full url couldn't be got here\n_req {object} the native node.js request object. The _req here refers to the CONNECT request.\n\n\nreturn value\ntrue or false, whether AnyProxy should intercept the https request\n\n\n\nonError\nonError(requestDetail, error)\n\nAnyProxy will call this method when an error happened in request handling.\nErrors usually are issued during requesting, e.g. DNS failure, request timeout\nrequestDetail is the same one as in beforeSendRequest\nAny of these return values are valid\n\ndo nothing, and AnyProxy will response a default error page\n\nreturn null;\n\n\nreturn a customized error page\n\nreturn {\n response: {\n statusCode: 200,\n header: { 'content-type': 'text/html' },\n body: 'this could be a or '\n }\n};\n\n\n\nonConnectError\nonConnectError(requestDetail, error)\n\nAnyProxy will call this method when failed to connect target server in https request\nrequestDetail is the same one as in beforeDealHttpsRequest\nno return value is required\n\nRule Samples\n\nhere are some samples about frequently used rule file\ntry these samples by anyproxy --rule http://....js\nhow to test with curl:\nrequest the server directly curl http://httpbin.org/\nrequest the server via proxy curl http://httpbin.org/ --proxy http://127.0.0.1:8001\n\n\n\nuse local response\n\nintercept the request towards http://httpbin.org , return the local-defined response\n\nanyproxy --rule rule_sample/sample_use_local_response.js\n\n/* \n sample: \n intercept all requests toward httpbin.org, use a local response\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n const localResponse = {\n statusCode: 200,\n header: { 'Content-Type': 'application/json' },\n body: '{\"hello\": \"this is local response\"}'\n };\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n return {\n response: localResponse\n };\n }\n },\n};\n\nmodify request header\n\nmodify the user-agent sent to httpbin.org\n\nanyproxy --rule rule_sample/sample_modify_request_header.js\n\n/* \n sample: \n modify the user-agent in requests toward httpbin.org\n test:\n curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0';\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\nmodify request body\n\nmodify the post body of http://httpbin.org/post\n\nanyproxy --rule rule_sample/sample_modify_request_data.js\n\n/*\n sample:\n modify the post data towards http://httpbin.org/post\n test:\n curl -H \"Content-Type: text/plain\" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001\n expected response:\n { \"data\": \"i-am-anyproxy-modified-post-data\" }\n*/\nmodule.exports = {\n summary: 'Rule to modify request data',\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) {\n return {\n requestData: 'i-am-anyproxy-modified-post-data'\n };\n }\n },\n};\n\nmodify the request target\n\nsend all the request towards http://httpbin.org/ to http://httpbin.org/user-agent\n\nanyproxy --rule rule_sample/sample_modify_request_path.js\n\n/*\n sample:\n redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html\n test:\n curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001\n expected response:\n 'hello world' from 127.0.0.1:8001/index.html\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n requestDetail.protocol = 'http';\n newRequestOptions.hostname = '127.0.0.1'\n newRequestOptions.port = '8008';\n newRequestOptions.path = '/index.html';\n newRequestOptions.method = 'GET';\n return requestDetail;\n }\n },\n *beforeDealHttpsRequest(requestDetail) {\n return true;\n }\n};\n\nmodify request protocol\n\nmodify the http request towards http://httpbin.org to https\n\nanyproxy --rule rule_sample/sample_modify_request_protocol.js\n\n/* \n sample: \n redirect all http requests of httpbin.org to https\n test:\n curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001\n expected response:\n { \"X-Forwarded-Protocol\": \"https\" }\n*/\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newOption = requestDetail.requestOptions;\n newOption.port = 443;\n return {\n protocol: 'https',\n requestOptions: newOption\n };\n }\n }\n};\n\nmodify response status code\n\nmodify all status code from http://httpbin.org to 404\n\nanyproxy --rule rule_sample/sample_modify_response_statuscode.js\n\n/* \n sample: \n modify all status code of http://httpbin.org/ to 404\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n HTTP/1.1 404 Not Found\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org') === 0) {\n const newResponse = responseDetail.response;\n newResponse.statusCode = 404;\n return {\n response: newResponse\n };\n }\n }\n};\n\nmodify the response header\n\nadd X-Proxy-By:AnyProxy to the response header from http://httpbin.org/user-agent\n\nanyproxy --rule rule_sample/sample_modify_response_header.js\n\n/* \n sample: \n modify response header of http://httpbin.org/user-agent\n test:\n curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n X-Proxy-By: AnyProxy\n*/\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) {\n const newResponse = responseDetail.response;\n newResponse.header['X-Proxy-By'] = 'AnyProxy';\n return {\n response: newResponse\n };\n }\n }\n};\n\nmodify response data and delay\n\nappend some info to the response of http://httpbin.org/user-agent, then delay the response for 5 seconds.\n\nanyproxy --rule rule_sample/sample_modify_response_data.js\n\n/* \n sample: \n modify response data of http://httpbin.org/user-agent\n test:\n curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001\n expected response:\n { \"user-agent\": \"curl/7.43.0\" } -- AnyProxy Hacked! --\n*/\n\nmodule.exports = {\n *beforeSendResponse(requestDetail, responseDetail) {\n if (requestDetail.url === 'http://httpbin.org/user-agent') {\n const newResponse = responseDetail.response;\n newResponse.body += '-- AnyProxy Hacked! --';\n return new Promise((resolve, reject) => {\n setTimeout(() => { // delay the response for 5s\n resolve({ response: newResponse });\n }, 5000);\n });\n }\n },\n};\n\nConfig Certification\nConfig root CA in OSX\n\nthis kind of errors is usually caused by untrusted root CA\n\n\n\nWarning: please keep your root CA safe since it may influence your system security.\n\ninstall :\n\ndouble click rootCA.crt\n\nadd cert into login or system\n\n\n\n\nfind the newly imported AnyProxy certificates, configured as Always Trust\n\n\nConfig root CA in windows\n\nConfig OSX system proxy\n\nthe config is in wifi - advanced\n\n\nconfig http proxy server\n\ntake Chrome extent [SwitchyOmega] as an example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例\n\n\ntrust root CA in iOS\n\nClick Root CA in web ui, and follow the instruction to install\n\n\ntrust root CA in iOS after 10.3\n\nBesides installing root CA, you have to \"turn on\" the certificate for web manually in settings - general - about - Certificate Trust Settings. Otherwire, safari will not trust the root CA generated by AnyProxy.\n\n\ntrust root CA in Android\nFirst of all, you need to download the root CA by clicking Root CA in web ui, and then scan the QR code.\nInstalling CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path.\n\nThe downloaded CA file can be directly installed by clicking, this is the easist way\nYou need to install the CA file from other menu, such as:\nSettings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install\nSettings -> Security -> Install from SD card, and find you CA file to install\n\n\n\nThere are several file extensions of CA file which may not be compatible with all kinds of Android phones. .crt file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing.\nconfig iOS/Android proxy server\n\nproxy settings are placed in wifi setting\n\niOS\n\n\n\n\nAndroid\n\n\nFAQ\nQ: can not deal https request in rule module.\n\nA: Any of these options could be used to change the way AnyProxy deall https requests\nconfig --intercept when luanching AnyProxy via cli, or use forceProxyHttps when using as an npm module\nplace a beforeDealHttpsRequest function in your rule file and determine which request to intercept by your own.\n\n\n\nQ: get an error says function is not yieldable\n\nA: Rule module is driven by co. The functions inside should be yieldable, i.e. return a promise or be a generator function.\n\nQ: The connection is not private\nAnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below.\n\nIf you run AnyProxy by command line\nPass in the option --ignore-unauthorized-ssl to ignore the certification errors, please mind that the option will be active for all connections.\nanyproxy -i --ignore-unauthorized-ssl\n\n\nIf you run AnyProxy by Nodejs\nPass in the option dangerouslyIgnoreUnauthorized:true, like this:\nconst options = {\n ...,\n dangerouslyIgnoreUnauthorized: true\n};\n\nconst anyproxyIns = new AnyProxy.ProxyCore(options);\nanyproxyIns.start();\n\nThis is also a global option, all certification errors will be ignored\n\nWith the help of AnyProxy Rule\nYou can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs Http.rquest, as we do in AnyProxy. A simple demo below:\nmodule.exports = {\n *beforeSendRequest(requestDetail) {\n if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) {\n const newRequestOptions = requestDetail.requestOptions;\n // set rejectUnauthorized as false\n newRequestOptions.rejectUnauthorized = false;\n return {\n requestOptions: newRequestOptions\n };\n }\n },\n};\n\nAnd we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to!\n\n\n"}}} ================================================ FILE: docs-src/CNAME ================================================ anyproxy.io ================================================ FILE: docs-src/LANGS.md ================================================ # Languages * [中文](cn/) * [English](en/) ================================================ FILE: docs-src/README.md ================================================ # THIS IS AUTO GENERATED FILE, DO NOT EDIT THE HTML DIRECTLY. # YOU CAN EDIT THE SOURCE IN docs-md FOLDER ================================================ FILE: docs-src/_layouts/layout.html ================================================ {% block title %}{{ config.title|d("GitBook", true) }}{% endblock %} {% if config.author %}{% endif %} {% if config.isbn %}{% endif %} {% block style %} {% for resource in plugins.resources.css %} {% if resource.url %} {% else %} {% endif %} {% endfor %} {% endblock %} {% block head %}{% endblock %} {% block body %}{% endblock %} {% block javascript %}{% endblock %} ================================================ FILE: docs-src/assets/auto-lang.js ================================================ /* eslint no-var: off */ /** * detect if the browser is in UTF-8 zone * @return boolean */ function isUTF8Zone() { return new Date().getTimezoneOffset() === -480; } /** * detect if the browser is already in a locale view */ function isInLocaleView() { return /(cn|en)/i.test(location.href); } function initDefaultLocaleAndStatic() { if (!isInLocaleView()) { location.href = isUTF8Zone() ? '/cn' : 'en'; } } initDefaultLocaleAndStatic(); ================================================ FILE: docs-src/assets/main.js ================================================ /* eslint no-var: off */ function injectBaiduStatic() { var _hmt = _hmt || []; var hm = document.createElement('script'); var s = document.getElementsByTagName('script')[0]; hm.src = '//hm.baidu.com/hm.js?4e51565b7d471fd6623c163a8fd79e07'; s.parentNode.insertBefore(hm, s); } injectBaiduStatic(); ================================================ FILE: docs-src/assets/website.css ================================================ .book .book-summary ul.summary li.active>a, .book .book-summary ul.summary li a:hover { color: #008cff; background: transparent; text-decoration: none !important; } h1 { color: #2674BA; } h2 { color: #0099CC; } h3 { color: #108ee9; } h4 { color: #662D91; } h5 { color: #444444; } .gitbook-link { display: none !important; } .summary-title-span { position: relative !important; padding: 0 !important; } .rule-title:after, .sample-title:after { font-size: 12px; padding: 0 3px; border-radius: 3px; color: #fff; } .rule-title:after{ content: 'rule'; background-color: #108ee9; } .sample-title:after { content: 'sample'; background-color: #00a854; } .page-inner { max-width: 1000px !important; } ================================================ FILE: docs-src/book.json ================================================ { "title": "AnyProxy", "author": "AnyProxy", "description": "A fully configurable http/https proxy in NodeJS", "plugins": [ ], "pluginsConfig": { } } ================================================ FILE: docs-src/cn/README.md ================================================ AnyProxy =================== > 本文档的适用范围是AnyProxy 4.0,欢迎提供反馈 Ref: [English Doc](../en) AnyProxy是一个开放式的HTTP代理服务器。 Github主页:https://github.com/alibaba/anyproxy 主要特性包括: * 基于Node.js,开放二次开发能力,允许自定义请求处理逻辑 * 支持Https的解析 * 提供GUI界面,用以观察请求 相比3.x版本,AnyProxy 4.0的主要变化: * 规则文件(Rule)全面支持Promise和Generator * 简化了规则文件内的接口 * Web版界面重构 # 快速开始 ## 作为全局模块 ### 安装 对于Debian或者Ubuntu系统,在安装AnyProxy之前,可能还需要安装 `nodejs-legacy` ```bash sudo apt-get install nodejs-legacy ``` 然后,安装AnyProxy ```bash npm install -g anyproxy ``` ### 启动 * 命令行启动AnyProxy,默认端口号8001 ```bash anyproxy ``` * 启动后将终端http代理服务器配置为127.0.0.1:8001即可 * 访问http://127.0.0.1:8002 ,web界面上能看到所有的请求信息 ### 其他命令 * 配置启动端口,如1080端口启动 ```bash anyproxy --port 1080 ``` ## 作为npm模块使用 AnyProxy可以作为一个npm模块使用,整合进其他工具。 > 如要启用https解析,请在代理服务器启动前自行调用`AnyProxy.utils.certMgr`相关方法生成证书,并引导用户信任安装。或引导用户使用`anyproxy-ca`方法。 * 引入 ```bash npm i anyproxy --save ``` * 使用举例 ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, // 不开启websocket代理 silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * 创建代理服务器 ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} 必选,代理服务器端口 * `rule` {object} 自定义规则模块 * `throttle` {number} 限速值,单位kb/s,默认不限速 * `forceProxyHttps` {boolean} 是否强制拦截所有的https,忽略规则模块的返回,默认`false` * `silent` {boolean} 是否屏蔽所有console输出,默认`false` * `dangerouslyIgnoreUnauthorized` {boolean} 是否忽略请求中的证书错误,默认`false` * `wsIntercept` {boolean} 是否开启websocket代理,默认`false` * `webInterface` {object} web版界面配置 * `enable` {boolean} 是否启用web版界面,默认`false` * `webPort` {number} web版界面端口号,默认`8002` * Event: `ready` * 代理服务器启动完成 * 示例 ```js proxy.on('ready', function() { }) ``` * Event: `error` * 代理服务器发生错误 * 示例 ```js proxy.on('error', function() { }) ``` * Method: `start` * 启动代理服务器 * 示例 ```js proxy.start(); ``` * Method: `close` * 关闭代理服务器 * 示例 ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * 管理系统的全局代理配置,方法调用时可能会弹出密码框 * 使用示例 ```js // 配置127.0.0.1:8001为全局http代理服务器 AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // 关闭全局代理服务器 AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * 管理AnyProxy的证书 * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * 校验系统内是否存在AnyProxy的根证书 * `AnyProxy.utils.certMgr.generateRootCA(callback)` * 生成AnyProxy的rootCA,完成后请引导用户信任.crt文件 * 样例 ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # 代理HTTPS * AnyProxy默认不对https请求做处理,如需看到明文信息,需要配置CA证书 > 解析https请求的原理是中间人攻击(man-in-the-middle),用户必须信任AnyProxy生成的CA证书,才能进行后续流程 * 生成证书并解析所有https请求 ```bash anyproxy-ca #生成rootCA证书,生成后需要手动信任 anyproxy --intercept #启动AnyProxy,并解析所有https请求 ``` * [附录:如何信任CA证书](#证书配置) # 代理WebSocket ```bash anyproxy --ws-intercept ``` > 当启用`HTTPS`代理时,`wss`也会被代理,但是不会被AnyProxy记录。需要开启`--ws-intercept`后才会从界面上看到相应内容。 # rule模块 AnyProxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。 >注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题 规则模块的能力范围包括: * 拦截并修改正在发送的请求 * 可修改内容包括请求头(request header),请求体(request body),甚至是请求的目标地址等 * 拦截并修改服务端响应 * 可修改的内容包括http状态码(status code)、响应头(response header)、响应内容等 * 拦截https请求,对内容做修改 * 本质是中间人攻击(man-in-the-middle attack),需要客户端提前信任AnyProxy生成的CA ### 开发示例 * 举例 * 需要编写一个规则模块,在 GET http://httpbin.org/user-agent 的返回值里加上测试信息,并延迟5秒返回 * Step 1,编写规则 ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, 启动AnyProxy,加载规则 * 运行 `anyproxy --rule sample.js` * Step 3, 测试规则 * 用curl测试 ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * 用浏览器测试:配置浏览器http代理为 127.0.0.1:8001,访问 http://httpbin.org/user-agent * 经过代理服务器后,期望的返回如下 ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, 查看请求信息 * 浏览器访问http://127.0.0.1:8002 ,界面上能看到刚才的请求信息 ### 处理流程 * 处理流程图如下 * 当http请求经过代理服务器时,具体处理过程是: * 收集请求所有请求参数,包括method, header, body等 * AnyProxy调用规则模块`beforeSendRequest`方法,由模块做处理,返回新的请求参数,或返回响应内容 * 如果`beforeSendRequest`返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。 * 根据请求参数,向服务端发出请求,接收服务端响应。 * 调用规则模块`beforeSendResponse`方法,由模块对响应内容进行处理 * 把响应信息返回给客户端 * 当代理服务器收到https请求时,AnyProxy可以替换证书,对请求做明文解析。 * 调用规则模块`beforeDealHttpsRequest`方法,如果返回`true`,会明文解析这个请求,其他请求不处理 * 被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。 ### 如何引用 如下几种方案都可以用来引用规则模块: * 使用本地路径 ```bash anyproxy --rule ./rule.js ``` * 使用在线地址 ```bash anyproxy --rule https://sample.com/rule.js ``` * 使用npm包 * AnyProxy使用`require()`加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包 ```bash anyproxy --rule ./myRulePkg/ #本地包 npm i -g myRulePkg && anyproxy --rule myRulePkg #全局包 ``` # rule接口文档 规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。 ```js module.exports = { // 模块介绍 summary: 'my customized rule for AnyProxy', // 发送请求前拦截处理 *beforeSendRequest(requestDetail) { /* ... */ }, // 发送响应前处理 *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // 是否处理https请求 *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // 请求出错的事件 *onError(requestDetail, error) { /* ... */ }, // https连接服务器出错 *onConnectError(requestDetail, error) { /* ... */ } }; ``` > 规则文件中,除了summary,都是由 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。 ### summary #### summary(): string | summary:string * 规则模块的介绍文案,用于AnyProxy提示用户, 可以是一个函数,也可以是一个普通的字符串 ### beforeSendRequest #### beforeSendRequest(requestDetail) * AnyProxy向服务端发送请求前,会调用`beforeSendRequest`,并带上参数`requestDetail` * `requestDetail` * `protocol` {string} 请求使用的协议,http或者https * `requestOptions` {object} 即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} 请求Body * `url` {string} 请求url * `_req` {object} 请求的原始request * 举例:请求 *anyproxy.io* 时,`requestDetail`参数内容大致如下 ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改请求协议,如强制改用https发起请求 ```js return { protocol: 'https' }; ``` * 修改请求参数 ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * 修改请求body ```js return { requestData: 'my new request data' //这里也可以同时加上requestOptions }; ``` * 直接返回客户端,不再发起请求,其中`statusCode` `header` 是必选字段 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * AnyProxy向客户端发送请求前,会调用`beforeSendResponse`,并带上参数`requestDetail` `responseDetail` * `requestDetail` 同`beforeSendRequest`中的参数 * `responseDetail` * `response` {object} 服务端的返回信息,包括`statusCode` `header` `body`三个字段 * `_res` {object} 原始的服务端返回对象 * 举例,请求 *anyproxy.io* 时,`responseDetail`参数内容大致如下 ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改返回的状态码 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * 修改返回的内容 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * AnyProxy收到https请求时,会调用`beforeDealHttpsRequest`,并带上参数`requestDetail` * 如果配置了全局解析https的参数,则AnyProxy会略过这个调用 * 只有返回`true`时,AnyProxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。 * 注意:https over http的代理模式中,这里的request是CONNECT请求 * `requestDetail` * `host` {string} 请求目标的Host,受制于协议,这里无法获取完整url * `_req` {object} 请求的原始request * 返回值 * `true`或者`false`,表示是否需要AnyProxy替换证书并解析https ### onError #### onError(requestDetail, error) * 在请求处理过程中发生错误时,AnyProxy会调用`onError`方法,并提供对应的错误信息 * 多数场景下,错误会在请求目标服务器的时候发生,比如DNS解析失败、请求超时等 * `requestDetail` 同`beforeSendRequest`中的参数 * 以下几种返回都是合法的 * 不做任何处理。此时AnyProxy会返回一个默认的错误页。 ```js return null; ``` * 返回自定义错误页 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy在与目标HTTPS服务器建立连接的过程中,如果发生错误,AnyProxy会调用这个方法 * `requestDetail` 同`beforeDealHttpsRequest`中的参数 * 此处无法控制向客户端的返回信息,无需返回值。 # rule样例 * 这里提供一些样例,来讲解规则模块的常见用法 * 你可以通过 `anyproxy --rule http://....js` 来加载模块并体验 * 用curl发请求测试的方法如下 * 直接请求服务器:`curl http://httpbin.org/` * 通过代理服务器请求:`curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### 使用本地数据 * 拦截发送到 http://httpbin.org 的请求,使用本地数据代替服务端返回 ```bash anyproxy --rule rule_sample/sample_use_local_response.js ``` ```js /* sample: intercept all requests toward httpbin.org, use a local response test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { const localResponse = { statusCode: 200, header: { 'Content-Type': 'application/json' }, body: '{"hello": "this is local response"}' }; if (requestDetail.url.indexOf('http://httpbin.org') === 0) { return { response: localResponse }; } }, }; ``` ### 修改请求头 * 修改发送到 httpbin.org 的user-agent ```bash anyproxy --rule rule_sample/sample_modify_request_header.js ``` ```js /* sample: modify the user-agent in requests toward httpbin.org test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newRequestOptions = requestDetail.requestOptions; newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0'; return { requestOptions: newRequestOptions }; } }, }; ``` ### 修改请求数据 * 修改发送到 http://httpbin.org/post 的post数据 ```bash anyproxy --rule rule_sample/sample_modify_request_data.js ``` ```js /* sample: modify the post data towards http://httpbin.org/post test: curl -H "Content-Type: text/plain" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001 expected response: { "data": "i-am-anyproxy-modified-post-data" } */ module.exports = { summary: 'Rule to modify request data', *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) { return { requestData: 'i-am-anyproxy-modified-post-data' }; } }, }; ``` ### 修改请求的目标地址 * 把所有发送到 http://httpbin.org/ 的请求全部改到 http://httpbin.org/user-agent ```bash anyproxy --rule rule_sample/sample_modify_request_path.js ``` ```js /* sample: redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html test: curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001 expected response: 'hello world' from 127.0.0.1:8001/index.html */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) { const newRequestOptions = requestDetail.requestOptions; requestDetail.protocol = 'http'; newRequestOptions.hostname = '127.0.0.1' newRequestOptions.port = '8008'; newRequestOptions.path = '/index.html'; newRequestOptions.method = 'GET'; return requestDetail; } }, *beforeDealHttpsRequest(requestDetail) { return true; } }; ``` ### 修改请求协议 * 把用http协议请求的 http://httpbin.org 改成https并发送 ```bash anyproxy --rule rule_sample/sample_modify_request_protocol.js ``` ```js /* sample: redirect all http requests of httpbin.org to https test: curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001 expected response: { "X-Forwarded-Protocol": "https" } */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newOption = requestDetail.requestOptions; newOption.port = 443; return { protocol: 'https', requestOptions: newOption }; } } }; ``` ### 修改返回状态码 * 把 所有http://httpbin.org 的返回状态码都改成404 ```bash anyproxy --rule rule_sample/sample_modify_response_statuscode.js ``` ```js /* sample: modify all status code of http://httpbin.org/ to 404 test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: HTTP/1.1 404 Not Found */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newResponse = responseDetail.response; newResponse.statusCode = 404; return { response: newResponse }; } } }; ``` ### 修改返回头 * 在 http://httpbin.org/user-agent 的返回头里加上 X-Proxy-By:AnyProxy ```bash anyproxy --rule rule_sample/sample_modify_response_header.js ``` ```js /* sample: modify response header of http://httpbin.org/user-agent test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: X-Proxy-By: AnyProxy */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) { const newResponse = responseDetail.response; newResponse.header['X-Proxy-By'] = 'AnyProxy'; return { response: newResponse }; } } }; ``` ### 修改返回内容并延迟 * 在 http://httpbin.org/user-agent 的返回最后追加AnyProxy的签名,并延迟5秒 ```bash anyproxy --rule rule_sample/sample_modify_response_data.js ``` ```js /* sample: modify response data of http://httpbin.org/user-agent test: curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: { "user-agent": "curl/7.43.0" } -- AnyProxy Hacked! -- */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '-- AnyProxy Hacked! --'; return new Promise((resolve, reject) => { setTimeout(() => { // delay the response for 5s resolve({ response: newResponse }); }, 5000); }); } }, }; ``` # 证书配置 ### OSX系统信任CA证书 * 类似这种报错都是因为系统没有信任AnyProxy生成的CA所造成的 > 警告:CA证书和系统安全息息相关,建议亲自生成,并妥善保管 安装CA: * 双击打开*rootCA.crt* * 确认将证书添加到login或system * 找到刚刚导入的AnyProxy证书,配置为信任(Always Trust) ### Windows系统信任CA证书 ### 配置OSX系统代理 * 在wifi高级设置中,配置http代理即可 ### 配置浏览器HTTP代理 * 以Chrome的[SwitchyOmega插件](https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### iOS系统信任CA证书 * 点击web ui中的 *Root CA*,按提示扫描二维码即可安装 ### iOS >= 10.3信任CA证书 * 除了上述证书安装过程,还需要在 *设置->通用->关于本机->证书信任设置* 中把AnyProxy证书的开关打开,否则safari将报错。 ### 安卓系统信任CA证书 首先和iOS类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。 * 下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可 * 证书下载到指定目录后,需要从其他入口进行安装,包括: * 设置 -> 安全性与位置信息 -> 加密与凭据 -> 从存储设备安装。找到你下载的证书文件,进行安装 * 设置 -> 安全 -> 从SD卡安装证书。找到你下载的证书文件,进行安装 不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为 .crt 的证书文件,少部分仅支持 .cer 文件(已知如 OPPO R15),AnyProxy 提供了多种类型的证书文件,可在下载安装时选择。 ### 配置iOS/Android系统代理 * 代理服务器都在wifi设置中配置 * iOS HTTP代理配置 * Android HTTP代理配置 # FAQ #### Q: 为什么https请求不能进入处理函数? A: 以下任意一项都能用来改变https的处理特性: 1. 命令行启动AnyProxy时配置`--intercept`参数,按npm模块启动时配置`forceProxyHttps`参数,所有Https请求都会被替换证书并解析 2. 规则文件内提供`beforeDealHttpsRequest`方法,返回 *true* 的https请求会被解析 #### Q: 提示 *function is not yieldable* * A: 规则模块是用 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以使用generator方法或是返回Promise。 #### Q: The connection is not private 当访问特定的HTTPS站点,AnyProxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问: - 命令行直接启动 通过启动参数 `--ignore-unauthorized-ssl` 来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。 ```bash anyproxy -i --ignore-unauthorized-ssl ``` - 在Nodejs代码中启动 在构造AnyProxy实例的时候,传入参数`dangerouslyIgnoreUnauthorized:true`, 如下: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *通过这种方式初始化的AnyProxy,其配置也是全局性的,所有网站的证书问题都会被忽略* - 通过自定义的Rule来修改 我们自然也可以借助自定义的Rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。 ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // 设置属性 rejectUnauthorized 为 false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` ================================================ FILE: docs-src/cn/SUMMARY.md ================================================ # Summary * [简介](README.md) * [快速开始](README.md#快速开始) * [安装](README.md#安装) * [启动](README.md#启动) * [其他命令](README.md#其他命令) * [作为npm模块启动](README.md#作为npm模块使用) * [代理HTTPS](README.md#代理https) * [代理WebSocket](README.md#代理websocket) * [rule模块](README.md#rule模块) * [开发示例](README.md#开发示例) * [处理流程](README.md#处理流程) * [如何引用](README.md#如何引用) * [rule接口文档](README.md#rule接口文档) * [summary_class=rule-title](README.md#summary) * [beforeSendRequest_class=rule-title](README.md#beforesendrequest) * [beforeSendResponse_class=rule-title](README.md#beforesendresponse) * [beforeDealHttpsRequest_class=rule-title](README.md#beforedealhttpsrequest) * [onError_class=rule-title](README.md#onerror) * [onConnectError_class=rule-title](README.md#onconnecterror) * [rule样例](README.md#rule样例) * [使用本地数据_class=sample-title](README.md#使用本地数据) * [修改请求头_class=sample-title](README.md#修改请求头) * [修改请求数据_class=sample-title](README.md#修改请求数据) * [修改请求的目标地址_class=sample-title](README.md#修改请求的目标地址) * [修改请求协议_class=sample-title](README.md#修改请求协议) * [修改返回状态码_class=sample-title](README.md#修改返回状态码) * [修改返回头_class=sample-title](README.md#修改返回头) * [修改返回内容并延迟_class=sample-title](README.md#修改返回内容并延迟) * [证书配置](README.md#证书配置) * [OSX系统信任CA证书](README.md#osx系统信任ca证书) * [Windows系统信任CA证书](README.md#windows系统信任ca证书) * [配置OSX系统代理](README.md#配置osx系统代理) * [配置浏览器HTTP代理](README.md#配置浏览器http代理) * [iOS系统信任CA证书](README.md#ios系统信任ca证书) * [iOS >= 10.3信任CA证书](README.md#ios--103信任ca证书) * [安卓系统信任CA证书](README.md#安卓系统信任ca证书) * [配置iOS/Android系统代理](README.md#配置iosandroid系统代理) * [FAQ](README.md#faq) ================================================ FILE: docs-src/cn/_layouts/layout.html ================================================ {% block title %}{{ config.title|d("GitBook", true) }}{% endblock %} {% if config.author %}{% endif %} {% if config.isbn %}{% endif %} {% block style %} {% for resource in plugins.resources.css %} {% if resource.url %} {% else %} {% endif %} {% endfor %} {% endblock %}
      {% set regExp = r/^foo.*/g %}
      {% block body %}{% endblock %} {% block javascript %}{% endblock %} ================================================ FILE: docs-src/cn/_layouts/website/summary.html ================================================ {% macro articles(_articles) %} {% for article in _articles %}
    • {% if article.path and getPageByPath(article.path) %} {% elif article.url %} {% else %} {% endif %} {% if article.level != "0" and config.pluginsConfig['theme-default'].showLevel %} {{ article.level }}. {% endif %}
      {{ article.title | replace(r/_class=.+/, '') }}
      {% if article.path or article.url %}
      {% else %} {% endif %} {% if article.articles.length > 0 %}
        {{ articles(article.articles, file, config) }}
      {% endif %}
    • {% endfor %} {% endmacro %}
        {% set _divider = false %} {% if config.links.sidebar %} {% for linkTitle, link in config.links.sidebar %} {% set _divider = true %}
      • {{ linkTitle }}
      • {% endfor %} {% endif %} {% if _divider %}
      • {% endif %} {% for part in summary.parts %} {% if part.title %}
      • {{ part.title }}
      • {% elif not loop.first %}
      • {% endif %} {{ articles(part.articles, file, config) }} {% endfor %}
      • {{ "GITBOOK_LINK"|t }}
      ================================================ FILE: docs-src/cn/src_doc.md ================================================ AnyProxy =================== > 本文档的适用范围是AnyProxy 4.0,欢迎提供反馈 Ref: [English Doc](../en) AnyProxy是一个开放式的HTTP代理服务器。 Github主页:https://github.com/alibaba/anyproxy 主要特性包括: * 基于Node.js,开放二次开发能力,允许自定义请求处理逻辑 * 支持Https的解析 * 提供GUI界面,用以观察请求 相比3.x版本,AnyProxy 4.0的主要变化: * 规则文件(Rule)全面支持Promise和Generator * 简化了规则文件内的接口 * Web版界面重构 # 快速开始 ## 作为全局模块 ### 安装 对于Debian或者Ubuntu系统,在安装AnyProxy之前,可能还需要安装 `nodejs-legacy` ```bash sudo apt-get install nodejs-legacy ``` 然后,安装AnyProxy ```bash npm install -g anyproxy ``` ### 启动 * 命令行启动AnyProxy,默认端口号8001 ```bash anyproxy ``` * 启动后将终端http代理服务器配置为127.0.0.1:8001即可 * 访问http://127.0.0.1:8002 ,web界面上能看到所有的请求信息 ### 其他命令 * 配置启动端口,如1080端口启动 ```bash anyproxy --port 1080 ``` ## 作为npm模块使用 AnyProxy可以作为一个npm模块使用,整合进其他工具。 > 如要启用https解析,请在代理服务器启动前自行调用`AnyProxy.utils.certMgr`相关方法生成证书,并引导用户信任安装。或引导用户使用`anyproxy-ca`方法。 * 引入 ```bash npm i anyproxy --save ``` * 使用举例 ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, // 不开启websocket代理 silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * 创建代理服务器 ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} 必选,代理服务器端口 * `rule` {object} 自定义规则模块 * `throttle` {number} 限速值,单位kb/s,默认不限速 * `forceProxyHttps` {boolean} 是否强制拦截所有的https,忽略规则模块的返回,默认`false` * `silent` {boolean} 是否屏蔽所有console输出,默认`false` * `dangerouslyIgnoreUnauthorized` {boolean} 是否忽略请求中的证书错误,默认`false` * `wsIntercept` {boolean} 是否开启websocket代理,默认`false` * `webInterface` {object} web版界面配置 * `enable` {boolean} 是否启用web版界面,默认`false` * `webPort` {number} web版界面端口号,默认`8002` * Event: `ready` * 代理服务器启动完成 * 示例 ```js proxy.on('ready', function() { }) ``` * Event: `error` * 代理服务器发生错误 * 示例 ```js proxy.on('error', function() { }) ``` * Method: `start` * 启动代理服务器 * 示例 ```js proxy.start(); ``` * Method: `close` * 关闭代理服务器 * 示例 ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * 管理系统的全局代理配置,方法调用时可能会弹出密码框 * 使用示例 ```js // 配置127.0.0.1:8001为全局http代理服务器 AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // 关闭全局代理服务器 AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * 管理AnyProxy的证书 * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * 校验系统内是否存在AnyProxy的根证书 * `AnyProxy.utils.certMgr.generateRootCA(callback)` * 生成AnyProxy的rootCA,完成后请引导用户信任.crt文件 * 样例 ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # 代理HTTPS * AnyProxy默认不对https请求做处理,如需看到明文信息,需要配置CA证书 > 解析https请求的原理是中间人攻击(man-in-the-middle),用户必须信任AnyProxy生成的CA证书,才能进行后续流程 * 生成证书并解析所有https请求 ```bash anyproxy-ca #生成rootCA证书,生成后需要手动信任 anyproxy --intercept #启动AnyProxy,并解析所有https请求 ``` * [附录:如何信任CA证书](#证书配置) # 代理WebSocket ```bash anyproxy --ws-intercept ``` > 当启用`HTTPS`代理时,`wss`也会被代理,但是不会被AnyProxy记录。需要开启`--ws-intercept`后才会从界面上看到相应内容。 # rule模块 AnyProxy提供了二次开发的能力,你可以用js编写自己的规则模块(rule),来自定义网络请求的处理逻辑。 >注意:引用规则前,请务必确保文件来源可靠,以免发生安全问题 规则模块的能力范围包括: * 拦截并修改正在发送的请求 * 可修改内容包括请求头(request header),请求体(request body),甚至是请求的目标地址等 * 拦截并修改服务端响应 * 可修改的内容包括http状态码(status code)、响应头(response header)、响应内容等 * 拦截https请求,对内容做修改 * 本质是中间人攻击(man-in-the-middle attack),需要客户端提前信任AnyProxy生成的CA ### 开发示例 * 举例 * 需要编写一个规则模块,在 GET http://httpbin.org/user-agent 的返回值里加上测试信息,并延迟5秒返回 * Step 1,编写规则 ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, 启动AnyProxy,加载规则 * 运行 `anyproxy --rule sample.js` * Step 3, 测试规则 * 用curl测试 ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * 用浏览器测试:配置浏览器http代理为 127.0.0.1:8001,访问 http://httpbin.org/user-agent * 经过代理服务器后,期望的返回如下 ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, 查看请求信息 * 浏览器访问http://127.0.0.1:8002 ,界面上能看到刚才的请求信息 ### 处理流程 * 处理流程图如下 * 当http请求经过代理服务器时,具体处理过程是: * 收集请求所有请求参数,包括method, header, body等 * AnyProxy调用规则模块`beforeSendRequest`方法,由模块做处理,返回新的请求参数,或返回响应内容 * 如果`beforeSendRequest`返回了响应内容,则立即把此响应返回到客户端(而不再发送到真正的服务端),流程结束。 * 根据请求参数,向服务端发出请求,接收服务端响应。 * 调用规则模块`beforeSendResponse`方法,由模块对响应内容进行处理 * 把响应信息返回给客户端 * 当代理服务器收到https请求时,AnyProxy可以替换证书,对请求做明文解析。 * 调用规则模块`beforeDealHttpsRequest`方法,如果返回`true`,会明文解析这个请求,其他请求不处理 * 被明文解析后的https请求,处理流程同http一致。未明文解析请求不会再进入规则模块做处理。 ### 如何引用 如下几种方案都可以用来引用规则模块: * 使用本地路径 ```bash anyproxy --rule ./rule.js ``` * 使用在线地址 ```bash anyproxy --rule https://sample.com/rule.js ``` * 使用npm包 * AnyProxy使用`require()`加载本地规则,你可以在参数里传入一个本地的npm包路径,或是某个全局安装的npm包 ```bash anyproxy --rule ./myRulePkg/ #本地包 npm i -g myRulePkg && anyproxy --rule myRulePkg #全局包 ``` # rule接口文档 规则模块应该符合cmd规范,一个典型的规则模块代码结构如下。模块中所有方法都是可选的,只需实现业务感兴趣的部分即可。 ```js module.exports = { // 模块介绍 summary: 'my customized rule for AnyProxy', // 发送请求前拦截处理 *beforeSendRequest(requestDetail) { /* ... */ }, // 发送响应前处理 *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // 是否处理https请求 *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // 请求出错的事件 *onError(requestDetail, error) { /* ... */ }, // https连接服务器出错 *onConnectError(requestDetail, error) { /* ... */ } }; ``` > 规则文件中,除了summary,都是由 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以返回promise或使用generator函数。 ### summary #### summary(): string | summary:string * 规则模块的介绍文案,用于AnyProxy提示用户, 可以是一个函数,也可以是一个普通的字符串 ### beforeSendRequest #### beforeSendRequest(requestDetail) * AnyProxy向服务端发送请求前,会调用`beforeSendRequest`,并带上参数`requestDetail` * `requestDetail` * `protocol` {string} 请求使用的协议,http或者https * `requestOptions` {object} 即将发送的请求配置,供require('http').request作为使用。详见:https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} 请求Body * `url` {string} 请求url * `_req` {object} 请求的原始request * 举例:请求 *anyproxy.io* 时,`requestDetail`参数内容大致如下 ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改请求协议,如强制改用https发起请求 ```js return { protocol: 'https' }; ``` * 修改请求参数 ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * 修改请求body ```js return { requestData: 'my new request data' //这里也可以同时加上requestOptions }; ``` * 直接返回客户端,不再发起请求,其中`statusCode` `header` 是必选字段 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * AnyProxy向客户端发送请求前,会调用`beforeSendResponse`,并带上参数`requestDetail` `responseDetail` * `requestDetail` 同`beforeSendRequest`中的参数 * `responseDetail` * `response` {object} 服务端的返回信息,包括`statusCode` `header` `body`三个字段 * `_res` {object} 原始的服务端返回对象 * 举例,请求 *anyproxy.io* 时,`responseDetail`参数内容大致如下 ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * 以下几种返回都是合法的 * 不做任何处理,返回null ```js return null; ``` * 修改返回的状态码 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * 修改返回的内容 ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * AnyProxy收到https请求时,会调用`beforeDealHttpsRequest`,并带上参数`requestDetail` * 如果配置了全局解析https的参数,则AnyProxy会略过这个调用 * 只有返回`true`时,AnyProxy才会尝试替换证书、解析https。否则只做数据流转发,无法看到明文数据。 * 注意:https over http的代理模式中,这里的request是CONNECT请求 * `requestDetail` * `host` {string} 请求目标的Host,受制于协议,这里无法获取完整url * `_req` {object} 请求的原始request * 返回值 * `true`或者`false`,表示是否需要AnyProxy替换证书并解析https ### onError #### onError(requestDetail, error) * 在请求处理过程中发生错误时,AnyProxy会调用`onError`方法,并提供对应的错误信息 * 多数场景下,错误会在请求目标服务器的时候发生,比如DNS解析失败、请求超时等 * `requestDetail` 同`beforeSendRequest`中的参数 * 以下几种返回都是合法的 * 不做任何处理。此时AnyProxy会返回一个默认的错误页。 ```js return null; ``` * 返回自定义错误页 ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy在与目标HTTPS服务器建立连接的过程中,如果发生错误,AnyProxy会调用这个方法 * `requestDetail` 同`beforeDealHttpsRequest`中的参数 * 此处无法控制向客户端的返回信息,无需返回值。 # rule样例 * 这里提供一些样例,来讲解规则模块的常见用法 * 你可以通过 `anyproxy --rule http://....js` 来加载模块并体验 * 用curl发请求测试的方法如下 * 直接请求服务器:`curl http://httpbin.org/` * 通过代理服务器请求:`curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### 使用本地数据 * 拦截发送到 http://httpbin.org 的请求,使用本地数据代替服务端返回 {{sample-rule:rule_sample/sample_use_local_response.js}} ### 修改请求头 * 修改发送到 httpbin.org 的user-agent {{sample-rule:rule_sample/sample_modify_request_header.js}} ### 修改请求数据 * 修改发送到 http://httpbin.org/post 的post数据 {{sample-rule:rule_sample/sample_modify_request_data.js}} ### 修改请求的目标地址 * 把所有发送到 http://httpbin.org/ 的请求全部改到 http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_request_path.js}} ### 修改请求协议 * 把用http协议请求的 http://httpbin.org 改成https并发送 {{sample-rule:rule_sample/sample_modify_request_protocol.js}} ### 修改返回状态码 * 把 所有http://httpbin.org 的返回状态码都改成404 {{sample-rule:rule_sample/sample_modify_response_statuscode.js}} ### 修改返回头 * 在 http://httpbin.org/user-agent 的返回头里加上 X-Proxy-By:AnyProxy {{sample-rule:rule_sample/sample_modify_response_header.js}} ### 修改返回内容并延迟 * 在 http://httpbin.org/user-agent 的返回最后追加AnyProxy的签名,并延迟5秒 {{sample-rule:rule_sample/sample_modify_response_data.js}} # 证书配置 ### OSX系统信任CA证书 * 类似这种报错都是因为系统没有信任AnyProxy生成的CA所造成的 > 警告:CA证书和系统安全息息相关,建议亲自生成,并妥善保管 安装CA: * 双击打开*rootCA.crt* * 确认将证书添加到login或system * 找到刚刚导入的AnyProxy证书,配置为信任(Always Trust) ### Windows系统信任CA证书 ### 配置OSX系统代理 * 在wifi高级设置中,配置http代理即可 ### 配置浏览器HTTP代理 * 以Chrome的[SwitchyOmega插件](https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### iOS系统信任CA证书 * 点击web ui中的 *Root CA*,按提示扫描二维码即可安装 ### iOS >= 10.3信任CA证书 * 除了上述证书安装过程,还需要在 *设置->通用->关于本机->证书信任设置* 中把AnyProxy证书的开关打开,否则safari将报错。 ### 安卓系统信任CA证书 首先和iOS类似,需要先扫描证书的二维码进行下载。然后不同的安卓系统安装证书的方式可能有所不同,但是安装的步骤是类似的,我们列举了几种类型。 * 下载后的证书可以直接单击打开并安装,这种方式是最简单的,直接安装即可 * 证书下载到指定目录后,需要从其他入口进行安装,包括: * 设置 -> 安全性与位置信息 -> 加密与凭据 -> 从存储设备安装。找到你下载的证书文件,进行安装 * 设置 -> 安全 -> 从SD卡安装证书。找到你下载的证书文件,进行安装 不同安卓系统支持安装的证书文件类型不尽相同,大多支持安装拓展名为 .crt 的证书文件,少部分仅支持 .cer 文件(已知如 OPPO R15),AnyProxy 提供了多种类型的证书文件,可在下载安装时选择。 ### 配置iOS/Android系统代理 * 代理服务器都在wifi设置中配置 * iOS HTTP代理配置 * Android HTTP代理配置 # FAQ #### Q: 为什么https请求不能进入处理函数? A: 以下任意一项都能用来改变https的处理特性: 1. 命令行启动AnyProxy时配置`--intercept`参数,按npm模块启动时配置`forceProxyHttps`参数,所有Https请求都会被替换证书并解析 2. 规则文件内提供`beforeDealHttpsRequest`方法,返回 *true* 的https请求会被解析 #### Q: 提示 *function is not yieldable* * A: 规则模块是用 [co](https://www.npmjs.com/package/co) 驱动的,函数需要满足yieldable。可以使用generator方法或是返回Promise。 #### Q: The connection is not private 当访问特定的HTTPS站点,AnyProxy会提示该站点不是一个安全的网站,这通常是因为站点的证书设置不能被正确识别导致的(比如,站点的证书是自签发的)。如果您信任该网站,可以用以下方式来继续访问: - 命令行直接启动 通过启动参数 `--ignore-unauthorized-ssl` 来忽略证书认证的错误。需要注意的是,该参数是全局生效的,如果你在此期间访问了其他未知的网站,他们的证书问题也会被忽略,这可能会带来安全隐患。 ```bash anyproxy -i --ignore-unauthorized-ssl ``` - 在Nodejs代码中启动 在构造AnyProxy实例的时候,传入参数`dangerouslyIgnoreUnauthorized:true`, 如下: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *通过这种方式初始化的AnyProxy,其配置也是全局性的,所有网站的证书问题都会被忽略* - 通过自定义的Rule来修改 我们自然也可以借助自定义的Rule来实现这个效果,而且我们还可以控制到只允许指定网址的证书错误,对不在列表的网址,进行证书的强验证。 ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // 设置属性 rejectUnauthorized 为 false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` ================================================ FILE: docs-src/en/README.md ================================================ AnyProxy =================== AnyProxy is a fully configurable http/https proxy in NodeJS. Ref: [中文文档](../cn) Github: * https://github.com/alibaba/anyproxy Features: * Offer you the ablity to handle http traffic by invoking a js module * Intercept https * GUI webinterface Change Logs since 3.x: * Support Promise and Generator in rule module * Simplified interface in rule module * A newly designed web interface # Getting Start ### install To Debian and Ubuntu users, you may need to install `nodejs-legacy` at the same time ```bash sudo apt-get install nodejs-legacy ``` Then install the AnyProxy ```bash npm install -g anyproxy ``` ### launch * start AnyProxy in command line, with default port 8001 ```bash anyproxy ``` * now you can use http proxy server by 127.0.0.1:8001 * visit http://127.0.0.1:8002 to see the http requests ### options * specify the port of http proxy ```bash anyproxy --port 1080 ``` ### Use AnyProxy as an npm module AnyProxy can be used as an npm module > To enable https feature, please guide users to use `anyproxy-ca` in cli. Or use methods under `AnyProxy.utils.certMgr` to generate certificates. * install ```bash npm i anyproxy --save ``` * sample ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * create a proxy server ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} required, port number of proxy server * `rule` {object} your rule module * `throttle` {number} throttle in kb/s, unlimited for default * `forceProxyHttps` {boolean} in force intercept all https request, default to `false` * `silent` {boolean} if keep silent in console, false for default `false` * `dangerouslyIgnoreUnauthorized` {boolean} if ignore certificate error in request, default to `false` * `wsIntercept` {boolean} whether to intercept websocket, default to `false` * `webInterface` {object} config for web interface * `enable` {boolean} if enable web interface, default to `false` * `webPort` {number} port number for web interface * Event: `ready` * emit when proxy server is ready * sample ```js proxy.on('ready', function() { }) ``` * Event: `error` * emit when error happened inside proxy server * sample ```js proxy.on('error', function() { }) ``` * Method: `start` * start proxy server * sample ```js proxy.start(); ``` * Method: `close` * close proxy server * sample ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * manage the system proxy config. sudo password may be required * sample ```js // set 127.0.0.1:8001 as system http server AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // disable global proxy server AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * Manage certificates of AnyProxy * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * detect if AnyProx rootCA exists * `AnyProxy.utils.certMgr.generateRootCA(callback)` * generate a rootCA * Sample ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # Proxy Https * AnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate. > Under the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network. * generate certifycates and intercept ```bash anyproxy-ca #generate root CA. manually trust it after that. anyproxy --intercept #launch anyproxy and intercept all https traffic ``` * [Appendix:how to trust CA](#config-certification) # Proxy WebSocket ```bash anyproxy --ws-intercept ``` > The `wss` requests will be handled automatically when the `HTTPS` intercept is turned on, but AnyProxy will not record the data by default. You need to specify the `--ws-intercept` to tell AnyProxy to record it. # Rule Introduction AnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests. > Make sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk. Rule module could do the following stuff: * intercept and modify the request which is being sent * editable fields include request header, body, target address * intercept and modify the response from server * editable fields include response status code, header, body * intercept https requests, modify request and response ### sample * Target * write a rule module to append some text to the response of GET http://httpbin.org/user-agent, and delay the response for 5 seconds * Step 1,Write the rule file, save as sample.js ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, start AnyProxy and load the rule file * run `anyproxy --rule sample.js` * Step 3, test * use curl ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * use browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent * the expected response from proxy is ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, view the request log * visit http://127.0.0.1:8002, the request just sent should be listed here ### how does it work * The flow chart is as follows * When got an http request, the entire process of proxy server is * AnyProxy collects all the quest info, include method, header, body * AnyProxy calls `beforeSendRequest` of the rule module. Rule module deal the request, return new request param or response content * If `beforeSendRequest` returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here. * Send request to target server, collect response * Call `beforeSendResponse` of the rule module. Rule module deal the response data * Send response to client * When AnyProxy get https request, it could replace the certificate and decrypt the request data * AnyProxy calls `beforeDealHttpsRequest` of the rule module * If the function returns `true`, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed. ### how to load rule module * use local file ```bash anyproxy --rule ./rule.js ``` * use an online rule file ```bash anyproxy --rule https://sample.com/rule.js ``` * use an npm module * AnyProxy uses `require()` to load rule module. You could either load a local npm module or a global-installed one. ```bash anyproxy --rule ./myRulePkg/ #local module npm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module ``` # Rule module interface A typical rule module is as follows. All the functions are optional, just write the part you are interested in. ```js module.exports = { // introduction summary: 'my customized rule for AnyProxy', // intercept before send request to server *beforeSendRequest(requestDetail) { /* ... */ }, // deal response before send to client *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // if deal https request *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // error happened when dealing requests *onError(requestDetail, error) { /* ... */ }, // error happened when connect to https server *onConnectError(requestDetail, error) { /* ... */ } }; ``` > All functions in your rule file, except summary, are all driven by [co](https://www.npmjs.com/package/co) . They should be yieldable, i.e. return a promise or be a generator function. ### summary #### summary * Introduction of this rule file. AnyProxy will read this field and give some tip to user. ### beforeSendRequest #### beforeSendRequest(requestDetail) * Before sending request to server, AnyProxy will call `beforeSendRequest` with param `requestDetail` * `requestDetail` * `protocol` {string} the protocol to use, http or https * `requestOptions` {object} the options of the request-to-go, a param of require('http').request . ref: https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} request body * `url` {string} request url * `_req` {object} the native node.js request object * e.g. When requesting *anyproxy.io*, `requestDetail` is something like the following ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the request protocol,i.e. force use https ```js return { protocol: 'https' }; ``` * modify request param ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * modify request body ```js return { requestData: 'my new request data' // requestOptions can also be used here }; ``` * give response to the client, not sending request any longer. `statusCode` `headers`are required is this situation. ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * Before sending response to client, AnyProxy will call `beforeSendResponse` with param `requestDetail` `responseDetail` * `requestDetail` is the same param as in `beforeSendRequest` * `responseDetail` * `response` {object} the response from server, includes `statusCode` `header` `body` * `_res` {object} the native node.js response object * e.g. When requesting *anyproxy.io*, `responseDetail` is something like the following ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the response status code ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * modify the response content ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * When receiving https request, AnyProxy will call `beforeDealHttpsRequest` with param `requestDetail` * If configed with `forceProxyHttps` in launching, AnyProxy will skip calling this method * Only by returning true, AnyProxy will try to replace the certificate and intercept the https request. * `requestDetail` * `host` {string} the target host to request. Due to the request protocol, full url couldn't be got here * `_req` {object} the native node.js request object. The `_req` here refers to the CONNECT request. * return value * `true` or `false`, whether AnyProxy should intercept the https request ### onError #### onError(requestDetail, error) * AnyProxy will call this method when an error happened in request handling. * Errors usually are issued during requesting, e.g. DNS failure, request timeout * `requestDetail` is the same one as in `beforeSendRequest` * Any of these return values are valid * do nothing, and AnyProxy will response a default error page ```js return null; ``` * return a customized error page ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy will call this method when failed to connect target server in https request * `requestDetail` is the same one as in `beforeDealHttpsRequest` * no return value is required # Rule Samples * here are some samples about frequently used rule file * try these samples by `anyproxy --rule http://....js` * how to test with curl: * request the server directly `curl http://httpbin.org/` * request the server via proxy `curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### use local response * intercept the request towards http://httpbin.org , return the local-defined response ```bash anyproxy --rule rule_sample/sample_use_local_response.js ``` ```js /* sample: intercept all requests toward httpbin.org, use a local response test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { const localResponse = { statusCode: 200, header: { 'Content-Type': 'application/json' }, body: '{"hello": "this is local response"}' }; if (requestDetail.url.indexOf('http://httpbin.org') === 0) { return { response: localResponse }; } }, }; ``` ### modify request header * modify the user-agent sent to httpbin.org ```bash anyproxy --rule rule_sample/sample_modify_request_header.js ``` ```js /* sample: modify the user-agent in requests toward httpbin.org test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newRequestOptions = requestDetail.requestOptions; newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0'; return { requestOptions: newRequestOptions }; } }, }; ``` ### modify request body * modify the post body of http://httpbin.org/post ```bash anyproxy --rule rule_sample/sample_modify_request_data.js ``` ```js /* sample: modify the post data towards http://httpbin.org/post test: curl -H "Content-Type: text/plain" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001 expected response: { "data": "i-am-anyproxy-modified-post-data" } */ module.exports = { summary: 'Rule to modify request data', *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) { return { requestData: 'i-am-anyproxy-modified-post-data' }; } }, }; ``` ### modify the request target * send all the request towards http://httpbin.org/ to http://httpbin.org/user-agent ```bash anyproxy --rule rule_sample/sample_modify_request_path.js ``` ```js /* sample: redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html test: curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001 expected response: 'hello world' from 127.0.0.1:8001/index.html */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) { const newRequestOptions = requestDetail.requestOptions; requestDetail.protocol = 'http'; newRequestOptions.hostname = '127.0.0.1' newRequestOptions.port = '8008'; newRequestOptions.path = '/index.html'; newRequestOptions.method = 'GET'; return requestDetail; } }, *beforeDealHttpsRequest(requestDetail) { return true; } }; ``` ### modify request protocol * modify the http request towards http://httpbin.org to https ```bash anyproxy --rule rule_sample/sample_modify_request_protocol.js ``` ```js /* sample: redirect all http requests of httpbin.org to https test: curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001 expected response: { "X-Forwarded-Protocol": "https" } */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newOption = requestDetail.requestOptions; newOption.port = 443; return { protocol: 'https', requestOptions: newOption }; } } }; ``` ### modify response status code * modify all status code from http://httpbin.org to 404 ```bash anyproxy --rule rule_sample/sample_modify_response_statuscode.js ``` ```js /* sample: modify all status code of http://httpbin.org/ to 404 test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: HTTP/1.1 404 Not Found */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newResponse = responseDetail.response; newResponse.statusCode = 404; return { response: newResponse }; } } }; ``` ### modify the response header * add X-Proxy-By:AnyProxy to the response header from http://httpbin.org/user-agent ```bash anyproxy --rule rule_sample/sample_modify_response_header.js ``` ```js /* sample: modify response header of http://httpbin.org/user-agent test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: X-Proxy-By: AnyProxy */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) { const newResponse = responseDetail.response; newResponse.header['X-Proxy-By'] = 'AnyProxy'; return { response: newResponse }; } } }; ``` ### modify response data and delay * append some info to the response of http://httpbin.org/user-agent, then delay the response for 5 seconds. ```bash anyproxy --rule rule_sample/sample_modify_response_data.js ``` ```js /* sample: modify response data of http://httpbin.org/user-agent test: curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: { "user-agent": "curl/7.43.0" } -- AnyProxy Hacked! -- */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '-- AnyProxy Hacked! --'; return new Promise((resolve, reject) => { setTimeout(() => { // delay the response for 5s resolve({ response: newResponse }); }, 5000); }); } }, }; ``` # Config Certification ### Config root CA in OSX * this kind of errors is usually caused by untrusted root CA > Warning: please keep your root CA safe since it may influence your system security. install : * double click *rootCA.crt* * add cert into login or system * find the newly imported AnyProxy certificates, configured as **Always Trust** ### Config root CA in windows ### Config OSX system proxy * the config is in wifi - advanced ### config http proxy server * take Chrome extent [SwitchyOmega] as an example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### trust root CA in iOS * Click *Root CA* in web ui, and follow the instruction to install ### trust root CA in iOS after 10.3 * Besides installing root CA, you have to "turn on" the certificate for web manually in *settings - general - about - Certificate Trust Settings*. Otherwire, safari will not trust the root CA generated by AnyProxy. ### trust root CA in Android First of all, you need to download the root CA by clicking *Root CA* in web ui, and then scan the QR code. Installing CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path. * The downloaded CA file can be directly installed by clicking, this is the easist way * You need to install the CA file from other menu, such as: * Settings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install * Settings -> Security -> Install from SD card, and find you CA file to install There are several file extensions of CA file which may not be compatible with all kinds of Android phones. `.crt` file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing. ### config iOS/Android proxy server * proxy settings are placed in wifi setting * iOS * Android # FAQ ### Q: can not deal https request in rule module. * A: Any of these options could be used to change the way AnyProxy deall https requests 1. config `--intercept` when luanching AnyProxy via cli, or use `forceProxyHttps` when using as an npm module 2. place a `beforeDealHttpsRequest` function in your rule file and determine which request to intercept by your own. ### Q: get an error says *function is not yieldable* * A: Rule module is driven by [co](https://www.npmjs.com/package/co). The functions inside should be yieldable, i.e. return a promise or be a generator function. ### Q: The connection is not private AnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below. - If you run AnyProxy by command line Pass in the option `--ignore-unauthorized-ssl` to ignore the certification errors, please mind that the option will be active for all connections. ```bash anyproxy -i --ignore-unauthorized-ssl ``` - If you run AnyProxy by Nodejs Pass in the option `dangerouslyIgnoreUnauthorized:true`, like this: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *This is also a global option, all certification errors will be ignored* - With the help of AnyProxy Rule You can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs `Http.rquest`, as we do in AnyProxy. A simple demo below: ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // set rejectUnauthorized as false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` And we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to! ================================================ FILE: docs-src/en/SUMMARY.md ================================================ # Summary * [Introduction](README.md) * [Getting-Start](README.md#getting-start) * [Install](README.md#install) * [Launch](README.md#launch) * [Options](README.md#options) * [As Node Module](README.md#use-anyproxy-as-an-npm-module) * [Proxy HTTPS](README.md#proxy-https) * [Proxy WebSocket](README.md#proxy-websocket) * [Rule Introduction](README.md#rule-introduction) * [Sample](README.md#sample) * [How Does It Work](README.md#how-does-it-work) * [Load A Rule](README.md#how-to-load-rule-module) * [Rule Module Interfaces](README.md#rule-module-interface) * [summary_class=rule-title](README.md#summary) * [beforeSendRequest_class=rule-title](README.md#beforesendrequest) * [beforeSendResponse_class=rule-title](README.md#beforesendresponse) * [beforeDealHttpsRequest_class=rule-title](README.md#beforedealhttpsrequest) * [onError_class=rule-title](README.md#onerror) * [onConnectError_class=rule-title](README.md#onconnecterror) * [Rule Samples](README.md#rule-samples) * [Use local response_class=sample-title](README.md#use-local-response) * [Modify Request Header_class=sample-title](README.md#modify-request-header) * [Modify Request Body_class=sample-title](README.md#modify-request-body) * [Modify The Request Target_class=sample-title](README.md#modify-the-request-target) * [Modify Request Protocol_class=sample-title](README.md#modify-request-protocol) * [Modify Response Status Code_class=sample-title](README.md#modify-response-status-code) * [Modify The Response Header_class=sample-title](README.md#modify-the-response-header) * [Modify Response Data And Delay_class=sample-title](README.md#modify-response-data-and-delay) * [Config Certification](README.md#config-certification) * [Config Root CA In OSX](README.md#config-root-ca-in-osx) * [Configure Root CA In windows](README.md#config-root-ca-in-windows) * [Config OSX System Proxy](README.md#config-osx-system-proxy) * [Config As Http Proxy Server](README.md#config-http-proxy-server) * [Trust Root CA In IOS](README.md#trust-root-ca-in-ios) * [Trust Root CA In iOS after 10.3](README.md#trust-root-ca-in-ios-after-103) * [Trust Root CA In Android](README.md#trust-root-ca-in-android) * [Config IOS/Android Proxy Server](README.md#config-iosandroid-proxy-server) * [FAQ](README.md) ================================================ FILE: docs-src/en/_layouts/layout.html ================================================ {% block title %}{{ config.title|d("GitBook", true) }}{% endblock %} {% if config.author %}{% endif %} {% if config.isbn %}{% endif %} {% block style %} {% for resource in plugins.resources.css %} {% if resource.url %} {% else %} {% endif %} {% endfor %} {% endblock %} {% block body %}{% endblock %} {% block javascript %}{% endblock %} ================================================ FILE: docs-src/en/_layouts/website/summary.html ================================================ {% macro articles(_articles) %} {% for article in _articles %}
    • {% if article.path and getPageByPath(article.path) %} {% elif article.url %} {% else %} {% endif %} {% if article.level != "0" and config.pluginsConfig['theme-default'].showLevel %} {{ article.level }}. {% endif %}
      {{ article.title | replace(r/_class=.+/, '') }}
      {% if article.path or article.url %}
      {% else %} {% endif %} {% if article.articles.length > 0 %}
        {{ articles(article.articles, file, config) }}
      {% endif %}
    • {% endfor %} {% endmacro %}
        {% set _divider = false %} {% if config.links.sidebar %} {% for linkTitle, link in config.links.sidebar %} {% set _divider = true %}
      • {{ linkTitle }}
      • {% endfor %} {% endif %} {% if _divider %}
      • {% endif %} {% for part in summary.parts %} {% if part.title %}
      • {{ part.title }}
      • {% elif not loop.first %}
      • {% endif %} {{ articles(part.articles, file, config) }} {% endfor %}
      • {{ "GITBOOK_LINK"|t }}
      ================================================ FILE: docs-src/en/book.json ================================================ { "styles": { "website": "styles/website.css" } } ================================================ FILE: docs-src/en/src_doc.md ================================================ AnyProxy =================== AnyProxy is a fully configurable http/https proxy in NodeJS. Ref: [中文文档](../cn) Github: * https://github.com/alibaba/anyproxy Features: * Offer you the ablity to handle http traffic by invoking a js module * Intercept https * GUI webinterface Change Logs since 3.x: * Support Promise and Generator in rule module * Simplified interface in rule module * A newly designed web interface # Getting Start ### install To Debian and Ubuntu users, you may need to install `nodejs-legacy` at the same time ```bash sudo apt-get install nodejs-legacy ``` Then install the AnyProxy ```bash npm install -g anyproxy ``` ### launch * start AnyProxy in command line, with default port 8001 ```bash anyproxy ``` * now you can use http proxy server by 127.0.0.1:8001 * visit http://127.0.0.1:8002 to see the http requests ### options * specify the port of http proxy ```bash anyproxy --port 1080 ``` ### Use AnyProxy as an npm module AnyProxy can be used as an npm module > To enable https feature, please guide users to use `anyproxy-ca` in cli. Or use methods under `AnyProxy.utils.certMgr` to generate certificates. * install ```bash npm i anyproxy --save ``` * sample ```js const AnyProxy = require('anyproxy'); const options = { port: 8001, rule: require('myRuleModule'), webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: false, wsIntercept: false, silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { /* */ }); proxyServer.on('error', (e) => { /* */ }); proxyServer.start(); //when finished proxyServer.close(); ``` * Class: AnyProxy.proxyServer * create a proxy server ```js const proxy = new AnyProxy.proxyServer(options) ``` * `options` * `port` {number} required, port number of proxy server * `rule` {object} your rule module * `throttle` {number} throttle in kb/s, unlimited for default * `forceProxyHttps` {boolean} in force intercept all https request, default to `false` * `silent` {boolean} if keep silent in console, false for default `false` * `dangerouslyIgnoreUnauthorized` {boolean} if ignore certificate error in request, default to `false` * `wsIntercept` {boolean} whether to intercept websocket, default to `false` * `webInterface` {object} config for web interface * `enable` {boolean} if enable web interface, default to `false` * `webPort` {number} port number for web interface * Event: `ready` * emit when proxy server is ready * sample ```js proxy.on('ready', function() { }) ``` * Event: `error` * emit when error happened inside proxy server * sample ```js proxy.on('error', function() { }) ``` * Method: `start` * start proxy server * sample ```js proxy.start(); ``` * Method: `close` * close proxy server * sample ```js proxy.close(); ``` * AnyProxy.utils.systemProxyMgr * manage the system proxy config. sudo password may be required * sample ```js // set 127.0.0.1:8001 as system http server AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); // disable global proxy server AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); ``` * AnyProxy.utils.certMgr * Manage certificates of AnyProxy * `AnyProxy.utils.certMgr.ifRootCAFileExists()` * detect if AnyProx rootCA exists * `AnyProxy.utils.certMgr.generateRootCA(callback)` * generate a rootCA * Sample ```js const AnyProxy = require('anyproxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } ``` # Proxy Https * AnyProxy does NOT intercept https requests by default. To view decrypted info, you have to config the CA certificate. > Under the hood, AnyProxy decryptes https requests by man-in-the-middle attack. Users have to trust the CA cert in advance. Otherwise, client side will issue errors about unsecure network. * generate certifycates and intercept ```bash anyproxy-ca #generate root CA. manually trust it after that. anyproxy --intercept #launch anyproxy and intercept all https traffic ``` * [Appendix:how to trust CA](#config-certification) # Proxy WebSocket ```bash anyproxy --ws-intercept ``` > The `wss` requests will be handled automatically when the `HTTPS` intercept is turned on, but AnyProxy will not record the data by default. You need to specify the `--ws-intercept` to tell AnyProxy to record it. # Rule Introduction AnyProxy provides the ability to load your own rules written in javascript. With rule module, you could customize the logic to handle requests. > Make sure your rule file is got from a trusted source. Otherwise, you may face some unknown security risk. Rule module could do the following stuff: * intercept and modify the request which is being sent * editable fields include request header, body, target address * intercept and modify the response from server * editable fields include response status code, header, body * intercept https requests, modify request and response ### sample * Target * write a rule module to append some text to the response of GET http://httpbin.org/user-agent, and delay the response for 5 seconds * Step 1,Write the rule file, save as sample.js ```js // file: sample.js module.exports = { summary: 'a rule to hack response', *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '- AnyProxy Hacked!'; return new Promise((resolve, reject) => { setTimeout(() => { // delay resolve({ response: newResponse }); }, 5000); }); } }, }; ``` * Step 2, start AnyProxy and load the rule file * run `anyproxy --rule sample.js` * Step 3, test * use curl ```bash curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 ``` * use browser. Point the http proxy of browser to 127.0.0.1:8001, then visit http://httpbin.org/user-agent * the expected response from proxy is ``` { "user-agent": "curl/7.43.0" } - AnyProxy Hacked! ``` * Step 4, view the request log * visit http://127.0.0.1:8002, the request just sent should be listed here ### how does it work * The flow chart is as follows * When got an http request, the entire process of proxy server is * AnyProxy collects all the quest info, include method, header, body * AnyProxy calls `beforeSendRequest` of the rule module. Rule module deal the request, return new request param or response content * If `beforeSendRequest` returns the response content, AnyProxy will send the response to client without sending to target server. The process ends here. * Send request to target server, collect response * Call `beforeSendResponse` of the rule module. Rule module deal the response data * Send response to client * When AnyProxy get https request, it could replace the certificate and decrypt the request data * AnyProxy calls `beforeDealHttpsRequest` of the rule module * If the function returns `true`, AnyProxy will do the man-in-the-middle attack to it. Otherwise, the request will not be dealed. ### how to load rule module * use local file ```bash anyproxy --rule ./rule.js ``` * use an online rule file ```bash anyproxy --rule https://sample.com/rule.js ``` * use an npm module * AnyProxy uses `require()` to load rule module. You could either load a local npm module or a global-installed one. ```bash anyproxy --rule ./myRulePkg/ #local module npm i -g myRulePkg && anyproxy --rule myRulePkg #global-installed module ``` # Rule module interface A typical rule module is as follows. All the functions are optional, just write the part you are interested in. ```js module.exports = { // introduction summary: 'my customized rule for AnyProxy', // intercept before send request to server *beforeSendRequest(requestDetail) { /* ... */ }, // deal response before send to client *beforeSendResponse(requestDetail, responseDetail) { /* ... */ }, // if deal https request *beforeDealHttpsRequest(requestDetail) { /* ... */ }, // error happened when dealing requests *onError(requestDetail, error) { /* ... */ }, // error happened when connect to https server *onConnectError(requestDetail, error) { /* ... */ } }; ``` > All functions in your rule file, except summary, are all driven by [co](https://www.npmjs.com/package/co) . They should be yieldable, i.e. return a promise or be a generator function. ### summary #### summary * Introduction of this rule file. AnyProxy will read this field and give some tip to user. ### beforeSendRequest #### beforeSendRequest(requestDetail) * Before sending request to server, AnyProxy will call `beforeSendRequest` with param `requestDetail` * `requestDetail` * `protocol` {string} the protocol to use, http or https * `requestOptions` {object} the options of the request-to-go, a param of require('http').request . ref: https://nodejs.org/api/http.html#http_http_request_options_callback * `requestData` {object} request body * `url` {string} request url * `_req` {object} the native node.js request object * e.g. When requesting *anyproxy.io*, `requestDetail` is something like the following ```js { protocol: 'http', url: 'http://anyproxy.io/', requestOptions: { hostname: 'anyproxy.io', port: 80, path: '/', method: 'GET', headers: { Host: 'anyproxy.io', 'Proxy-Connection': 'keep-alive', 'User-Agent': '...' } }, requestData: '...', _req: { /* ... */} } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the request protocol,i.e. force use https ```js return { protocol: 'https' }; ``` * modify request param ```js var newOption = Object.assign({}, requestDetail.requestOptions); newOption.path = '/redirect/to/another/path'; return { requestOptions: newOption }; ``` * modify request body ```js return { requestData: 'my new request data' // requestOptions can also be used here }; ``` * give response to the client, not sending request any longer. `statusCode` `headers`are required is this situation. ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### beforeSendResponse #### beforeSendResponse(requestDetail, responseDetail) * Before sending response to client, AnyProxy will call `beforeSendResponse` with param `requestDetail` `responseDetail` * `requestDetail` is the same param as in `beforeSendRequest` * `responseDetail` * `response` {object} the response from server, includes `statusCode` `header` `body` * `_res` {object} the native node.js response object * e.g. When requesting *anyproxy.io*, `responseDetail` is something like the following ```js { response: { statusCode: 200, header: { 'Content-Type': 'image/gif', Connection: 'close', 'Cache-Control': '...' }, body: '...' }, _res: { /* ... */ } } ``` * Any of these return values are valid * do nothing, and return null ```js return null; ``` * modify the response status code ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.statusCode = 404; return { response: newResponse }; ``` * modify the response content ```js var newResponse = Object.assign({}, responseDetail.response); newResponse.body += '--from anyproxy--'; return { response: newResponse }; ``` ### beforeDealHttpsRequest #### beforeDealHttpsRequest(requestDetail) * When receiving https request, AnyProxy will call `beforeDealHttpsRequest` with param `requestDetail` * If configed with `forceProxyHttps` in launching, AnyProxy will skip calling this method * Only by returning true, AnyProxy will try to replace the certificate and intercept the https request. * `requestDetail` * `host` {string} the target host to request. Due to the request protocol, full url couldn't be got here * `_req` {object} the native node.js request object. The `_req` here refers to the CONNECT request. * return value * `true` or `false`, whether AnyProxy should intercept the https request ### onError #### onError(requestDetail, error) * AnyProxy will call this method when an error happened in request handling. * Errors usually are issued during requesting, e.g. DNS failure, request timeout * `requestDetail` is the same one as in `beforeSendRequest` * Any of these return values are valid * do nothing, and AnyProxy will response a default error page ```js return null; ``` * return a customized error page ```js return { response: { statusCode: 200, header: { 'content-type': 'text/html' }, body: 'this could be a or ' } }; ``` ### onConnectError #### onConnectError(requestDetail, error) * AnyProxy will call this method when failed to connect target server in https request * `requestDetail` is the same one as in `beforeDealHttpsRequest` * no return value is required # Rule Samples * here are some samples about frequently used rule file * try these samples by `anyproxy --rule http://....js` * how to test with curl: * request the server directly `curl http://httpbin.org/` * request the server via proxy `curl http://httpbin.org/ --proxy http://127.0.0.1:8001` ### use local response * intercept the request towards http://httpbin.org , return the local-defined response {{sample-rule:rule_sample/sample_use_local_response.js}} ### modify request header * modify the user-agent sent to httpbin.org {{sample-rule:rule_sample/sample_modify_request_header.js}} ### modify request body * modify the post body of http://httpbin.org/post {{sample-rule:rule_sample/sample_modify_request_data.js}} ### modify the request target * send all the request towards http://httpbin.org/ to http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_request_path.js}} ### modify request protocol * modify the http request towards http://httpbin.org to https {{sample-rule:rule_sample/sample_modify_request_protocol.js}} ### modify response status code * modify all status code from http://httpbin.org to 404 {{sample-rule:rule_sample/sample_modify_response_statuscode.js}} ### modify the response header * add X-Proxy-By:AnyProxy to the response header from http://httpbin.org/user-agent {{sample-rule:rule_sample/sample_modify_response_header.js}} ### modify response data and delay * append some info to the response of http://httpbin.org/user-agent, then delay the response for 5 seconds. {{sample-rule:rule_sample/sample_modify_response_data.js}} # Config Certification ### Config root CA in OSX * this kind of errors is usually caused by untrusted root CA > Warning: please keep your root CA safe since it may influence your system security. install : * double click *rootCA.crt* * add cert into login or system * find the newly imported AnyProxy certificates, configured as **Always Trust** ### Config root CA in windows ### Config OSX system proxy * the config is in wifi - advanced ### config http proxy server * take Chrome extent [SwitchyOmega] as an example(https://chrome.google.com/webstore/detail/padekgcemlokbadohgkifijomclgjgif)为例 ### trust root CA in iOS * Click *Root CA* in web ui, and follow the instruction to install ### trust root CA in iOS after 10.3 * Besides installing root CA, you have to "turn on" the certificate for web manually in *settings - general - about - Certificate Trust Settings*. Otherwire, safari will not trust the root CA generated by AnyProxy. ### trust root CA in Android First of all, you need to download the root CA by clicking *Root CA* in web ui, and then scan the QR code. Installing CA in Android could be different based on the system, we list some common steps as below, but you can find the right way in you system with similar menu path. * The downloaded CA file can be directly installed by clicking, this is the easist way * You need to install the CA file from other menu, such as: * Settings -> Security & Location > Encryption & credentials -> Install from storage, and find your CA file to install * Settings -> Security -> Install from SD card, and find you CA file to install There are several file extensions of CA file which may not be compatible with all kinds of Android phones. `.crt` file is the most popular, while a few systems could only use .cer file such as OPPO R15. In AnyProxy, you can choose the type of certificate you need before installing. ### config iOS/Android proxy server * proxy settings are placed in wifi setting * iOS * Android # FAQ ### Q: can not deal https request in rule module. * A: Any of these options could be used to change the way AnyProxy deall https requests 1. config `--intercept` when luanching AnyProxy via cli, or use `forceProxyHttps` when using as an npm module 2. place a `beforeDealHttpsRequest` function in your rule file and determine which request to intercept by your own. ### Q: get an error says *function is not yieldable* * A: Rule module is driven by [co](https://www.npmjs.com/package/co). The functions inside should be yieldable, i.e. return a promise or be a generator function. ### Q: The connection is not private AnyProxy will propmt this message when the certification of the site you're visiting is not issued by a common known CA. This happens when the certification is self-signed. If you know and trust it, you can ignore the error as below. - If you run AnyProxy by command line Pass in the option `--ignore-unauthorized-ssl` to ignore the certification errors, please mind that the option will be active for all connections. ```bash anyproxy -i --ignore-unauthorized-ssl ``` - If you run AnyProxy by Nodejs Pass in the option `dangerouslyIgnoreUnauthorized:true`, like this: ```js const options = { ..., dangerouslyIgnoreUnauthorized: true }; const anyproxyIns = new AnyProxy.ProxyCore(options); anyproxyIns.start(); ``` *This is also a global option, all certification errors will be ignored* - With the help of AnyProxy Rule You can change the request with rule of course. For this scenario, all you need is to pass in an option to Nodejs `Http.rquest`, as we do in AnyProxy. A simple demo below: ```js module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://the-site-you-know.com') === 0) { const newRequestOptions = requestDetail.requestOptions; // set rejectUnauthorized as false newRequestOptions.rejectUnauthorized = false; return { requestOptions: newRequestOptions }; } }, }; ``` And we get a bonous here, AnyProxy will only ignore the errors for the site(s) we want it to! ================================================ FILE: jest.config.js ================================================ // For a detailed explanation regarding each configuration property, visit: // https://jestjs.io/docs/en/configuration.html module.exports = { testTimeout: 10 * 1000, // All imported modules in your tests should be mocked automatically // automock: false, // Stop running tests after `n` failures // bail: 0, // Respect "browser" field in package.json when resolving modules // browser: false, // The directory where Jest should store its cached dependency information // cacheDirectory: "/private/var/folders/dn/t1cpcmtx6ng82d7qf8b02w7r0000gn/T/jest_dx", // Automatically clear mock calls and instances between every test clearMocks: true, // Indicates whether the coverage information should be collected while executing the test // collectCoverage: false, // An array of glob patterns indicating a set of files for which coverage information should be collected // collectCoverageFrom: null, // The directory where Jest should output its coverage files coverageDirectory: 'coverage', // An array of regexp pattern strings used to skip coverage collection // coveragePathIgnorePatterns: [ // "/node_modules/" // ], // A list of reporter names that Jest uses when writing coverage reports // coverageReporters: [ // "json", // "text", // "lcov", // "clover" // ], // An object that configures minimum threshold enforcement for coverage results // coverageThreshold: null, // A path to a custom dependency extractor // dependencyExtractor: null, // Make calling deprecated APIs throw helpful error messages // errorOnDeprecated: false, // Force coverage collection from ignored files using an array of glob patterns // forceCoverageMatch: [], // A path to a module which exports an async function that is triggered once before all test suites // globalSetup: null, // A path to a module which exports an async function that is triggered once after all test suites // globalTeardown: null, // A set of global variables that need to be available in all test environments // globals: {}, // The maximum amount of workers used to run your tests. Can be specified as % or a number. E.g. maxWorkers: 10% will use 10% of your CPU amount + 1 as the maximum worker number. maxWorkers: 2 will use a maximum of 2 workers. // maxWorkers: "50%", // An array of directory names to be searched recursively up from the requiring module's location // moduleDirectories: [ // "node_modules" // ], // An array of file extensions your modules use // moduleFileExtensions: [ // "js", // "json", // "jsx", // "ts", // "tsx", // "node" // ], // A map from regular expressions to module names that allow to stub out resources with a single module // moduleNameMapper: {}, // An array of regexp pattern strings, matched against all module paths before considered 'visible' to the module loader // modulePathIgnorePatterns: [], // Activates notifications for test results // notify: false, // An enum that specifies notification mode. Requires { notify: true } // notifyMode: "failure-change", // A preset that is used as a base for Jest's configuration // preset: null, // Run tests from one or more projects // projects: null, // Use this configuration option to add custom reporters to Jest // reporters: undefined, // Automatically reset mock state between every test // resetMocks: false, // Reset the module registry before running each individual test // resetModules: false, // A path to a custom resolver // resolver: null, // Automatically restore mock state between every test // restoreMocks: false, // The root directory that Jest should scan for tests and modules within // rootDir: null, // A list of paths to directories that Jest should use to search for files in // roots: [ // "" // ], // Allows you to use a custom runner instead of Jest's default test runner // runner: "jest-runner", // The paths to modules that run some code to configure or set up the testing environment before each test // setupFiles: [], // A list of paths to modules that run some code to configure or set up the testing framework before each test // setupFilesAfterEnv: [], // A list of paths to snapshot serializer modules Jest should use for snapshot testing // snapshotSerializers: [], // The test environment that will be used for testing testEnvironment: 'node', // Options that will be passed to the testEnvironment // testEnvironmentOptions: {}, // Adds a location field to test results // testLocationInResults: false, // The glob patterns Jest uses to detect test files // testMatch: [ // "**/__tests__/**/*.[jt]s?(x)", // "**/?(*.)+(spec|test).[tj]s?(x)" // ], // An array of regexp pattern strings that are matched against all test paths, matched tests are skipped // testPathIgnorePatterns: [ // "/node_modules/" // ], // The regexp pattern or array of patterns that Jest uses to detect test files // testRegex: [], // This option allows the use of a custom results processor // testResultsProcessor: null, // This option allows use of a custom test runner // testRunner: "jasmine2", // This option sets the URL for the jsdom environment. It is reflected in properties such as location.href // testURL: "http://localhost", // Setting this value to "fake" allows the use of fake timers for functions such as "setTimeout" // timers: "real", // A map from regular expressions to paths to transformers // transform: null, // An array of regexp pattern strings that are matched against all source file paths, matched files will skip transformation // transformIgnorePatterns: [ // "/node_modules/" // ], // An array of regexp pattern strings that are matched against all modules before the module loader will automatically return a mock for them // unmockedModulePathPatterns: undefined, // Indicates whether each individual test should be reported during the run // verbose: null, // An array of regexp patterns that are matched against all source file paths before re-running tests in watch mode // watchPathIgnorePatterns: [], // Whether to use watchman for file crawling // watchman: true, }; ================================================ FILE: lib/certMgr.js ================================================ 'use strict' const EasyCert = require('node-easy-cert'); const co = require('co'); const os = require('os'); const inquirer = require('inquirer'); const util = require('./util'); const logUtil = require('./log'); const options = { rootDirPath: util.getAnyProxyPath('certificates'), inMemory: false, defaultCertAttrs: [ { name: 'countryName', value: 'CN' }, { name: 'organizationName', value: 'AnyProxy' }, { shortName: 'ST', value: 'SH' }, { shortName: 'OU', value: 'AnyProxy SSL Proxy' } ] }; const easyCert = new EasyCert(options); const crtMgr = util.merge({}, easyCert); // rename function crtMgr.ifRootCAFileExists = easyCert.isRootCAFileExists; crtMgr.generateRootCA = function (cb) { doGenerate(false); // set default common name of the cert function doGenerate(overwrite) { const rootOptions = { commonName: 'AnyProxy', overwrite: !!overwrite }; easyCert.generateRootCA(rootOptions, (error, keyPath, crtPath) => { cb(error, keyPath, crtPath); }); } }; crtMgr.getCAStatus = function *() { return co(function *() { const result = { exist: false, }; const ifExist = easyCert.isRootCAFileExists(); if (!ifExist) { return result; } else { result.exist = true; if (!/^win/.test(process.platform)) { result.trusted = yield easyCert.ifRootCATrusted; } return result; } }); } /** * trust the root ca by command */ crtMgr.trustRootCA = function *() { const platform = os.platform(); const rootCAPath = crtMgr.getRootCAFilePath(); const trustInquiry = [ { type: 'list', name: 'trustCA', message: 'The rootCA is not trusted yet, install it to the trust store now?', choices: ['Yes', "No, I'll do it myself"] } ]; if (platform === 'darwin') { const answer = yield inquirer.prompt(trustInquiry); if (answer.trustCA === 'Yes') { logUtil.info('About to trust the root CA, this may requires your password'); // https://ss64.com/osx/security-cert.html const result = util.execScriptSync(`sudo security add-trusted-cert -d -k /Library/Keychains/System.keychain ${rootCAPath}`); if (result.status === 0) { logUtil.info('Root CA install, you are ready to intercept the https now'); } else { console.error(result); logUtil.info('Failed to trust the root CA, please trust it manually'); util.guideToHomePage(); } } else { logUtil.info('Please trust the root CA manually so https interception works'); util.guideToHomePage(); } } if (/^win/.test(process.platform)) { logUtil.info('You can install the root CA manually.'); } logUtil.info('The root CA file path is: ' + crtMgr.getRootCAFilePath()); } module.exports = crtMgr; ================================================ FILE: lib/configUtil.js ================================================ /** * a util to set and get all configuable constant * */ const path = require('path'); const USER_HOME = process.env.HOME || process.env.USERPROFILE; const DEFAULT_ANYPROXY_HOME = path.join(USER_HOME, '/.anyproxy/'); /** * return AnyProxy's home path */ module.exports.getAnyProxyHome = function () { const ENV_ANYPROXY_HOME = process.env.ANYPROXY_HOME || ''; return ENV_ANYPROXY_HOME || DEFAULT_ANYPROXY_HOME; } ================================================ FILE: lib/httpsServerMgr.js ================================================ 'use strict' //manage https servers const async = require('async'), https = require('https'), tls = require('tls'), assert = require('assert'), crypto = require('crypto'), color = require('colorful'), certMgr = require('./certMgr'), logUtil = require('./log'), util = require('./util'), wsServerMgr = require('./wsServerMgr'), constants = require('constants'), asyncTask = require('async-task-mgr'); /** * Create an https server * * @param {object} config * @param {number} config.port * @param {function} config.handler */ function createHttpsSNIServer(port, handler) { assert(port && handler, 'invalid param for https SNI server'); const createSecureContext = tls.createSecureContext || crypto.createSecureContext; function SNIPrepareCert(serverName, SNICallback) { let keyContent, crtContent, ctx; async.series([ (callback) => { certMgr.getCertificate(serverName, (err, key, crt) => { if (err) { callback(err); } else { keyContent = key; crtContent = crt; callback(); } }); }, (callback) => { try { ctx = createSecureContext({ key: keyContent, cert: crtContent }); callback(); } catch (e) { callback(e); } } ], (err) => { if (!err) { const tipText = 'proxy server for __NAME established'.replace('__NAME', serverName); logUtil.printLog(color.yellow(color.bold('[internal https]')) + color.yellow(tipText)); SNICallback(null, ctx); } else { logUtil.printLog('err occurred when prepare certs for SNI - ' + err, logUtil.T_ERR); logUtil.printLog('err occurred when prepare certs for SNI - ' + err.stack, logUtil.T_ERR); SNICallback(err); } }); } return new Promise((resolve) => { const server = https.createServer({ secureOptions: constants.SSL_OP_NO_SSLv3 || constants.SSL_OP_NO_TLSv1, SNICallback: SNIPrepareCert, }, handler).listen(port); resolve(server); }); } function createHttpsIPServer(ip, port, handler) { assert(ip && port && handler, 'invalid param for https IP server'); return new Promise((resolve, reject) => { certMgr.getCertificate(ip, (err, keyContent, crtContent) => { if (err) return reject(err); const server = https.createServer({ secureOptions: constants.SSL_OP_NO_SSLv3 || constants.SSL_OP_NO_TLSv1, key: keyContent, cert: crtContent, }, handler).listen(port); resolve(server); }); }); } class httpsServerMgr { constructor(config) { if (!config || !config.handler) { throw new Error('handler is required'); } this.httpsAsyncTask = new asyncTask(); this.handler = config.handler; this.wsHandler = config.wsHandler this.asyncSNITaskName = `https_SNI_${Math.random()}`; this.activeServers = []; } getSharedHttpsServer(hostname) { const self = this; const ifIPHost = hostname && util.isIp(hostname); const serverHost = '127.0.0.1'; function prepareServer(callback) { let port; Promise.resolve(util.getFreePort()) .then(freePort => { port = freePort; if (ifIPHost) { return createHttpsIPServer(hostname, port, self.handler); } else { return createHttpsSNIServer(port, self.handler); } }) .then(httpsServer => { self.activeServers.push(httpsServer); wsServerMgr.getWsServer({ server: httpsServer, connHandler: self.wsHandler }); httpsServer.on('upgrade', (req, cltSocket, head) => { logUtil.debug('will let WebSocket server to handle the upgrade event'); }); const result = { host: serverHost, port, }; callback(null, result); }) .catch(e => { callback(e); }); } // same server for same host return new Promise((resolve, reject) => { self.httpsAsyncTask.addTask(ifIPHost ? hostname : serverHost, prepareServer, (error, serverInfo) => { if (error) { reject(error); } else { resolve(serverInfo); } }); }); } close() { this.activeServers.forEach(server => { server.close(); }); } } module.exports = httpsServerMgr; ================================================ FILE: lib/log.js ================================================ 'use strict' const color = require('colorful'); const util = require('./util'); let ifPrint = true; let logLevel = 0; const LogLevelMap = { tip: 0, system_error: 1, rule_error: 2, warn: 3, debug: 4, }; function setPrintStatus(status) { ifPrint = !!status; } function setLogLevel(level) { logLevel = parseInt(level, 10); } function printLog(content, type) { if (!ifPrint) { return; } const timeString = util.formatDate(new Date(), 'YYYY-MM-DD hh:mm:ss'); switch (type) { case LogLevelMap.tip: { if (logLevel > 0) { return; } console.log(color.cyan(`[AnyProxy Log][${timeString}]: ` + content)); break; } case LogLevelMap.system_error: { if (logLevel > 1) { return; } console.error(color.red(`[AnyProxy ERROR][${timeString}]: ` + content)); break; } case LogLevelMap.rule_error: { if (logLevel > 2) { return; } console.error(color.red(`[AnyProxy RULE_ERROR][${timeString}]: ` + content)); break; } case LogLevelMap.warn: { if (logLevel > 3) { return; } console.error(color.yellow(`[AnyProxy WARN][${timeString}]: ` + content)); break; } case LogLevelMap.debug: { console.log(color.cyan(`[AnyProxy Log][${timeString}]: ` + content)); return; } default : { console.log(color.cyan(`[AnyProxy Log][${timeString}]: ` + content)); break; } } } module.exports.printLog = printLog; module.exports.debug = (content) => { printLog(content, LogLevelMap.debug); }; module.exports.info = (content) => { printLog(content, LogLevelMap.tip); }; module.exports.warn = (content) => { printLog(content, LogLevelMap.warn); }; module.exports.error = (content) => { printLog(content, LogLevelMap.system_error); }; module.exports.ruleError = (content) => { printLog(content, LogLevelMap.rule_error); }; module.exports.setPrintStatus = setPrintStatus; module.exports.setLogLevel = setLogLevel; module.exports.T_TIP = LogLevelMap.tip; module.exports.T_ERR = LogLevelMap.system_error; module.exports.T_RULE_ERROR = LogLevelMap.rule_error; module.exports.T_WARN = LogLevelMap.warn; module.exports.T_DEBUG = LogLevelMap.debug; ================================================ FILE: lib/recorder.js ================================================ 'use strict' //start recording and share a list when required const Datastore = require('nedb'), path = require('path'), fs = require('fs'), logUtil = require('./log'), events = require('events'), iconv = require('iconv-lite'), fastJson = require('fast-json-stringify'), proxyUtil = require('./util'); const wsMessageStingify = fastJson({ title: 'ws message stringify', type: 'object', properties: { time: { type: 'integer' }, message: { type: 'string' }, isToServer: { type: 'boolean' } } }); const BODY_FILE_PRFIX = 'res_body_'; const WS_MESSAGE_FILE_PRFIX = 'ws_message_'; const CACHE_DIR_PREFIX = 'cache_r'; function getCacheDir() { const rand = Math.floor(Math.random() * 1000000), cachePath = path.join(proxyUtil.getAnyProxyTmpPath(), './' + CACHE_DIR_PREFIX + rand); fs.mkdirSync(cachePath); return cachePath; } function normalizeInfo(id, info) { const singleRecord = {}; //general singleRecord._id = id; singleRecord.id = id; singleRecord.url = info.url; singleRecord.host = info.host; singleRecord.path = info.path; singleRecord.method = info.method; //req singleRecord.reqHeader = info.req.headers; singleRecord.startTime = info.startTime; singleRecord.reqBody = info.reqBody || ''; singleRecord.protocol = info.protocol || ''; //res if (info.endTime) { singleRecord.statusCode = info.statusCode; singleRecord.endTime = info.endTime; singleRecord.resHeader = info.resHeader; singleRecord.length = info.length; const contentType = info.resHeader['content-type'] || info.resHeader['Content-Type']; if (contentType) { singleRecord.mime = contentType.split(';')[0]; } else { singleRecord.mime = ''; } singleRecord.duration = info.endTime - info.startTime; } else { singleRecord.statusCode = ''; singleRecord.endTime = ''; singleRecord.resHeader = ''; singleRecord.length = ''; singleRecord.mime = ''; singleRecord.duration = ''; } return singleRecord; } class Recorder extends events.EventEmitter { constructor(config) { super(config); this.globalId = 1; this.cachePath = getCacheDir(); this.db = new Datastore(); this.recordBodyMap = []; // id - body } setDbAutoCompact() { this.db.persistence.setAutocompactionInterval(5001); } stopDbAutoCompact() { try { this.db.persistence.stopAutocompaction(); } catch (e) { logUtil.printLog(e, logUtil.T_ERR); } } emitUpdate(id, info) { const self = this; if (info) { self.emit('update', info); } else { self.getSingleRecord(id, (err, doc) => { if (!err && !!doc && !!doc[0]) { self.emit('update', doc[0]); } }); } } emitUpdateLatestWsMessage(id, message) { this.emit('updateLatestWsMsg', message); } updateRecord(id, info) { if (id < 0) return; const self = this; const db = self.db; const finalInfo = normalizeInfo(id, info); db.update({ _id: id }, finalInfo); self.updateRecordBody(id, info); self.emitUpdate(id, finalInfo); } /** * This method shall be called at each time there are new message * */ updateRecordWsMessage(id, message) { if (id < 0) return; try { this.getCacheFile(WS_MESSAGE_FILE_PRFIX + id, (err, recordWsMessageFile) => { if (err) return; fs.appendFile(recordWsMessageFile, wsMessageStingify(message) + ',', () => {}); }); } catch (e) { console.error(e); logUtil.error(e.message + e.stack); } this.emitUpdateLatestWsMessage(id, { id: id, message: message }); } updateExtInfo(id, extInfo) { const self = this; const db = self.db; db.update({ _id: id }, { $set: { ext: extInfo } }, {}, (err, nums) => { if (!err) { self.emitUpdate(id); } }); } appendRecord(info) { if (info.req.headers.anyproxy_web_req) { //TODO request from web interface return -1; } const self = this; const db = self.db; const thisId = self.globalId++; const finalInfo = normalizeInfo(thisId, info); db.insert(finalInfo); self.updateRecordBody(thisId, info); self.emitUpdate(thisId, finalInfo); return thisId; } updateRecordBody(id, info) { const self = this; if (id === -1) return; if (!id || typeof info.resBody === 'undefined') return; //add to body map //ignore image data self.getCacheFile(BODY_FILE_PRFIX + id, (err, bodyFile) => { if (err) return; fs.writeFile(bodyFile, info.resBody, () => {}); }); } /** * get body and websocket file * */ getBody(id, cb) { const self = this; if (id < 0) { cb && cb(''); return; } self.getCacheFile(BODY_FILE_PRFIX + id, (error, bodyFile) => { if (error) { cb && cb(error); return; } fs.access(bodyFile, fs.F_OK || fs.R_OK, (err) => { if (err) { cb && cb(err); } else { fs.readFile(bodyFile, cb); } }); }); } getDecodedBody(id, cb) { const self = this; const result = { method: '', type: 'unknown', mime: '', content: '' }; self.getSingleRecord(id, (err, doc) => { //check whether this record exists if (!doc || !doc[0]) { cb(new Error('failed to find record for this id')); return; } // also put the `method` back, so the client can decide whether to load ws messages result.method = doc[0].method; self.getBody(id, (error, bodyContent) => { if (error) { cb(error); } else if (!bodyContent) { cb(null, result); } else { const record = doc[0], resHeader = record.resHeader || {}; try { const headerStr = JSON.stringify(resHeader), charsetMatch = headerStr.match(/charset='?([a-zA-Z0-9-]+)'?/), contentType = resHeader && (resHeader['content-type'] || resHeader['Content-Type']); if (charsetMatch && charsetMatch.length) { const currentCharset = charsetMatch[1].toLowerCase(); if (currentCharset !== 'utf-8' && iconv.encodingExists(currentCharset)) { bodyContent = iconv.decode(bodyContent, currentCharset); } result.content = bodyContent.toString(); result.type = contentType && /application\/json/i.test(contentType) ? 'json' : 'text'; } else if (contentType && /image/i.test(contentType)) { result.type = 'image'; result.content = bodyContent; } else { result.type = contentType; result.content = bodyContent.toString(); } result.mime = contentType; result.fileName = path.basename(record.path); result.statusCode = record.statusCode; } catch (e) { console.error(e); } cb(null, result); } }); }); } /** * get decoded WebSoket messages * */ getDecodedWsMessage(id, cb) { if (id < 0) { cb && cb([]); return; } this.getCacheFile(WS_MESSAGE_FILE_PRFIX + id, (outError, wsMessageFile) => { if (outError) { cb && cb(outError); return; } fs.access(wsMessageFile, fs.F_OK || fs.R_OK, (err) => { if (err) { cb && cb(err); } else { fs.readFile(wsMessageFile, 'utf8', (error, content) => { if (error) { cb && cb(err); } try { // remove the last dash "," if it has, since it's redundant // and also add brackets to make it a complete JSON structure content = `[${content.replace(/,$/, '')}]`; const messages = JSON.parse(content); cb(null, messages); } catch (e) { console.error(e); logUtil.error(e.message + e.stack); cb(e); } }); } }); }); } getSingleRecord(id, cb) { const self = this; const db = self.db; db.find({ _id: parseInt(id, 10) }, cb); } getSummaryList(cb) { const self = this; const db = self.db; db.find({}, cb); } getRecords(idStart, limit, cb) { const self = this; const db = self.db; limit = limit || 10; idStart = typeof idStart === 'number' ? idStart : (self.globalId - limit); db.find({ _id: { $gte: parseInt(idStart, 10) } }) .sort({ _id: 1 }) .limit(limit) .exec(cb); } clear() { logUtil.printLog('clearing cache file...'); const self = this; proxyUtil.deleteFolderContentsRecursive(self.cachePath, true); } getCacheFile(fileName, cb) { const self = this; const cachePath = self.cachePath; const filepath = path.join(cachePath, fileName); if (filepath.indexOf(cachePath) !== 0) { cb && cb(new Error('invalid cache file path')); } else { cb && cb(null, filepath); return filepath; } } } module.exports = Recorder; ================================================ FILE: lib/requestErrorHandler.js ================================================ 'use strict'; /* * handle all request error here, * */ const pug = require('pug'); const path = require('path'); const error502PugFn = pug.compileFile(path.join(__dirname, '../resource/502.pug')); const certPugFn = pug.compileFile(path.join(__dirname, '../resource/cert_error.pug')); /** * get error content for certification issues */ function getCertErrorContent(error, fullUrl) { let content; const title = 'The connection is not private. '; let explain = 'There are error with the certfication of the site.'; switch (error.code) { case 'UNABLE_TO_GET_ISSUER_CERT_LOCALLY': { explain = 'The certfication of the site you are visiting is not issued by a known agency, ' + 'It usually happenes when the cert is a self-signed one.
      ' + 'If you know and trust the site, you can run AnyProxy with option -ignore-unauthorized-ssl to continue.' break; } default: { explain = '' break; } } try { content = certPugFn({ title: title, explain: explain, code: error.code }); } catch (parseErro) { content = error.stack; } return content; } /* * get the default error content */ function getDefaultErrorCotent(error, fullUrl) { let content; try { content = error502PugFn({ error, url: fullUrl, errorStack: error.stack.split(/\n/) }); } catch (parseErro) { content = error.stack; } return content; } /* * get mapped error content for each error */ module.exports.getErrorContent = function (error, fullUrl) { let content = ''; error = error || {}; switch (error.code) { case 'UNABLE_TO_GET_ISSUER_CERT_LOCALLY': { content = getCertErrorContent(error, fullUrl); break; } default: { content = getDefaultErrorCotent(error, fullUrl); break; } } return content; } ================================================ FILE: lib/requestHandler.js ================================================ 'use strict'; const http = require('http'), https = require('https'), net = require('net'), url = require('url'), zlib = require('zlib'), color = require('colorful'), Buffer = require('buffer').Buffer, util = require('./util'), Stream = require('stream'), logUtil = require('./log'), co = require('co'), WebSocket = require('ws'), HttpsServerMgr = require('./httpsServerMgr'), brotliTorb = require('brotli'), Readable = require('stream').Readable; const requestErrorHandler = require('./requestErrorHandler'); // to fix issue with TLS cache, refer to: https://github.com/nodejs/node/issues/8368 https.globalAgent.maxCachedSessions = 0; const DEFAULT_CHUNK_COLLECT_THRESHOLD = 20 * 1024 * 1024; // about 20 mb class CommonReadableStream extends Readable { constructor(config) { super({ highWaterMark: DEFAULT_CHUNK_COLLECT_THRESHOLD * 5 }); } _read(size) { } } /* * get error response for exception scenarios */ const getErrorResponse = (error, fullUrl) => { // default error response const errorResponse = { statusCode: 500, header: { 'Content-Type': 'text/html; charset=utf-8', 'Proxy-Error': true, 'Proxy-Error-Message': error ? JSON.stringify(error) : 'null' }, body: requestErrorHandler.getErrorContent(error, fullUrl) }; return errorResponse; } /** * fetch remote response * * @param {string} protocol * @param {object} options options of http.request * @param {buffer} reqData request body * @param {object} config * @param {boolean} config.dangerouslyIgnoreUnauthorized * @param {boolean} config.chunkSizeThreshold * @returns */ function fetchRemoteResponse(protocol, options, reqData, config) { reqData = reqData || ''; return new Promise((resolve, reject) => { delete options.headers['content-length']; // will reset the content-length after rule delete options.headers['Content-Length']; delete options.headers['Transfer-Encoding']; delete options.headers['transfer-encoding']; if (config.dangerouslyIgnoreUnauthorized) { options.rejectUnauthorized = false; } if (!config.chunkSizeThreshold) { throw new Error('chunkSizeThreshold is required'); } //send request const proxyReq = (/https/i.test(protocol) ? https : http).request(options, (res) => { res.headers = util.getHeaderFromRawHeaders(res.rawHeaders); //deal response header const statusCode = res.statusCode; const resHeader = res.headers; let resDataChunks = []; // array of data chunks or stream const rawResChunks = []; // the original response chunks let resDataStream = null; let resSize = 0; const finishCollecting = () => { new Promise((fulfill, rejectParsing) => { if (resDataStream) { fulfill(resDataStream); } else { const serverResData = Buffer.concat(resDataChunks); const originContentLen = util.getByteSize(serverResData); // remove gzip related header, and ungzip the content // note there are other compression types like deflate const contentEncoding = resHeader['content-encoding'] || resHeader['Content-Encoding']; const ifServerGzipped = /gzip/i.test(contentEncoding); const isServerDeflated = /deflate/i.test(contentEncoding); const isBrotlied = /br/i.test(contentEncoding); /** * when the content is unzipped, update the header content */ const refactContentEncoding = () => { if (contentEncoding) { resHeader['x-anyproxy-origin-content-encoding'] = contentEncoding; delete resHeader['content-encoding']; delete resHeader['Content-Encoding']; } } // set origin content length into header resHeader['x-anyproxy-origin-content-length'] = originContentLen; // only do unzip when there is res data if (ifServerGzipped && originContentLen) { refactContentEncoding(); zlib.gunzip(serverResData, (err, buff) => { if (err) { rejectParsing(err); } else { fulfill(buff); } }); } else if (isServerDeflated && originContentLen) { refactContentEncoding(); zlib.inflate(serverResData, (err, buff) => { if (err) { rejectParsing(err); } else { fulfill(buff); } }); } else if (isBrotlied && originContentLen) { refactContentEncoding(); try { // an Unit8Array returned by decompression const result = brotliTorb.decompress(serverResData); fulfill(Buffer.from(result)); } catch (e) { rejectParsing(e); } } else { fulfill(serverResData); } } }).then((serverResData) => { resolve({ statusCode, header: resHeader, body: serverResData, rawBody: rawResChunks, _res: res, }); }).catch((e) => { reject(e); }); }; //deal response data res.on('data', (chunk) => { rawResChunks.push(chunk); if (resDataStream) { // stream mode resDataStream.push(chunk); } else { // dataChunks resSize += chunk.length; resDataChunks.push(chunk); // stop collecting, convert to stream mode if (resSize >= config.chunkSizeThreshold) { resDataStream = new CommonReadableStream(); while (resDataChunks.length) { resDataStream.push(resDataChunks.shift()); } resDataChunks = null; finishCollecting(); } } }); res.on('end', () => { if (resDataStream) { resDataStream.push(null); // indicate the stream is end } else { finishCollecting(); } }); res.on('error', (error) => { logUtil.printLog('error happend in response:' + error, logUtil.T_ERR); reject(error); }); }); proxyReq.on('error', reject); proxyReq.end(reqData); }); } /** * get request info from the ws client, includes: host port path protocol ws/wss @param @required wsClient the ws client of WebSocket * */ function getWsReqInfo(wsReq) { const headers = wsReq.headers || {}; const host = headers.host; const hostName = host.split(':')[0]; const port = host.split(':')[1]; // TODO 如果是windows机器,url是不是全路径?需要对其过滤,取出 const path = wsReq.url || '/'; const isEncript = wsReq.connection && wsReq.connection.encrypted; /** * construct the request headers based on original connection, * but delete the `sec-websocket-*` headers as they are already consumed by AnyProxy */ const getNoWsHeaders = () => { const originHeaders = Object.assign({}, headers); const originHeaderKeys = Object.keys(originHeaders); originHeaderKeys.forEach((key) => { // if the key matchs 'sec-websocket', delete it if (/sec-websocket/ig.test(key)) { delete originHeaders[key]; } }); delete originHeaders.connection; delete originHeaders.upgrade; return originHeaders; } return { headers: headers, // the full headers of origin ws connection noWsHeaders: getNoWsHeaders(), hostName: hostName, port: port, path: path, protocol: isEncript ? 'wss' : 'ws' }; } /** * get a request handler for http/https server * * @param {RequestHandler} reqHandlerCtx * @param {object} userRule * @param {Recorder} recorder * @returns */ function getUserReqHandler(userRule, recorder) { const reqHandlerCtx = this return function (req, userRes) { /* note req.url is wired in http server: http://www.example.com/a/b/c in https server: /a/b/c */ const host = req.headers.host; const protocol = (!!req.connection.encrypted && !(/^http:/).test(req.url)) ? 'https' : 'http'; // try find fullurl https://github.com/alibaba/anyproxy/issues/419 let fullUrl = protocol + '://' + host + req.url; if (protocol === 'http') { const reqUrlPattern = url.parse(req.url); if (reqUrlPattern.host && reqUrlPattern.protocol) { fullUrl = req.url; } } const urlPattern = url.parse(fullUrl); const path = urlPattern.path; const chunkSizeThreshold = DEFAULT_CHUNK_COLLECT_THRESHOLD; let resourceInfo = null; let resourceInfoId = -1; let reqData; let requestDetail; // refer to https://github.com/alibaba/anyproxy/issues/103 // construct the original headers as the reqheaders req.headers = util.getHeaderFromRawHeaders(req.rawHeaders); logUtil.printLog(color.green(`received request to: ${req.method} ${host}${path}`)); /** * fetch complete req data */ const fetchReqData = () => new Promise((resolve) => { const postData = []; req.on('data', (chunk) => { postData.push(chunk); }); req.on('end', () => { reqData = Buffer.concat(postData); resolve(); }); }); /** * prepare detailed request info */ const prepareRequestDetail = () => { const options = { hostname: urlPattern.hostname || req.headers.host, port: urlPattern.port || req.port || (/https/.test(protocol) ? 443 : 80), path, method: req.method, headers: req.headers }; requestDetail = { requestOptions: options, protocol, url: fullUrl, requestData: reqData, _req: req }; return Promise.resolve(); }; /** * send response to client * * @param {object} finalResponseData * @param {number} finalResponseData.statusCode * @param {object} finalResponseData.header * @param {buffer|string} finalResponseData.body */ const sendFinalResponse = (finalResponseData) => { const responseInfo = finalResponseData.response; const resHeader = responseInfo.header; const responseBody = responseInfo.body || ''; const transferEncoding = resHeader['transfer-encoding'] || resHeader['Transfer-Encoding'] || ''; const contentLength = resHeader['content-length'] || resHeader['Content-Length']; const connection = resHeader.Connection || resHeader.connection; if (contentLength) { delete resHeader['content-length']; delete resHeader['Content-Length']; } // set proxy-connection if (connection) { resHeader['x-anyproxy-origin-connection'] = connection; delete resHeader.connection; delete resHeader.Connection; } if (!responseInfo) { throw new Error('failed to get response info'); } else if (!responseInfo.statusCode) { throw new Error('failed to get response status code') } else if (!responseInfo.header) { throw new Error('filed to get response header'); } // if there is no transfer-encoding, set the content-length if (!global._throttle && transferEncoding !== 'chunked' && !(responseBody instanceof CommonReadableStream) ) { resHeader['Content-Length'] = util.getByteSize(responseBody); } userRes.writeHead(responseInfo.statusCode, resHeader); if (global._throttle) { if (responseBody instanceof CommonReadableStream) { responseBody.pipe(global._throttle.throttle()).pipe(userRes); } else { const thrStream = new Stream(); thrStream.pipe(global._throttle.throttle()).pipe(userRes); thrStream.emit('data', responseBody); thrStream.emit('end'); } } else { if (responseBody instanceof CommonReadableStream) { responseBody.pipe(userRes); } else { userRes.end(responseBody); } } return responseInfo; } // fetch complete request data co(fetchReqData) .then(prepareRequestDetail) .then(() => { // record request info if (recorder) { resourceInfo = { host, method: req.method, path, protocol, url: protocol + '://' + host + path, req, startTime: new Date().getTime() }; resourceInfoId = recorder.appendRecord(resourceInfo); } try { resourceInfo.reqBody = reqData.toString(); //TODO: deal reqBody in webInterface.js recorder && recorder.updateRecord(resourceInfoId, resourceInfo); } catch (e) { } }) // invoke rule before sending request .then(co.wrap(function *() { const userModifiedInfo = (yield userRule.beforeSendRequest(Object.assign({}, requestDetail))) || {}; const finalReqDetail = {}; ['protocol', 'requestOptions', 'requestData', 'response'].map((key) => { finalReqDetail[key] = userModifiedInfo[key] || requestDetail[key] }); return finalReqDetail; })) // route user config .then(co.wrap(function *(userConfig) { if (userConfig.response) { // user-assigned local response userConfig._directlyPassToRespond = true; return userConfig; } else if (userConfig.requestOptions) { const remoteResponse = yield fetchRemoteResponse(userConfig.protocol, userConfig.requestOptions, userConfig.requestData, { dangerouslyIgnoreUnauthorized: reqHandlerCtx.dangerouslyIgnoreUnauthorized, chunkSizeThreshold, }); return { response: { statusCode: remoteResponse.statusCode, header: remoteResponse.header, body: remoteResponse.body, rawBody: remoteResponse.rawBody }, _res: remoteResponse._res, }; } else { throw new Error('lost response or requestOptions, failed to continue'); } })) // invoke rule before responding to client .then(co.wrap(function *(responseData) { if (responseData._directlyPassToRespond) { return responseData; } else if (responseData.response.body && responseData.response.body instanceof CommonReadableStream) { // in stream mode return responseData; } else { // TODO: err etimeout return (yield userRule.beforeSendResponse(Object.assign({}, requestDetail), Object.assign({}, responseData))) || responseData; } })) .catch(co.wrap(function *(error) { logUtil.printLog(util.collectErrorLog(error), logUtil.T_ERR); let errorResponse = getErrorResponse(error, fullUrl); // call user rule try { const userResponse = yield userRule.onError(Object.assign({}, requestDetail), error); if (userResponse && userResponse.response && userResponse.response.header) { errorResponse = userResponse.response; } } catch (e) {} return { response: errorResponse }; })) .then(sendFinalResponse) //update record info .then((responseInfo) => { resourceInfo.endTime = new Date().getTime(); resourceInfo.res = { //construct a self-defined res object statusCode: responseInfo.statusCode, headers: responseInfo.header, }; resourceInfo.statusCode = responseInfo.statusCode; resourceInfo.resHeader = responseInfo.header; resourceInfo.resBody = responseInfo.body instanceof CommonReadableStream ? '(big stream)' : (responseInfo.body || ''); resourceInfo.length = resourceInfo.resBody.length; // console.info('===> resbody in record', resourceInfo); recorder && recorder.updateRecord(resourceInfoId, resourceInfo); }) .catch((e) => { logUtil.printLog(color.green('Send final response failed:' + e.message), logUtil.T_ERR); }); } } /** * get a handler for CONNECT request * * @param {RequestHandler} reqHandlerCtx * @param {object} userRule * @param {Recorder} recorder * @param {object} httpsServerMgr * @returns */ function getConnectReqHandler(userRule, recorder, httpsServerMgr) { const reqHandlerCtx = this; reqHandlerCtx.conns = new Map(); reqHandlerCtx.cltSockets = new Map() return function (req, cltSocket, head) { const host = req.url.split(':')[0], targetPort = req.url.split(':')[1]; let shouldIntercept; let interceptWsRequest = false; let requestDetail; let resourceInfo = null; let resourceInfoId = -1; const requestStream = new CommonReadableStream(); /* 1. write HTTP/1.1 200 to client 2. get request data 3. tell if it is a websocket request 4.1 if (websocket || do_not_intercept) --> pipe to target server 4.2 else --> pipe to local server and do man-in-the-middle attack */ co(function *() { // determine whether to use the man-in-the-middle server logUtil.printLog(color.green('received https CONNECT request ' + host)); requestDetail = { host: req.url, _req: req }; // the return value in default rule is null // so if the value is null, will take it as final value shouldIntercept = yield userRule.beforeDealHttpsRequest(requestDetail); // otherwise, will take the passed in option if (shouldIntercept === null) { shouldIntercept = reqHandlerCtx.forceProxyHttps; } }) .then(() => { return new Promise((resolve) => { // mark socket connection as established, to detect the request protocol cltSocket.write('HTTP/' + req.httpVersion + ' 200 OK\r\n\r\n', 'UTF-8', resolve); }); }) .then(() => { return new Promise((resolve, reject) => { let resolved = false; cltSocket.on('data', (chunk) => { requestStream.push(chunk); if (!resolved) { resolved = true; try { const chunkString = chunk.toString(); if (chunkString.indexOf('GET ') === 0) { shouldIntercept = false; // websocket, do not intercept // if there is '/do-not-proxy' in the request, do not intercept the websocket // to avoid AnyProxy itself be proxied if (reqHandlerCtx.wsIntercept && chunkString.indexOf('GET /do-not-proxy') !== 0) { interceptWsRequest = true; } } } catch (e) { console.error(e); } resolve(); } }); cltSocket.on('error', (error) => { logUtil.printLog(util.collectErrorLog(error), logUtil.T_ERR); co.wrap(function *() { try { yield userRule.onClientSocketError(requestDetail, error); } catch (e) { } }); }); cltSocket.on('end', () => { requestStream.push(null); }); }); }) .then((result) => { // log and recorder if (shouldIntercept) { logUtil.printLog('will forward to local https server'); } else { logUtil.printLog('will bypass the man-in-the-middle proxy'); } //record if (recorder) { resourceInfo = { host, method: req.method, path: '', url: 'https://' + host, req, startTime: new Date().getTime() }; resourceInfoId = recorder.appendRecord(resourceInfo); } }) .then(() => { // determine the request target if (!shouldIntercept) { // server info from the original request const originServer = { host, port: (targetPort === 80) ? 443 : targetPort } const localHttpServer = { host: 'localhost', port: reqHandlerCtx.httpServerPort } // for ws request, redirect them to local ws server return interceptWsRequest ? localHttpServer : originServer; } else { return httpsServerMgr.getSharedHttpsServer(host).then(serverInfo => ({ host: serverInfo.host, port: serverInfo.port })); } }) .then((serverInfo) => { if (!serverInfo.port || !serverInfo.host) { throw new Error('failed to get https server info'); } return new Promise((resolve, reject) => { const conn = net.connect(serverInfo.port, serverInfo.host, () => { //throttle for direct-foward https if (global._throttle && !shouldIntercept) { requestStream.pipe(conn); conn.pipe(global._throttle.throttle()).pipe(cltSocket); } else { requestStream.pipe(conn); conn.pipe(cltSocket); } resolve(); }); conn.on('error', (e) => { reject(e); }); reqHandlerCtx.conns.set(serverInfo.host + ':' + serverInfo.port, conn) reqHandlerCtx.cltSockets.set(serverInfo.host + ':' + serverInfo.port, cltSocket) }); }) .then(() => { if (recorder) { resourceInfo.endTime = new Date().getTime(); resourceInfo.statusCode = '200'; resourceInfo.resHeader = {}; resourceInfo.resBody = ''; resourceInfo.length = 0; recorder && recorder.updateRecord(resourceInfoId, resourceInfo); } }) .catch(co.wrap(function *(error) { logUtil.printLog(util.collectErrorLog(error), logUtil.T_ERR); try { yield userRule.onConnectError(requestDetail, error); } catch (e) { } try { let errorHeader = 'Proxy-Error: true\r\n'; errorHeader += 'Proxy-Error-Message: ' + (error || 'null') + '\r\n'; errorHeader += 'Content-Type: text/html\r\n'; cltSocket.write('HTTP/1.1 502\r\n' + errorHeader + '\r\n\r\n'); } catch (e) { } })); } } /** * get a websocket event handler @param @required {object} wsClient */ function getWsHandler(userRule, recorder, wsClient, wsReq) { const self = this; try { let resourceInfoId = -1; const resourceInfo = { wsMessages: [] // all ws messages go through AnyProxy }; const clientMsgQueue = []; const serverInfo = getWsReqInfo(wsReq); const serverInfoPort = serverInfo.port ? `:${serverInfo.port}` : ''; const wsUrl = `${serverInfo.protocol}://${serverInfo.hostName}${serverInfoPort}${serverInfo.path}`; const proxyWs = new WebSocket(wsUrl, '', { rejectUnauthorized: !self.dangerouslyIgnoreUnauthorized, headers: serverInfo.noWsHeaders }); if (recorder) { Object.assign(resourceInfo, { host: serverInfo.hostName, method: 'WebSocket', path: serverInfo.path, url: wsUrl, req: wsReq, startTime: new Date().getTime() }); resourceInfoId = recorder.appendRecord(resourceInfo); } /** * store the messages before the proxy ws is ready */ const sendProxyMessage = (event) => { const message = event.data; if (proxyWs.readyState === 1) { // if there still are msg queue consuming, keep it going if (clientMsgQueue.length > 0) { clientMsgQueue.push(message); } else { proxyWs.send(message); } } else { clientMsgQueue.push(message); } } /** * consume the message in queue when the proxy ws is not ready yet * will handle them from the first one-by-one */ const consumeMsgQueue = () => { while (clientMsgQueue.length > 0) { const message = clientMsgQueue.shift(); proxyWs.send(message); } } /** * When the source ws is closed, we need to close the target websocket. * If the source ws is normally closed, that is, the code is reserved, we need to transfrom them */ const getCloseFromOriginEvent = (event) => { const code = event.code || ''; const reason = event.reason || ''; let targetCode = ''; let targetReason = ''; if (code >= 1004 && code <= 1006) { targetCode = 1000; // normal closure targetReason = `Normally closed. The origin ws is closed at code: ${code} and reason: ${reason}`; } else { targetCode = code; targetReason = reason; } return { code: targetCode, reason: targetReason } } /** * consruct a message Record from message event * @param @required {event} messageEvent the event from websockt.onmessage * @param @required {boolean} isToServer whether the message is to or from server * */ const recordMessage = (messageEvent, isToServer) => { const message = { time: Date.now(), message: messageEvent.data, isToServer: isToServer }; // resourceInfo.wsMessages.push(message); recorder && recorder.updateRecordWsMessage(resourceInfoId, message); }; proxyWs.onopen = () => { consumeMsgQueue(); } // this event is fired when the connection is build and headers is returned proxyWs.on('upgrade', (response) => { resourceInfo.endTime = new Date().getTime(); const headers = response.headers; resourceInfo.res = { //construct a self-defined res object statusCode: response.statusCode, headers: headers, }; resourceInfo.statusCode = response.statusCode; resourceInfo.resHeader = headers; resourceInfo.resBody = ''; resourceInfo.length = resourceInfo.resBody.length; recorder && recorder.updateRecord(resourceInfoId, resourceInfo); }); proxyWs.onerror = (e) => { // https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes wsClient.close(1001, e.message); proxyWs.close(1001); } proxyWs.onmessage = (event) => { recordMessage(event, false); wsClient.readyState === 1 && wsClient.send(event.data); } proxyWs.onclose = (event) => { logUtil.debug(`proxy ws closed with code: ${event.code} and reason: ${event.reason}`); const targetCloseInfo = getCloseFromOriginEvent(event); wsClient.readyState !== 3 && wsClient.close(targetCloseInfo.code, targetCloseInfo.reason); } wsClient.onmessage = (event) => { recordMessage(event, true); sendProxyMessage(event); } wsClient.onclose = (event) => { logUtil.debug(`original ws closed with code: ${event.code} and reason: ${event.reason}`); const targetCloseInfo = getCloseFromOriginEvent(event); proxyWs.readyState !== 3 && proxyWs.close(targetCloseInfo.code, targetCloseInfo.reason); } } catch (e) { logUtil.debug('WebSocket Proxy Error:' + e.message); logUtil.debug(e.stack); console.error(e); } } class RequestHandler { /** * Creates an instance of RequestHandler. * * @param {object} config * @param {boolean} config.forceProxyHttps proxy all https requests * @param {boolean} config.dangerouslyIgnoreUnauthorized @param {number} config.httpServerPort the http port AnyProxy do the proxy * @param {object} rule * @param {Recorder} recorder * * @memberOf RequestHandler */ constructor(config, rule, recorder) { const reqHandlerCtx = this; this.forceProxyHttps = false; this.dangerouslyIgnoreUnauthorized = false; this.httpServerPort = ''; this.wsIntercept = false; if (config.forceProxyHttps) { this.forceProxyHttps = true; } if (config.dangerouslyIgnoreUnauthorized) { this.dangerouslyIgnoreUnauthorized = true; } if (config.wsIntercept) { this.wsIntercept = config.wsIntercept; } this.httpServerPort = config.httpServerPort; const default_rule = util.freshRequire('./rule_default'); const userRule = util.merge(default_rule, rule); reqHandlerCtx.userRequestHandler = getUserReqHandler.apply(reqHandlerCtx, [userRule, recorder]); reqHandlerCtx.wsHandler = getWsHandler.bind(this, userRule, recorder); reqHandlerCtx.httpsServerMgr = new HttpsServerMgr({ handler: reqHandlerCtx.userRequestHandler, wsHandler: reqHandlerCtx.wsHandler, // websocket hostname: '127.0.0.1', }); this.connectReqHandler = getConnectReqHandler.apply(reqHandlerCtx, [userRule, recorder, reqHandlerCtx.httpsServerMgr]); } } module.exports = RequestHandler; ================================================ FILE: lib/ruleLoader.js ================================================ 'use strict'; const proxyUtil = require('./util'); const path = require('path'); const fs = require('fs'); const request = require('request'); const cachePath = proxyUtil.getAnyProxyTmpPath(); /** * download a file and cache * * @param {any} url * @returns {string} cachePath */ function cacheRemoteFile(url) { return new Promise((resolve, reject) => { request(url, (error, response, body) => { if (error) { return reject(error); } else if (response.statusCode !== 200) { return reject(`failed to load with a status code ${response.statusCode}`); } else { const fileCreatedTime = proxyUtil.formatDate(new Date(), 'YYYY_MM_DD_hh_mm_ss'); const random = Math.ceil(Math.random() * 500); const fileName = `remote_rule_${fileCreatedTime}_r${random}.js`; const filePath = path.join(cachePath, fileName); fs.writeFileSync(filePath, body); resolve(filePath); } }); }); } /** * load a local npm module * * @param {any} filePath * @returns module */ function loadLocalPath(filePath) { return new Promise((resolve, reject) => { const ruleFilePath = path.resolve(process.cwd(), filePath); if (fs.existsSync(ruleFilePath)) { resolve(require(ruleFilePath)); } else { resolve(require(filePath)); } }); } /** * load a module from url or local path * * @param {any} urlOrPath * @returns module */ function requireModule(urlOrPath) { return new Promise((resolve, reject) => { if (/^http/i.test(urlOrPath)) { resolve(cacheRemoteFile(urlOrPath)); } else { resolve(urlOrPath); } }).then(localPath => loadLocalPath(localPath)); } module.exports = { cacheRemoteFile, loadLocalPath, requireModule, }; ================================================ FILE: lib/rule_default.js ================================================ 'use strict'; module.exports = { summary: 'the default rule for AnyProxy', /** * * * @param {object} requestDetail * @param {string} requestDetail.protocol * @param {object} requestDetail.requestOptions * @param {object} requestDetail.requestData * @param {object} requestDetail.response * @param {number} requestDetail.response.statusCode * @param {object} requestDetail.response.header * @param {buffer} requestDetail.response.body * @returns */ *beforeSendRequest(requestDetail) { return null; }, /** * * * @param {object} requestDetail * @param {object} responseDetail */ *beforeSendResponse(requestDetail, responseDetail) { return null; }, /** * default to return null * the user MUST return a boolean when they do implement the interface in rule * * @param {any} requestDetail * @returns */ *beforeDealHttpsRequest(requestDetail) { return null; }, /** * * * @param {any} requestDetail * @param {any} error * @returns */ *onError(requestDetail, error) { return null; }, /** * * * @param {any} requestDetail * @param {any} error * @returns */ *onConnectError(requestDetail, error) { return null; }, /** * * * @param {any} requestDetail * @param {any} error * @returns */ *onClientSocketError(requestDetail, error) { return null; }, }; ================================================ FILE: lib/systemProxyMgr.js ================================================ 'use strict' const child_process = require('child_process'); const networkTypes = ['Ethernet', 'Thunderbolt Ethernet', 'Wi-Fi']; function execSync(cmd) { let stdout, status = 0; try { stdout = child_process.execSync(cmd); } catch (err) { stdout = err.stdout; status = err.status; } return { stdout: stdout.toString(), status }; } /** * proxy for CentOs * ------------------------------------------------------------------------ * * file: ~/.bash_profile * * http_proxy=http://proxy_server_address:port * export no_proxy=localhost,127.0.0.1,192.168.0.34 * export http_proxy * ------------------------------------------------------------------------ */ /** * proxy for Ubuntu * ------------------------------------------------------------------------ * * file: /etc/environment * more info: http://askubuntu.com/questions/150210/how-do-i-set-systemwide-proxy-servers-in-xubuntu-lubuntu-or-ubuntu-studio * * http_proxy=http://proxy_server_address:port * export no_proxy=localhost,127.0.0.1,192.168.0.34 * export http_proxy * ------------------------------------------------------------------------ */ /** * ------------------------------------------------------------------------ * mac proxy manager * ------------------------------------------------------------------------ */ const macProxyManager = {}; macProxyManager.getNetworkType = () => { for (let i = 0; i < networkTypes.length; i++) { const type = networkTypes[i], result = execSync('networksetup -getwebproxy ' + type); if (result.status === 0) { macProxyManager.networkType = type; return type; } } throw new Error('Unknown network type'); }; macProxyManager.enableGlobalProxy = (ip, port, proxyType) => { if (!ip || !port) { console.log('failed to set global proxy server.\n ip and port are required.'); return; } proxyType = proxyType || 'http'; const networkType = macProxyManager.networkType || macProxyManager.getNetworkType(); return /^http$/i.test(proxyType) ? // set http proxy execSync( 'networksetup -setwebproxy ${networkType} ${ip} ${port} && networksetup -setproxybypassdomains ${networkType} 127.0.0.1 localhost' .replace(/\${networkType}/g, networkType) .replace('${ip}', ip) .replace('${port}', port)) : // set https proxy execSync('networksetup -setsecurewebproxy ${networkType} ${ip} ${port} && networksetup -setproxybypassdomains ${networkType} 127.0.0.1 localhost' .replace(/\${networkType}/g, networkType) .replace('${ip}', ip) .replace('${port}', port)); }; macProxyManager.disableGlobalProxy = (proxyType) => { proxyType = proxyType || 'http'; const networkType = macProxyManager.networkType || macProxyManager.getNetworkType(); return /^http$/i.test(proxyType) ? // set http proxy execSync( 'networksetup -setwebproxystate ${networkType} off' .replace('${networkType}', networkType)) : // set https proxy execSync( 'networksetup -setsecurewebproxystate ${networkType} off' .replace('${networkType}', networkType)); }; macProxyManager.getProxyState = () => { const networkType = macProxyManager.networkType || macProxyManager.getNetworkType(); const result = execSync('networksetup -getwebproxy ${networkType}'.replace('${networkType}', networkType)); return result; }; /** * ------------------------------------------------------------------------ * windows proxy manager * * netsh does not alter the settings for IE * ------------------------------------------------------------------------ */ const winProxyManager = {}; winProxyManager.enableGlobalProxy = (ip, port) => { if (!ip && !port) { console.log('failed to set global proxy server.\n ip and port are required.'); return; } return execSync( // set proxy 'reg add "HKCU\\Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings" /v ProxyServer /t REG_SZ /d ${ip}:${port} /f & ' .replace('${ip}', ip) .replace('${port}', port) + // enable proxy 'reg add "HKCU\\Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings" /v ProxyEnable /t REG_DWORD /d 1 /f'); }; winProxyManager.disableGlobalProxy = () => execSync('reg add "HKCU\\Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings" /v ProxyEnable /t REG_DWORD /d 0 /f'); winProxyManager.getProxyState = () => '' winProxyManager.getNetworkType = () => '' module.exports = /^win/.test(process.platform) ? winProxyManager : macProxyManager; ================================================ FILE: lib/util.js ================================================ 'use strict'; const fs = require('fs'), path = require('path'), mime = require('mime-types'), color = require('colorful'), child_process = require('child_process'), os = require('os'), Buffer = require('buffer').Buffer, logUtil = require('./log'); const networkInterfaces = os.networkInterfaces(); // {"Content-Encoding":"gzip"} --> {"content-encoding":"gzip"} module.exports.lower_keys = (obj) => { for (const key in obj) { const val = obj[key]; delete obj[key]; obj[key.toLowerCase()] = val; } return obj; }; module.exports.merge = function (baseObj, extendObj) { for (const key in extendObj) { baseObj[key] = extendObj[key]; } return baseObj; }; function getUserHome() { return process.env.HOME || process.env.USERPROFILE; } module.exports.getUserHome = getUserHome; function getAnyProxyHome() { const home = path.join(getUserHome(), '/.anyproxy/'); if (!fs.existsSync(home)) { fs.mkdirSync(home); } return home; } module.exports.getAnyProxyHome = getAnyProxyHome; module.exports.getAnyProxyPath = function (pathName) { const home = getAnyProxyHome(); const targetPath = path.join(home, pathName); if (!fs.existsSync(targetPath)) { fs.mkdirSync(targetPath); } return targetPath; } module.exports.getAnyProxyTmpPath = function () { const targetPath = path.join(os.tmpdir(), 'anyproxy', 'cache'); if (!fs.existsSync(targetPath)) { fs.mkdirSync(targetPath, { recursive: true }); } return targetPath; } module.exports.simpleRender = function (str, object, regexp) { return String(str).replace(regexp || (/\{\{([^{}]+)\}\}/g), (match, name) => { if (match.charAt(0) === '\\') { return match.slice(1); } return (object[name] != null) ? object[name] : ''; }); }; module.exports.filewalker = function (root, cb) { root = root || process.cwd(); const ret = { directory: [], file: [] }; fs.readdir(root, (err, list) => { if (list && list.length) { list.map((item) => { const fullPath = path.join(root, item), stat = fs.lstatSync(fullPath); if (stat.isFile()) { ret.file.push({ name: item, fullPath }); } else if (stat.isDirectory()) { ret.directory.push({ name: item, fullPath }); } }); } cb && cb.apply(null, [null, ret]); }); }; /* * 获取文件所对应的content-type以及content-length等信息 * 比如在useLocalResponse的时候会使用到 */ module.exports.contentType = function (filepath) { return mime.contentType(path.extname(filepath)); }; /* * 读取file的大小,以byte为单位 */ module.exports.contentLength = function (filepath) { try { const stat = fs.statSync(filepath); return stat.size; } catch (e) { logUtil.printLog(color.red('\nfailed to ready local file : ' + filepath)); logUtil.printLog(color.red(e)); return 0; } }; /* * remove the cache before requiring, the path SHOULD BE RELATIVE TO UTIL.JS */ module.exports.freshRequire = function (modulePath) { delete require.cache[require.resolve(modulePath)]; return require(modulePath); }; /* * format the date string * @param date Date or timestamp * @param formatter YYYYMMDDHHmmss */ module.exports.formatDate = function (date, formatter) { if (typeof date !== 'object') { date = new Date(date); } const transform = function (value) { return value < 10 ? '0' + value : value; }; return formatter.replace(/^YYYY|MM|DD|hh|mm|ss/g, (match) => { switch (match) { case 'YYYY': return transform(date.getFullYear()); case 'MM': return transform(date.getMonth() + 1); case 'mm': return transform(date.getMinutes()); case 'DD': return transform(date.getDate()); case 'hh': return transform(date.getHours()); case 'ss': return transform(date.getSeconds()); default: return '' } }); }; /** * get headers(Object) from rawHeaders(Array) * @param rawHeaders [key, value, key2, value2, ...] */ module.exports.getHeaderFromRawHeaders = function (rawHeaders) { const headerObj = {}; const _handleSetCookieHeader = function (key, value) { if (headerObj[key].constructor === Array) { headerObj[key].push(value); } else { headerObj[key] = [headerObj[key], value]; } }; if (!!rawHeaders) { for (let i = 0; i < rawHeaders.length; i += 2) { const key = rawHeaders[i]; let value = rawHeaders[i + 1]; if (typeof value === 'string') { value = value.replace(/\0+$/g, ''); // 去除 \u0000的null字符串 } if (!headerObj[key]) { headerObj[key] = value; } else { // headers with same fields could be combined with comma. Ref: https://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2 // set-cookie should NOT be combined. Ref: https://tools.ietf.org/html/rfc6265 if (key.toLowerCase() === 'set-cookie') { _handleSetCookieHeader(key, value); } else { headerObj[key] = headerObj[key] + ',' + value; } } } } return headerObj; }; module.exports.getAllIpAddress = function getAllIpAddress() { const allIp = []; Object.keys(networkInterfaces).map((nic) => { networkInterfaces[nic].filter((detail) => { if (detail.family.toLowerCase() === 'ipv4') { allIp.push(detail.address); } }); }); return allIp.length ? allIp : ['127.0.0.1']; }; function deleteFolderContentsRecursive(dirPath, ifClearFolderItself) { if (!dirPath.trim() || dirPath === '/') { throw new Error('can_not_delete_this_dir'); } if (fs.existsSync(dirPath)) { fs.readdirSync(dirPath).forEach((file) => { const curPath = path.join(dirPath, file); if (fs.lstatSync(curPath).isDirectory()) { deleteFolderContentsRecursive(curPath, true); } else { // delete all files fs.unlinkSync(curPath); } }); if (ifClearFolderItself) { try { // ref: https://github.com/shelljs/shelljs/issues/49 const start = Date.now(); while (true) { try { fs.rmdirSync(dirPath); break; } catch (er) { if (process.platform === 'win32' && (er.code === 'ENOTEMPTY' || er.code === 'EBUSY' || er.code === 'EPERM')) { // Retry on windows, sometimes it takes a little time before all the files in the directory are gone if (Date.now() - start > 1000) throw er; } else if (er.code === 'ENOENT') { break; } else { throw er; } } } } catch (e) { throw new Error('could not remove directory (code ' + e.code + '): ' + dirPath); } } } } module.exports.deleteFolderContentsRecursive = deleteFolderContentsRecursive; module.exports.getFreePort = function () { return new Promise((resolve, reject) => { const server = require('net').createServer(); server.unref(); server.on('error', reject); server.listen(0, () => { const port = server.address().port; server.close(() => { resolve(port); }); }); }); } module.exports.collectErrorLog = function (error) { if (error && error.code && error.toString()) { return error.toString(); } else { let result = [error, error.stack].join('\n'); try { const errorString = error.toString(); if (errorString.indexOf('You may only yield a function') >= 0) { result = 'Function is not yieldable. Did you forget to provide a generator or promise in rule file ? \nFAQ http://anyproxy.io/4.x/#faq'; } } catch (e) {} return result } } module.exports.isFunc = function (source) { return source && Object.tostring.call(source) === '[object Function]'; }; /** * @param {object} content * @returns the size of the content */ module.exports.getByteSize = function (content) { return Buffer.byteLength(content); }; /* * identify whether the */ module.exports.isIp = function (domain) { if (!domain) { return false; } const ipReg = /^\d+?\.\d+?\.\d+?\.\d+?$/; return ipReg.test(domain); }; module.exports.execScriptSync = function (cmd) { let stdout, status = 0; try { stdout = child_process.execSync(cmd); } catch (err) { stdout = err.stdout; status = err.status; } return { stdout: stdout.toString(), status }; }; module.exports.guideToHomePage = function () { logUtil.info('Refer to http://anyproxy.io for more detail'); }; ================================================ FILE: lib/webInterface.js ================================================ 'use strict'; const express = require('express'), url = require('url'), bodyParser = require('body-parser'), fs = require('fs'), path = require('path'), events = require('events'), qrCode = require('qrcode-npm'), util = require('./util'), certMgr = require('./certMgr'), wsServer = require('./wsServer'), juicer = require('juicer'), ip = require('ip'), compress = require('compression'), pug = require('pug'); const DEFAULT_WEB_PORT = 8002; // port for web interface const packageJson = require('../package.json'); const MAX_CONTENT_SIZE = 1024 * 2000; // 2000kb const certFileTypes = ['crt', 'cer', 'pem', 'der']; /** * * * @class webInterface * @extends {events.EventEmitter} */ class webInterface extends events.EventEmitter { /** * Creates an instance of webInterface. * * @param {object} config * @param {number} config.webPort * @param {object} recorder * * @memberOf webInterface */ constructor(config, recorder) { if (!recorder) { throw new Error('recorder is required for web interface'); } super(); const self = this; self.webPort = config.webPort || DEFAULT_WEB_PORT; self.recorder = recorder; self.config = config || {}; self.app = this.getServer(); self.server = null; self.wsServer = null; } /** * get the express server */ getServer() { const self = this; const recorder = self.recorder; const ipAddress = ip.address(), // userRule = proxyInstance.proxyRule, webBasePath = 'web'; let ruleSummary = ''; let customMenu = []; try { ruleSummary = ''; //userRule.summary(); customMenu = ''; // userRule._getCustomMenu(); } catch (e) { } const staticDir = path.join(__dirname, '../', webBasePath); const app = express(); app.use(compress()); //invoke gzip app.use((req, res, next) => { res.setHeader('note', 'THIS IS A REQUEST FROM ANYPROXY WEB INTERFACE'); return next(); }); app.use(bodyParser.json()); app.get('/latestLog', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); recorder.getRecords(null, 10000, (err, docs) => { if (err) { res.end(err.toString()); } else { res.json(docs); } }); }); app.get('/downloadBody', (req, res) => { const query = req.query; recorder.getDecodedBody(query.id, (err, result) => { if (err || !result || !result.content) { res.json({}); } else if (result.mime) { if (query.raw === 'true') { //TODO : cache query result res.type(result.mime).end(result.content); } else if (query.download === 'true') { res.setHeader('Content-disposition', `attachment; filename=${result.fileName}`); res.setHeader('Content-type', result.mime); res.end(result.content); } } else { res.json({ }); } }); }); app.get('/fetchBody', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); const query = req.query; if (query && query.id) { recorder.getDecodedBody(query.id, (err, result) => { // 返回下载信息 const _resDownload = function (isDownload) { isDownload = typeof isDownload === 'boolean' ? isDownload : true; res.json({ id: query.id, type: result.type, method: result.meethod, fileName: result.fileName, ref: `/downloadBody?id=${query.id}&download=${isDownload}&raw=${!isDownload}` }); }; // 返回内容 const _resContent = () => { if (util.getByteSize(result.content || '') > MAX_CONTENT_SIZE) { _resDownload(true); return; } res.json({ id: query.id, type: result.type, method: result.method, resBody: result.content }); }; if (err || !result) { res.json({}); } else if (result.statusCode === 200 && result.mime) { // deal with 'application/x-javascript' and 'application/javascript' if (/json|text|javascript/.test(result.mime)) { _resContent(); } else if (result.type === 'image') { _resDownload(false); } else { _resDownload(true); } } else { _resContent(); } }); } else { res.end(''); } }); app.get('/fetchReqBody', (req, res) => { const query = req.query; if (query && query.id) { recorder.getSingleRecord(query.id, (err, doc) => { if (err || !doc[0]) { console.error(err); res.end(''); return; } res.setHeader('Content-disposition', `attachment; filename=request_${query.id}_body.txt`); res.setHeader('Content-type', 'text/plain'); res.end(doc[0].reqBody); }); } else { res.end(''); } }); app.get('/fetchWsMessages', (req, res) => { const query = req.query; if (query && query.id) { recorder.getDecodedWsMessage(query.id, (err, messages) => { if (err) { console.error(err); res.json([]); return; } res.json(messages); }); } else { res.json([]); } }); app.get('/downloadCrt', (req, res) => { const pageFn = pug.compileFile(path.join(__dirname, '../resource/cert_download.pug')); res.end(pageFn({ ua: req.get('user-agent') })); }); app.get('/fetchCrtFile', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); const _crtFilePath = certMgr.getRootCAFilePath(); if (_crtFilePath) { const fileType = certFileTypes.indexOf(req.query.type) !== -1 ? req.query.type : 'crt'; res.setHeader('Content-Type', 'application/x-x509-ca-cert'); res.setHeader('Content-Disposition', `attachment; filename="rootCA.${fileType}"`); res.end(fs.readFileSync(_crtFilePath, { encoding: null })); } else { res.setHeader('Content-Type', 'text/html'); res.end('can not file rootCA ,plase use anyproxy --root to generate one'); } }); app.get('/api/getQrCode', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); const qr = qrCode.qrcode(4, 'M'); const targetUrl = req.protocol + '://' + req.get('host') + '/downloadCrt'; const isRootCAFileExists = certMgr.isRootCAFileExists(); qr.addData(targetUrl); qr.make(); res.json({ status: 'success', url: targetUrl, isRootCAFileExists, qrImgDom: qr.createImgTag(4) }); }); // response init data app.get('/api/getInitData', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); const rootCAExists = certMgr.isRootCAFileExists(); const rootDirPath = certMgr.getRootDirPath(); const interceptFlag = false; //proxyInstance.getInterceptFlag(); TODO const globalProxyFlag = false; // TODO: proxyInstance.getGlobalProxyFlag(); res.json({ status: 'success', rootCAExists, rootCADirPath: rootDirPath, currentInterceptFlag: interceptFlag, currentGlobalProxyFlag: globalProxyFlag, ruleSummary: ruleSummary || '', ipAddress: util.getAllIpAddress(), port: '', //proxyInstance.proxyPort, // TODO appVersion: packageJson.version }); }); app.post('/api/generateRootCA', (req, res) => { res.setHeader('Access-Control-Allow-Origin', '*'); const rootExists = certMgr.isRootCAFileExists(); if (!rootExists) { certMgr.generateRootCA(() => { res.json({ status: 'success', code: 'done' }); }); } else { res.json({ status: 'success', code: 'root_ca_exists' }); } }); app.use((req, res, next) => { const indexTpl = fs.readFileSync(path.join(staticDir, '/index.html'), { encoding: 'utf8' }), opt = { rule: ruleSummary || '', customMenu: customMenu || [], ipAddress: ipAddress || '127.0.0.1' }; if (url.parse(req.url).pathname === '/') { res.setHeader('Content-Type', 'text/html'); res.end(juicer(indexTpl, opt)); } else { next(); } }); app.use(express.static(staticDir)); return app; } start() { const self = this; return new Promise((resolve, reject) => { self.server = self.app.listen(self.webPort); self.wsServer = new wsServer({ server: self.server }, self.recorder); self.wsServer.start(); resolve(); }) } close() { const self = this; return new Promise((resolve, reject) => { self.server && self.server.close(); self.wsServer && self.wsServer.closeAll(); self.server = null; self.wsServer = null; self.proxyInstance = null; resolve(); }); } } module.exports = webInterface; ================================================ FILE: lib/wsServer.js ================================================ 'use strict'; //websocket server manager const WebSocketServer = require('ws').Server; const logUtil = require('./log'); function resToMsg(msg, recorder, cb) { let result = {}, jsonData; try { jsonData = JSON.parse(msg); } catch (e) { result = { type: 'error', error: 'failed to parse your request : ' + e.toString() }; cb && cb(result); return; } if (jsonData.reqRef) { result.reqRef = jsonData.reqRef; } if (jsonData.type === 'reqBody' && jsonData.id) { result.type = 'body'; recorder.getBody(jsonData.id, (err, data) => { if (err) { result.content = { id: null, body: null, error: err.toString() }; } else { result.content = { id: jsonData.id, body: data.toString() }; } cb && cb(result); }); } else { // more req handler here return null; } } //config.server class wsServer { constructor(config, recorder) { if (!recorder) { throw new Error('proxy recorder is required'); } else if (!config || !config.server) { throw new Error('config.server is required'); } const self = this; self.config = config; self.recorder = recorder; self.checkBroadcastFlagTimer = null; } start() { const self = this; const config = self.config; const recorder = self.recorder; return new Promise((resolve, reject) => { //web socket interface const wss = new WebSocketServer({ server: config.server, clientTracking: true, }); resolve(); // the queue of the messages to be delivered let messageQueue = []; // the flat to indicate wheter to broadcast the record let broadcastFlag = true; self.checkBroadcastFlagTimer = setInterval(() => { broadcastFlag = true; sendMultipleMessage(); }, 50); function sendMultipleMessage(data) { // if the flag goes to be true, and there are records to send if (broadcastFlag && messageQueue.length > 0) { wss && wss.broadcast({ type: 'updateMultiple', content: messageQueue }); messageQueue = []; broadcastFlag = false; } else { data && messageQueue.push(data); } } wss.broadcast = function (data) { if (typeof data === 'object') { try { data = JSON.stringify(data); } catch (e) { console.error('==> error when do broadcast ', e, data); } } for (const client of wss.clients) { try { client.send(data); } catch (e) { logUtil.printLog('websocket failed to send data, ' + e, logUtil.T_ERR); } } }; wss.on('connection', (ws) => { ws.on('message', (msg) => { resToMsg(msg, recorder, (res) => { res && ws.send(JSON.stringify(res)); }); }); ws.on('error', (e) => { console.error('error in ws:', e); }); }); wss.on('error', (e) => { logUtil.printLog('websocket error, ' + e, logUtil.T_ERR); }); wss.on('close', () => { }); recorder.on('update', (data) => { try { sendMultipleMessage(data); } catch (e) { console.log('ws error'); console.log(e); } }); recorder.on('updateLatestWsMsg', (data) => { try { // console.info('==> update latestMsg ', data); wss && wss.broadcast({ type: 'updateLatestWsMsg', content: data }); } catch (e) { logUtil.error(e.message); logUtil.error(e.stack); console.error(e); } }); self.wss = wss; }); } closeAll() { const self = this; if (self.checkBroadcastFlagTimer) { clearInterval(self.checkBroadcastFlagTimer); } return new Promise((resolve, reject) => { self.wss.close((e) => { if (e) { reject(e); } else { resolve(); } }); }); } } module.exports = wsServer; ================================================ FILE: lib/wsServerMgr.js ================================================ /** * manage the websocket server * */ const ws = require('ws'); const logUtil = require('./log.js'); const WsServer = ws.Server; /** * get a new websocket server based on the server * @param @required {object} config {string} config.server {handler} config.handler */ function getWsServer(config) { const wss = new WsServer({ server: config.server }); wss.on('connection', config.connHandler); wss.on('headers', (headers) => { headers.push('x-anyproxy-websocket:true'); }); wss.on('error', e => { logUtil.error(`error in websocket proxy: ${e.message},\r\n ${e.stack}`); console.error('error happened in proxy websocket:', e) }); wss.on('close', e => { console.error('==> closing the ws server'); }); return wss; } module.exports.getWsServer = getWsServer; ================================================ FILE: module_sample/core_reload.js ================================================ const AnyProxy = require('../proxy'); const exec = require('child_process').exec; const AnyProxyRecorder = require('../lib/recorder'); const WebInterfaceLite = require('../lib/webInterface'); /* ------------------------------- | ProxyServerA | ProxyServerB | ------------------------------- ---------------------------- | Common Recorder | -------(by events)------| WebInterfaceLite | ------------------------------- ---------------------------- */ const commonRecorder = new AnyProxyRecorder(); // web interface依赖recorder new WebInterfaceLite({ // common web interface webPort: 8002 }, commonRecorder); // proxy core只依赖recorder,与webServer无关 const optionsA = { port: 8001, recorder: commonRecorder, // use common recorder }; const optionsB = { port: 8005, recorder: commonRecorder, // use common recorder }; const proxyServerA = new AnyProxy.ProxyCore(optionsA); const proxyServerB = new AnyProxy.ProxyCore(optionsB); proxyServerA.start(); proxyServerB.start(); // after both ready setTimeout(() => { exec('curl http://www.qq.com --proxy http://127.0.0.1:8001'); exec('curl http://www.sina.com.cn --proxy http://127.0.0.1:8005'); }, 1000); // visit http://127.0.0.1 , there should be two records ================================================ FILE: module_sample/https_config.js ================================================ const AnyProxy = require('../proxy'); const exec = require('child_process').exec; if (!AnyProxy.utils.certMgr.ifRootCAFileExists()) { AnyProxy.utils.certMgr.generateRootCA((error, keyPath) => { // let users to trust this CA before using proxy if (!error) { const certDir = require('path').dirname(keyPath); console.log('The cert is generated at', certDir); const isWin = /^win/.test(process.platform); if (isWin) { exec('start .', { cwd: certDir }); } else { exec('open .', { cwd: certDir }); } } else { console.error('error when generating rootCA', error); } }); } else { // clear all the certificates // AnyProxy.utils.certMgr.clearCerts() } ================================================ FILE: module_sample/normal_use.js ================================================ const AnyProxy = require('../proxy'); const options = { type: 'http', port: 8001, rule: null, webInterface: { enable: true, webPort: 8002 }, throttle: 10000, forceProxyHttps: true, silent: false }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.on('ready', () => { console.log('ready'); // set as system proxy proxyServer.close().then(() => { const proxyServerB = new AnyProxy.ProxyServer(options); proxyServerB.start(); }); console.log('closed'); // setTimeout(() => { // }, 2000); // AnyProxy.utils.systemProxyMgr.enableGlobalProxy('127.0.0.1', '8001'); }); proxyServer.on('error', (e) => { console.log('proxy error'); console.log(e); }); process.on('SIGINT', () => { // AnyProxy.utils.systemProxyMgr.disableGlobalProxy(); proxyServer.close(); process.exit(); }); proxyServer.start(); // const WebSocketServer = require('ws').Server; // const wsServer = new WebSocketServer({ port: 8003 },function(){ // console.log('ready'); // try { // const serverB = new WebSocketServer({ port: 8003 }, function (e, result) { // console.log('---in B---'); // console.log(e); // console.log(result); // }); // } catch(e) { // console.log(e); // console.log('e'); // } // // wsServer.close(function (e, result) { // // console.log('in close'); // // console.log(e); // // console.log(result); // // }); // }); ================================================ FILE: module_sample/simple_use.js ================================================ const AnyProxy = require('../proxy'); const options = { port: 8001, webInterface: { enable: true } }; const proxyServer = new AnyProxy.ProxyServer(options); proxyServer.start(); ================================================ FILE: package.json ================================================ { "name": "anyproxy", "version": "4.1.3", "description": "A fully configurable HTTP/HTTPS proxy in Node.js", "main": "proxy.js", "bin": { "anyproxy-ca": "bin/anyproxy-ca", "anyproxy": "bin/anyproxy" }, "dependencies": { "async": "~0.9.0", "async-task-mgr": ">=1.1.0", "body-parser": "^1.13.1", "brotli": "^1.3.2", "classnames": "^2.2.5", "clipboard-js": "^0.3.3", "co": "^4.6.0", "colorful": "^2.1.0", "commander": "~2.11.0", "component-emitter": "^1.2.1", "compression": "^1.4.4", "es6-promise": "^3.3.1", "express": "^4.8.5", "fast-json-stringify": "^0.17.0", "iconv-lite": "^0.4.6", "inquirer": "^5.2.0", "ip": "^0.3.2", "juicer": "^0.6.6-stable", "mime-types": "2.1.11", "moment": "^2.15.1", "nedb": "^1.8.0", "node-easy-cert": "^1.0.0", "pug": "^2.0.0-beta6", "qrcode-npm": "0.0.3", "request": "^2.74.0", "stream-throttle": "^0.1.3", "svg-inline-react": "^1.0.2", "thunkify": "^2.1.2", "whatwg-fetch": "^1.0.0", "ws": "^5.1.0" }, "devDependencies": { "@babel/core": "^7.8.3", "@babel/preset-env": "^7.8.3", "antd": "^2.5.0", "autoprefixer": "^6.4.1", "babel-core": "^6.14.0", "babel-eslint": "^7.0.0", "babel-jest": "^24.9.0", "babel-loader": "^6.2.5", "babel-plugin-import": "^1.0.0", "babel-plugin-transform-runtime": "^6.15.0", "babel-polyfill": "^6.13.0", "babel-preset-es2015": "^6.13.2", "babel-preset-react": "^6.11.1", "babel-preset-stage-0": "^6.5.0", "babel-register": "^6.11.6", "babel-runtime": "^6.11.6", "css-loader": "^0.23.1", "eslint": ">=4.18.2", "eslint-config-airbnb": "^15.1.0", "eslint-plugin-import": "^2.7.0", "eslint-plugin-jsx-a11y": "^5.1.1", "eslint-plugin-react": "^7.4.0", "extract-text-webpack-plugin": "^3.0.2", "file-loader": "^0.9.0", "jest": "^24.9.0", "less": "^2.7.1", "less-loader": "^2.2.3", "node-simhash": "^0.1.0", "nodeunit": "^0.9.1", "phantom": "^4.0.0", "postcss-loader": "^0.13.0", "pre-commit": "^1.2.2", "react": "^15.3.1", "react-addons-perf": "^15.4.0", "react-dom": "^15.3.1", "react-json-tree": "^0.10.0", "react-redux": "^4.4.5", "react-tap-event-plugin": "^1.0.0", "redux": "^3.6.0", "redux-saga": "^0.11.1", "stream-equal": "0.1.8", "style-loader": "^0.13.1", "svg-inline-loader": "^0.7.1", "tunnel": "^0.0.6", "url-loader": "^0.5.7", "urllib": "^2.34.2", "webpack": "^3.10.0", "worker-loader": "^0.7.1" }, "scripts": { "prepublish": "npm run buildweb", "test": "npx jest", "lint": "eslint .", "testserver": "node test/server/startServer.js", "buildweb": "NODE_ENV=production webpack --config web/webpack.config.js --colors", "webserver": "NODE_ENV=test webpack --config web/webpack.config.js --colors --watch", "doc:serve": "node build_scripts/prebuild-doc.js && gitbook serve ./docs-src ./docs --log debug", "doc:build": "./build_scripts/build-doc-site.sh" }, "pre-commit": [ "lint" ], "repository": { "type": "git", "url": "https://github.com/alibaba/anyproxy" }, "author": "ottomao@gmail.com", "license": "Apache-2.0", "engines": { "node": ">=6.0.0" } } ================================================ FILE: proxy.js ================================================ 'use strict'; const http = require('http'), https = require('https'), async = require('async'), color = require('colorful'), certMgr = require('./lib/certMgr'), Recorder = require('./lib/recorder'), logUtil = require('./lib/log'), util = require('./lib/util'), events = require('events'), co = require('co'), WebInterface = require('./lib/webInterface'), wsServerMgr = require('./lib/wsServerMgr'), ThrottleGroup = require('stream-throttle').ThrottleGroup; const T_TYPE_HTTP = 'http', T_TYPE_HTTPS = 'https', DEFAULT_TYPE = T_TYPE_HTTP; const PROXY_STATUS_INIT = 'INIT'; const PROXY_STATUS_READY = 'READY'; const PROXY_STATUS_CLOSED = 'CLOSED'; /** * * @class ProxyCore * @extends {events.EventEmitter} */ class ProxyCore extends events.EventEmitter { /** * Creates an instance of ProxyCore. * * @param {object} config - configs * @param {number} config.port - port of the proxy server * @param {object} [config.rule=null] - rule module to use * @param {string} [config.type=http] - type of the proxy server, could be 'http' or 'https' * @param {strign} [config.hostname=localhost] - host name of the proxy server, required when this is an https proxy * @param {number} [config.throttle] - speed limit in kb/s * @param {boolean} [config.forceProxyHttps=false] - if proxy all https requests * @param {boolean} [config.silent=false] - if keep the console silent * @param {boolean} [config.dangerouslyIgnoreUnauthorized=false] - if ignore unauthorized server response * @param {object} [config.recorder] - recorder to use * @param {boolean} [config.wsIntercept] - whether intercept websocket * * @memberOf ProxyCore */ constructor(config) { super(); config = config || {}; this.status = PROXY_STATUS_INIT; this.proxyPort = config.port; this.proxyType = /https/i.test(config.type || DEFAULT_TYPE) ? T_TYPE_HTTPS : T_TYPE_HTTP; this.proxyHostName = config.hostname || 'localhost'; this.recorder = config.recorder; if (parseInt(process.versions.node.split('.')[0], 10) < 4) { throw new Error('node.js >= v4.x is required for anyproxy'); } else if (config.forceProxyHttps && !certMgr.ifRootCAFileExists()) { logUtil.printLog('You can run `anyproxy-ca` to generate one root CA and then re-run this command'); throw new Error('root CA not found. Please run `anyproxy-ca` to generate one first.'); } else if (this.proxyType === T_TYPE_HTTPS && !config.hostname) { throw new Error('hostname is required in https proxy'); } else if (!this.proxyPort) { throw new Error('proxy port is required'); } else if (!this.recorder) { throw new Error('recorder is required'); } else if (config.forceProxyHttps && config.rule && config.rule.beforeDealHttpsRequest) { logUtil.printLog('both "-i(--intercept)" and rule.beforeDealHttpsRequest are specified, the "-i" option will be ignored.', logUtil.T_WARN); config.forceProxyHttps = false; } this.httpProxyServer = null; this.requestHandler = null; // copy the rule to keep the original proxyRule independent this.proxyRule = config.rule || {}; if (config.silent) { logUtil.setPrintStatus(false); } if (config.throttle) { logUtil.printLog('throttle :' + config.throttle + 'kb/s'); const rate = parseInt(config.throttle, 10); if (rate < 1) { throw new Error('Invalid throttle rate value, should be positive integer'); } global._throttle = new ThrottleGroup({ rate: 1024 * rate }); // rate - byte/sec } // init recorder this.recorder = config.recorder; // init request handler const RequestHandler = util.freshRequire('./requestHandler'); this.requestHandler = new RequestHandler({ wsIntercept: config.wsIntercept, httpServerPort: config.port, // the http server port for http proxy forceProxyHttps: !!config.forceProxyHttps, dangerouslyIgnoreUnauthorized: !!config.dangerouslyIgnoreUnauthorized }, this.proxyRule, this.recorder); } /** * manage all created socket * for each new socket, we put them to a map; * if the socket is closed itself, we remove it from the map * when the `close` method is called, we'll close the sockes before the server closed * * @param {Socket} the http socket that is creating * @returns undefined * @memberOf ProxyCore */ handleExistConnections(socket) { const self = this; self.socketIndex++; const key = `socketIndex_${self.socketIndex}`; self.socketPool[key] = socket; // if the socket is closed already, removed it from pool socket.on('close', () => { delete self.socketPool[key]; }); } /** * start the proxy server * * @returns ProxyCore * * @memberOf ProxyCore */ start() { const self = this; self.socketIndex = 0; self.socketPool = {}; if (self.status !== PROXY_STATUS_INIT) { throw new Error('server status is not PROXY_STATUS_INIT, can not run start()'); } async.series( [ //creat proxy server function (callback) { if (self.proxyType === T_TYPE_HTTPS) { certMgr.getCertificate(self.proxyHostName, (err, keyContent, crtContent) => { if (err) { callback(err); } else { self.httpProxyServer = https.createServer({ key: keyContent, cert: crtContent }, self.requestHandler.userRequestHandler); callback(null); } }); } else { self.httpProxyServer = http.createServer(self.requestHandler.userRequestHandler); callback(null); } }, //handle CONNECT request for https over http function (callback) { self.httpProxyServer.on('connect', self.requestHandler.connectReqHandler); callback(null); }, function (callback) { wsServerMgr.getWsServer({ server: self.httpProxyServer, connHandler: self.requestHandler.wsHandler }); // remember all sockets, so we can destory them when call the method 'close'; self.httpProxyServer.on('connection', (socket) => { self.handleExistConnections.call(self, socket); }); callback(null); }, //start proxy server function (callback) { self.httpProxyServer.listen(self.proxyPort); callback(null); }, ], //final callback (err, result) => { if (!err) { const tipText = (self.proxyType === T_TYPE_HTTP ? 'Http' : 'Https') + ' proxy started on port ' + self.proxyPort; logUtil.printLog(color.green(tipText)); if (self.webServerInstance) { const webTip = 'web interface started on port ' + self.webServerInstance.webPort; logUtil.printLog(color.green(webTip)); } let ruleSummaryString = ''; const ruleSummary = this.proxyRule.summary; if (ruleSummary) { co(function *() { if (typeof ruleSummary === 'string') { ruleSummaryString = ruleSummary; } else { ruleSummaryString = yield ruleSummary(); } logUtil.printLog(color.green(`Active rule is: ${ruleSummaryString}`)); }); } self.status = PROXY_STATUS_READY; self.emit('ready'); } else { const tipText = 'err when start proxy server :('; logUtil.printLog(color.red(tipText), logUtil.T_ERR); logUtil.printLog(err, logUtil.T_ERR); self.emit('error', { error: err }); } } ); return self; } /** * close the proxy server * * @returns ProxyCore * * @memberOf ProxyCore */ close() { // clear recorder cache return new Promise((resolve) => { if (this.httpProxyServer) { // destroy conns & cltSockets when closing proxy server for (const connItem of this.requestHandler.conns) { const key = connItem[0]; const conn = connItem[1]; logUtil.printLog(`destorying https connection : ${key}`); conn.end(); } for (const cltSocketItem of this.requestHandler.cltSockets) { const key = cltSocketItem[0]; const cltSocket = cltSocketItem[1]; logUtil.printLog(`closing https cltSocket : ${key}`); cltSocket.end(); } if (this.requestHandler.httpsServerMgr) { this.requestHandler.httpsServerMgr.close(); } if (this.socketPool) { for (const key in this.socketPool) { this.socketPool[key].destroy(); } } this.httpProxyServer.close((error) => { if (error) { console.error(error); logUtil.printLog(`proxy server close FAILED : ${error.message}`, logUtil.T_ERR); } else { this.httpProxyServer = null; this.status = PROXY_STATUS_CLOSED; logUtil.printLog(`proxy server closed at ${this.proxyHostName}:${this.proxyPort}`); } resolve(error); }); } else { resolve(); } }) } } /** * start proxy server as well as recorder and webInterface */ class ProxyServer extends ProxyCore { /** * * @param {object} config - config * @param {object} [config.webInterface] - config of the web interface * @param {boolean} [config.webInterface.enable=false] - if web interface is enabled * @param {number} [config.webInterface.webPort=8002] - http port of the web interface */ constructor(config) { // prepare a recorder const recorder = new Recorder(); const configForCore = Object.assign({ recorder, }, config); super(configForCore); this.proxyWebinterfaceConfig = config.webInterface; this.recorder = recorder; this.webServerInstance = null; } start() { if (this.recorder) { this.recorder.setDbAutoCompact(); } // start web interface if neeeded if (this.proxyWebinterfaceConfig && this.proxyWebinterfaceConfig.enable) { this.webServerInstance = new WebInterface(this.proxyWebinterfaceConfig, this.recorder); // start web server this.webServerInstance.start() // start proxy core .then(() => { super.start(); }) .catch((e) => { this.emit('error', e); }); } else { super.start(); } } close() { const self = this; // release recorder if (self.recorder) { self.recorder.stopDbAutoCompact(); self.recorder.clear(); } self.recorder = null; // close ProxyCore return super.close() // release webInterface .then(() => { if (self.webServerInstance) { const tmpWebServer = self.webServerInstance; self.webServerInstance = null; logUtil.printLog('closing webInterface...'); return tmpWebServer.close(); } }); } } module.exports.ProxyCore = ProxyCore; module.exports.ProxyServer = ProxyServer; module.exports.ProxyRecorder = Recorder; module.exports.ProxyWebServer = WebInterface; module.exports.utils = { systemProxyMgr: require('./lib/systemProxyMgr'), certMgr, }; ================================================ FILE: resource/502.pug ================================================ doctype html html(lang="en") head title AnyProxy Inner Error style. body { color: #666; line-height: 1.5; font-size: 13px; font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Helvetica,PingFang SC,Hiragino Sans GB,Microsoft YaHei,SimSun,sans-serif; } body * { box-sizing: border-box; } .stackError { border-radius: 5px; padding: 20px; border: 1px solid #fdc; background-color: #ffeee6; color: #666; } .stackError li { list-style-type: none; } .infoItem { position: relative; overflow: hidden; border: 1px solid #d5f1fd; background-color: #eaf8fe; border-radius: 4px; margin-bottom: 5px; padding-left: 70px; } .infoItem .label { position: absolute; top: 0; left: 0; bottom: 0; display: flex; justify-content: flex-start; align-items: center; width: 70px; font-weight: 300; background-color: #76abc1; color: #fff; padding: 5px; } .infoItem .value { overflow:hidden; padding: 5px; } .tipItem .label { background-color: #ecf6fd; } .tip { color: #808080; } body h1 # AnyProxy Inner Error h3 Oops! Error happend when AnyProxy handle the request. p.tip This is an error occurred inside AnyProxy, not from your target website. .infoItem .label | Error: .value #{error} .infoItem .label | URL: .value #{url} if tipMessage .infoItem .label | TIP: .value!= tipMessage p ul.stackError each item in errorStack li= item ================================================ FILE: resource/cert_download.pug ================================================ doctype html html(lang="en") head title Download rootCA meta(name='viewport', content='initial-scale=1, maximum-scale=0.5, minimum-scale=1, user-scalable=no') style. body { color: #666; line-height: 1.5; font-size: 16px; font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Helvetica,PingFang SC,Hiragino Sans GB,Microsoft YaHei,SimSun,sans-serif; } body * { box-sizing: border-box; } .logo { font-size: 36px; margin-bottom: 40px; text-align: center; } .any { font-weight: 500; } .proxy { font-weight: 100; } .title { font-weight: bold; margin: 20px 0 6px; } .button { text-align: center; padding: 4px 15px 5px 15px; font-size: 14px; font-weight: 500; border-radius: 4px; height: 32px; margin-bottom: 10px; display: block; text-decoration: none; border-color: #108ee9; color: rgba(0, 0, 0, .65); background-color: #fff; border-style: solid; border-width: 1px; border-style: solid; border-color: #d9d9d9; } .primary { color: #fff; background-color: #108ee9; border-color: #108ee9; } .more { text-align: center; font-size: 14px; } .content { word-break: break-all; font-size: 14px; line-height: 1.2; margin-bottom: 10px; } body .logo span.any Any span.proxy Proxy .title Download: .content Select a CA file to download, the .crt file is commonly used. a(href="/fetchCrtFile?type=crt").button.primary rootCA.crt a(href="/fetchCrtFile?type=cer").button rootCA.cer .more More .buttons(style='display: none') a(href="/fetchCrtFile?type=pem").button rootCA.pem a(href="/fetchCrtFile?type=der").button rootCA.der .title User-Agent: .content #{ua} script(type='text/javascript'). window.document.querySelector('.more').addEventListener('click', function (e) { e.target.style.display = 'none'; window.document.querySelector('.buttons').style.display = 'block'; }); ================================================ FILE: resource/cert_error.pug ================================================ doctype html html(lang="en") head title Security Vulnerable style. body { color: #666; line-height: 1.5; font-size: 13px; font-family: -apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Helvetica,PingFang SC,Hiragino Sans GB,Microsoft YaHei,SimSun,sans-serif; } body * { box-sizing: border-box; } .container { max-width: 1200px; padding: 20px; padding-top: 150px; margin: 0 auto; } .title { font-size: 20px; margin-bottom: 20px; } .explain { font-size: 14px; font-weight: 200; color: #666; } .explainCode { color: #999; margin-bottom: 10px; } body .container div.title | #{title} div.explainCode | #{code} div.explain div!= explain ================================================ FILE: rule_sample/sample_modify_request_data.js ================================================ /* sample: modify the post data towards http://httpbin.org/post test: curl -H "Content-Type: text/plain" -X POST -d 'original post data' http://httpbin.org/post --proxy http://127.0.0.1:8001 expected response: { "data": "i-am-anyproxy-modified-post-data" } */ module.exports = { summary: 'Rule to modify request data', *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org/post') === 0) { return { requestData: 'i-am-anyproxy-modified-post-data' }; } }, }; ================================================ FILE: rule_sample/sample_modify_request_header.js ================================================ /* sample: modify the user-agent in requests toward httpbin.org test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newRequestOptions = requestDetail.requestOptions; newRequestOptions.headers['User-Agent'] = 'AnyProxy/0.0.0'; return { requestOptions: newRequestOptions }; } }, }; ================================================ FILE: rule_sample/sample_modify_request_path.js ================================================ /* sample: redirect all https://httpbin.org/user-agent requests to http://localhost:8008/index.html test: curl https://httpbin.org/user-agent --proxy http://127.0.0.1:8001 expected response: 'hello world' from 127.0.0.1:8001/index.html */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('https://httpbin.org/user-agent') === 0) { const newRequestOptions = requestDetail.requestOptions; requestDetail.protocol = 'http'; newRequestOptions.hostname = '127.0.0.1' newRequestOptions.port = '8008'; newRequestOptions.path = '/index.html'; newRequestOptions.method = 'GET'; return requestDetail; } }, *beforeDealHttpsRequest(requestDetail) { return true; } }; ================================================ FILE: rule_sample/sample_modify_request_protocol.js ================================================ /* sample: redirect all http requests of httpbin.org to https test: curl 'http://httpbin.org/get?show_env=1' --proxy http://127.0.0.1:8001 expected response: { "X-Forwarded-Protocol": "https" } */ module.exports = { *beforeSendRequest(requestDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newOption = requestDetail.requestOptions; newOption.port = 443; return { protocol: 'https', requestOptions: newOption }; } } }; ================================================ FILE: rule_sample/sample_modify_response_data.js ================================================ /* sample: modify response data of http://httpbin.org/user-agent test: curl 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: { "user-agent": "curl/7.43.0" } -- AnyProxy Hacked! -- */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url === 'http://httpbin.org/user-agent') { const newResponse = responseDetail.response; newResponse.body += '-- AnyProxy Hacked! --'; return new Promise((resolve, reject) => { setTimeout(() => { // delay the response for 5s resolve({ response: newResponse }); }, 5000); }); } }, }; ================================================ FILE: rule_sample/sample_modify_response_header.js ================================================ /* sample: modify response header of http://httpbin.org/user-agent test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: X-Proxy-By: AnyProxy */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org/user-agent') === 0) { const newResponse = responseDetail.response; newResponse.header['X-Proxy-By'] = 'AnyProxy'; return { response: newResponse }; } } }; ================================================ FILE: rule_sample/sample_modify_response_statuscode.js ================================================ /* sample: modify all status code of http://httpbin.org/ to 404 test: curl -I 'http://httpbin.org/user-agent' --proxy http://127.0.0.1:8001 expected response: HTTP/1.1 404 Not Found */ module.exports = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('http://httpbin.org') === 0) { const newResponse = responseDetail.response; newResponse.statusCode = 404; return { response: newResponse }; } } }; ================================================ FILE: rule_sample/sample_use_local_response.js ================================================ /* sample: intercept all requests toward httpbin.org, use a local response test: curl http://httpbin.org/user-agent --proxy http://127.0.0.1:8001 */ module.exports = { *beforeSendRequest(requestDetail) { const localResponse = { statusCode: 200, header: { 'Content-Type': 'application/json' }, body: '{"hello": "this is local response"}' }; if (requestDetail.url.indexOf('http://httpbin.org') === 0) { return { response: localResponse }; } }, }; ================================================ FILE: test/__snapshots__/basic.spec.js.snap ================================================ // Jest Snapshot v1, https://goo.gl/fbAQLP exports[`http - HTTP verbs DELETE: args 1`] = ` Object { "foo": "bar", } `; exports[`http - HTTP verbs DELETE: data 1`] = `""`; exports[`http - HTTP verbs GET: args 1`] = ` Object { "param": "param_value", } `; exports[`http - HTTP verbs GET: data 1`] = `undefined`; exports[`http - HTTP verbs PATCH: args 1`] = `Object {}`; exports[`http - HTTP verbs PATCH: data 1`] = `""`; exports[`http - HTTP verbs POST body and header: args 1`] = `Object {}`; exports[`http - HTTP verbs POST body and header: data 1`] = ` "1 " `; exports[`http - HTTP verbs PUT: args 1`] = `Object {}`; exports[`http - HTTP verbs PUT: data 1`] = ` "1 " `; exports[`https - HTTP verbs DELETE: args 1`] = ` Object { "foo": "bar", } `; exports[`https - HTTP verbs DELETE: data 1`] = `""`; exports[`https - HTTP verbs GET: args 1`] = ` Object { "param": "param_value", } `; exports[`https - HTTP verbs GET: data 1`] = `undefined`; exports[`https - HTTP verbs PATCH: args 1`] = `Object {}`; exports[`https - HTTP verbs PATCH: data 1`] = `""`; exports[`https - HTTP verbs POST body and header: args 1`] = `Object {}`; exports[`https - HTTP verbs POST body and header: data 1`] = ` "1 " `; exports[`https - HTTP verbs PUT: args 1`] = `Object {}`; exports[`https - HTTP verbs PUT: data 1`] = ` "1 " `; ================================================ FILE: test/basic.spec.js ================================================ process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0'; const path = require('path'); const fs = require('fs'); const urllib = require('urllib'); const request = require('request'); const { basicProxyRequest, proxyServerWithRule, } = require('./util.js'); const http = require('http'); const WebSocket = require('ws'); const tunnel = require('tunnel'); let proxyServer; let proxyPort; let proxyHost; let proxyWebInterfaceHost; beforeAll(async () => { jest.DEFAULT_TIMEOUT_INTERVAL = 20 * 1000; proxyServer = await proxyServerWithRule({}, {}); proxyPort = proxyServer.proxyPort; proxyHost = `http://localhost:${proxyPort}`; const proxyWebInterfacePort = proxyServer.webServerInstance.webPort; proxyWebInterfaceHost = `http://localhost:${proxyWebInterfacePort}`; }); afterAll(() => { return proxyServer && proxyServer.close(); }); function doProxyWebSocket(url, headers = {}) { let agent = new tunnel.httpOverHttp({ proxy: { hostname: 'localhost', port: proxyPort, } }) if (url.indexOf('wss') === 0) { agent = new tunnel.httpsOverHttp({ rejectUnauthorized: false, proxy: { hostname: 'localhost', port: proxyPort, } }) } const ws = new WebSocket(url, { agent, rejectUnauthorized: false, headers }); return ws; } ['http', 'https'].forEach(protocol => { describe(`${protocol} - HTTP verbs`, () => { const assertProxyRes = (result) => { const proxyRes = result.response; const body = JSON.parse(result.body); expect(proxyRes.statusCode).toBe(200); expect(body.args).toMatchSnapshot('args'); expect(body.data).toMatchSnapshot('data'); return body; }; it('GET', async () => { const url = `${protocol}://httpbin.org/get`; const getParam = { param: 'param_value' }; await basicProxyRequest(proxyHost, 'GET', url, {}, getParam).then(assertProxyRes); }); it('POST body and header', async () => { const url = `${protocol}://httpbin.org/post`; const payloadStream = fs.createReadStream(path.resolve(__dirname, './fixtures/upload.txt')); const postHeaders = { anyproxy_header: 'header_value', }; const body = await basicProxyRequest(proxyHost, 'POST', url, postHeaders, {}, payloadStream).then(assertProxyRes); expect(body.headers['Anyproxy-Header']).toBe(postHeaders.anyproxy_header); }); it('PUT', async () => { const url = `${protocol}://httpbin.org/put`; const payloadStream = fs.createReadStream(path.resolve(__dirname, './fixtures/upload.txt')); await basicProxyRequest(proxyHost, 'PUT', url, {}, undefined, payloadStream).then(assertProxyRes); }); it('DELETE', async () => { const url = `${protocol}://httpbin.org/delete`; const param = { foo: 'bar', }; await basicProxyRequest(proxyHost, 'DELETE', url, {}, param).then(assertProxyRes); }); it('PATCH', async () => { const url = `${protocol}://httpbin.org/patch`; await basicProxyRequest(proxyHost, 'PATCH', url).then(assertProxyRes); }); it('Websocket', async () => { const expectEcho = (ws) => { return new Promise((resolve, reject) => { const wsMsg = Buffer.alloc(100 * 1024, 'a').toString(); // 100kb ws.on('open', () => { ws.send(wsMsg); }); ws.on('message', (msg) => { expect(msg).toBe(wsMsg); ws.close(); resolve(); }); }); }; const wsUrl = `${protocol === 'https' ? 'wss' : 'ws'}://echo.websocket.org`; const ws = doProxyWebSocket(wsUrl, {}); await expectEcho(ws); }); }); }); describe('status code and headers', () => { [302, 404, 500].forEach(statusCode => { it(`GET ${statusCode}`, async () => { const status = statusCode; const url = `http://httpbin.org/status/${status}`; const result = await basicProxyRequest(proxyHost, 'GET', url, {}, {}); const proxyRes = result.response; expect(proxyRes.statusCode).toBe(statusCode); }); it(`PUT ${statusCode}`, async () => { const status = statusCode; const url = `http://httpbin.org/status/${status}`; const result = await basicProxyRequest(proxyHost, 'PUT', url, {}, {}); const proxyRes = result.response; expect(proxyRes.statusCode).toBe(statusCode); }); }); }); describe('response data formats', () => { ['brotli', 'deflate', 'gzip'].forEach(encoding => { it(`GET ${encoding}`, async () => { const url = `http://httpbin.org/${encoding}`; const result = await basicProxyRequest(proxyHost, 'GET', url); const headers = result.response.headers; const body = JSON.parse(result.body); expect(headers['content-encoding']).toBeUndefined(); // should be removed by anyproxy expect(body.brotli || body.deflated || body.gzipped).toBeTruthy(); }); }); }); describe('big files', () => { const BIG_FILE_SIZE = 100 * 1024 * 1024 - 1; // 100 mb const BUFFER_FILL = 'a'; let server; beforeAll(() => { server = http.createServer({}, (req, res) => { if (/download/.test(req.url)) { const bufferContent = Buffer.alloc(BIG_FILE_SIZE, BUFFER_FILL); res.write(bufferContent); res.end(); } else if (/upload/.test(req.url)) { let reqPayloadSize = 0; req.on('data', (data) => { const bufferLength = data.length; reqPayloadSize += bufferLength; const expectBufferContent = Buffer.alloc(bufferLength, BUFFER_FILL); if (!expectBufferContent.equals(data)) { res.statusCode = 500; res.write('content not match'); } }).on('end', () => { if (res.statusCode === 500 || reqPayloadSize !== BIG_FILE_SIZE) { res.statusCode = 500; } else { res.statusCode = 200; } res.end(); }); } }); server.listen(3000); }); afterAll((done) => { server && server.close(done); }); it('download big file', (done) => { let responseSizeCount = 0; request({ url: 'http://127.0.0.1:3000/download', proxy: proxyHost, }).on('data', (data) => { const bufferLength = data.length; responseSizeCount += bufferLength; const expectBufferContent = Buffer.alloc(bufferLength, BUFFER_FILL); if (!expectBufferContent.equals(data)) { return done(new Error('download content not match')); } }).on('end', () => { if (responseSizeCount !== BIG_FILE_SIZE) { return done(new Error('file size not match')); } done(); }); }, 120 * 1000); it('upload big file', (done) => { const bufferContent = Buffer.alloc(BIG_FILE_SIZE, BUFFER_FILL); const req = request({ url: 'http://127.0.0.1:3000/upload', method: 'POST', proxy: proxyHost, }, (err, response, body) => { if (err) { return done(err); } else if (response.statusCode !== 200) { return done(new Error('upload failed ' + body)); } done(); }); req.write(bufferContent); req.end(); }, 120 * 1000); }); describe('web interface', () => { it('should be available', async () => { await urllib.request(proxyWebInterfaceHost).then((result) => { expect(result.status).toBe(200); }); }); }); ================================================ FILE: test/fixtures/someRule.js ================================================ module.exports = { foo: 'bar', }; ================================================ FILE: test/fixtures/upload.txt ================================================ 1 ================================================ FILE: test/lib/httpsServerMgr.spec.js ================================================ const tls = require('tls'); const httpsServerMgr = require('../../lib/httpsServerMgr'); describe('httpsServerMgr', () => { let serverMgrInstance; beforeAll(async () => { serverMgrInstance = new httpsServerMgr({ hostname: '127.0.0.1', handler: (req, res) => { res.end('hello world'); }, wsHandler: () => { }, }); }); afterAll(async () => { await serverMgrInstance.close(); }); it('SNI server should work properly', async () => { const sniServerA = await serverMgrInstance.getSharedHttpsServer('a.anyproxy.io'); const sniServerB = await serverMgrInstance.getSharedHttpsServer('b.anyproxy.io'); expect(sniServerA).toEqual(sniServerB); // SNI - common server const connectHostname = 'some_new_host.anyproxy.io'; const connectOpt = { servername: connectHostname, // servername is required for sni server rejectUnauthorized: false, } await new Promise((resolve, reject) => { const socketToSNIServer = tls.connect(sniServerA.port, '127.0.0.1', connectOpt, (tlsSocket) => { // console.log('client to SNI server connected, ', socketToSNIServer.authorized ? 'authorized' : 'unauthorized'); const certSubject = socketToSNIServer.getPeerCertificate().subject; expect(certSubject.CN).toEqual(connectHostname); socketToSNIServer.end(); resolve(); }); socketToSNIServer.on('keylog', line => { console.log(line); }) }); }); it('IP server should work properly', async () => { const ipServerHost = '1.2.3.4'; const anotherSNIServer = await serverMgrInstance.getSharedHttpsServer('c.anyproxy.io'); const ipServerA = await serverMgrInstance.getSharedHttpsServer(ipServerHost); const ipServerB = await serverMgrInstance.getSharedHttpsServer('5.6.7.8'); expect(ipServerA).not.toEqual(ipServerB); expect(anotherSNIServer).not.toEqual(ipServerA); const connectOpt = { rejectUnauthorized: false, } await new Promise((resolve, reject) => { const socketToIpServer = tls.connect(ipServerA.port, '127.0.0.1', connectOpt, () => { const certSubject = socketToIpServer.getPeerCertificate().subject; expect(certSubject.CN).toEqual(ipServerHost); socketToIpServer.end(); resolve(); }); }); }); }); ================================================ FILE: test/lib/ruleLoader.spec.js ================================================ const ruleLoader = require('../../lib/ruleLoader'); const fs = require('fs'); const path = require('path'); const localModulePath = path.join(__dirname, '../fixtures/someRule.js'); describe('ruleLoader', () => { it('should successfully cache a remote file', async () => { await ruleLoader.cacheRemoteFile('https://cdn.bootcss.com/lodash.js/4.16.4/lodash.min.js') .then(filePath => { let content; if (filePath) { content = fs.readFileSync(filePath, { encoding: 'utf8' }); } expect(content && content.length > 100).toBe(true); }); }); it('should load a local module ../util/CommonUtil', async () => { await ruleLoader.loadLocalPath(localModulePath) .then(module => { expect(module.foo).not.toBeUndefined(); }); }); it('should smart load a remote module', done => { ruleLoader.requireModule('https://cdn.bootcss.com/lodash.js/4.16.4/lodash.min.js') .then(module => { expect(module.VERSION).toEqual('4.16.4'); done(); }) .catch(done.fail); }); it('should smart load a local module', done => { ruleLoader.requireModule(localModulePath) .then(module => { expect(module.foo).not.toBeUndefined(); done(); }) .catch(done.fail); }); }); ================================================ FILE: test/lib/util.spec.js ================================================ const util = require('../../lib/util'); describe('utils', () => { it('getFreePort', async () => { const count = 100; const tasks = []; for (let i = 1; i <= count; i++) { tasks.push(util.getFreePort()); } await Promise.all(tasks) .then((results) => { // ensure ports are unique const portMap = {}; results.map((portNumber) => { portMap[portNumber] = true; }); expect(Object.keys(portMap).length).toEqual(count); }); }); }); ================================================ FILE: test/rule/beforeDealHttpsRequest.spec.js ================================================ const fs = require('fs'); const path = require('path'); const { basicProxyRequest, proxyServerWithRule, } = require('../util.js'); const RULE_PAYLOAD = 'this is something in rule'; const rule = { *beforeSendRequest(requestDetail) { const requestOptions = requestDetail.requestOptions; return { requestOptions, requestData: RULE_PAYLOAD, }; }, *beforeDealHttpsRequest(requestDetail) { return requestDetail.host.indexOf('httpbin.org') >= 0; } }; describe('Rule beforeDealHttpsRequest', () => { let proxyServer; let proxyPort; let proxyHost; beforeAll(async () => { proxyServer = await proxyServerWithRule(rule); proxyPort = proxyServer.proxyPort; proxyHost = `http://localhost:${proxyPort}`; }); afterAll(() => { return proxyServer && proxyServer.close(); }); it('Should replace the https request body', async () => { const url = 'https://httpbin.org/put'; const payloadStream = fs.createReadStream(path.resolve(__dirname, '../fixtures/upload.txt')); const postHeaders = { anyproxy_header: 'header_value', }; await basicProxyRequest(proxyHost, 'PUT', url, postHeaders, {}, payloadStream).then((result) => { const proxyRes = result.response; const body = JSON.parse(result.body); expect(proxyRes.statusCode).toBe(200); expect(body.data).toEqual(RULE_PAYLOAD); expect(body.url.indexOf('/put')).toBeGreaterThan(0); }); }); }); ================================================ FILE: test/rule/beforeSendRequest.spec.js ================================================ const fs = require('fs'); const path = require('path'); const { basicProxyRequest, proxyServerWithRule, } = require('../util.js'); const RULE_PAYLOAD = 'this is something in rule'; const RULE_REPLACE_HEADER_KEY = 'rule_replace_header_key'; const RULE_REPLACE_HEADER_VALUE = 'rule_replace_header_value'; const rule = { *beforeSendRequest(requestDetail) { const reqUrl = requestDetail.url; if (reqUrl.indexOf('/post') >= 0) { const requestOptions = requestDetail.requestOptions; requestOptions.path = '/put'; requestOptions.method = 'PUT'; return { requestOptions, requestData: RULE_PAYLOAD, }; } else if (reqUrl.indexOf('/status/302') >= 0) { return { response: { statusCode: 404, header: { [RULE_REPLACE_HEADER_KEY]: RULE_REPLACE_HEADER_VALUE, 'content-type': 'plain/text', }, body: RULE_PAYLOAD } }; } else if (reqUrl.indexOf('/should_be_replaced') >= 0) { const requestOptions = requestDetail.requestOptions; requestOptions.hostname = 'httpbin.org'; requestOptions.path = '/status/302'; requestOptions.port = '443'; return { protocol: 'https', requestOptions, }; } } }; describe('Rule replaceRequestData', () => { let proxyServer; let proxyPort; let proxyHost; beforeAll(async () => { proxyServer = await proxyServerWithRule(rule); proxyPort = proxyServer.proxyPort; proxyHost = `http://localhost:${proxyPort}`; }); afterAll(() => { return proxyServer && proxyServer.close(); }); it('should replace the request data in proxy if the assertion is true', async () => { const url = 'http://httpbin.org/post'; const payloadStream = fs.createReadStream(path.resolve(__dirname, '../fixtures/upload.txt')); const postHeaders = { anyproxy_header: 'header_value', }; await basicProxyRequest(proxyHost, 'POST', url, postHeaders, {}, payloadStream).then((result) => { const proxyRes = result.response; const body = JSON.parse(result.body); expect(proxyRes.statusCode).toBe(200); expect(body.data).toEqual(RULE_PAYLOAD); expect(body.url.indexOf('/put')).toBeGreaterThan(0); }); }); it('should respond content specified in rule', async () => { const url = 'http://httpbin.org/status/302'; await basicProxyRequest(proxyHost, 'GET', url).then((result) => { const proxyRes = result.response; const body = result.body; expect(body).toBe(RULE_PAYLOAD); expect(proxyRes.statusCode).toBe(404); expect(proxyRes.headers[RULE_REPLACE_HEADER_KEY]).toBe(RULE_REPLACE_HEADER_VALUE); }); }); it('should replace protocol and url', async () => { const url = 'http://domain_not_exists.anyproxy.io/should_be_replaced'; await basicProxyRequest(proxyHost, 'GET', url).then((result) => { const proxyRes = result.response; expect(proxyRes.statusCode).toBe(302); }); }); }); ================================================ FILE: test/rule/beforeSendResponse.js ================================================ const { basicProxyRequest, proxyServerWithRule, } = require('../util.js'); const RULE_REPLACE_HEADER_KEY = 'rule_replace_header_key'; const RULE_REPLACE_HEADER_VALUE = 'rule_replace_header_value'; const RULE_REPLACE_BODY = 'RULE_REPLACE_BODY'; const rule = { *beforeSendResponse(requestDetail, responseDetail) { if (requestDetail.url.indexOf('/uuid') >= 0) { const newResponse = responseDetail.response; newResponse.header[RULE_REPLACE_HEADER_KEY] = RULE_REPLACE_HEADER_VALUE; newResponse.body = RULE_REPLACE_BODY; newResponse.statusCode = 502; return { response: newResponse, }; } }, }; describe('Rule replaceResponseData', () => { let proxyServer; let proxyPort; let proxyHost; beforeAll(async () => { proxyServer = await proxyServerWithRule(rule); proxyPort = proxyServer.proxyPort; proxyHost = `http://localhost:${proxyPort}`; }); afterAll(() => { return proxyServer && proxyServer.close(); }); it('Should replace the header and body', async () => { const url = 'http://httpbin.org/uuid'; await basicProxyRequest(proxyHost, 'GET', url).then((result) => { const proxyRes = result.response; const body = result.body; expect(proxyRes.statusCode).toBe(502); expect(proxyRes.headers[RULE_REPLACE_HEADER_KEY]).toBe(RULE_REPLACE_HEADER_VALUE); expect(body).toBe(RULE_REPLACE_BODY); }); }); }); ================================================ FILE: test/rule/onError.spec.js ================================================ const { basicProxyRequest, proxyServerWithRule, } = require('../util.js'); const jestMockErrorFn = jest.fn(); const jestMockConnectErrorFn = jest.fn(); const ERROR_PAGE_IN_RULE = 'this is my error page'; const rule = { onConnectError: jestMockConnectErrorFn, *onError(requestDetail, error) { jestMockErrorFn(requestDetail, error); return { response: { statusCode: '200', header: {}, body: ERROR_PAGE_IN_RULE, } }; }, *beforeDealHttpsRequest(requestDetail) { return requestDetail.host.indexOf('intercept') === 0; }, }; describe('Rule replaceResponseData', () => { let proxyServer; let proxyPort; let proxyHost; beforeAll(async () => { proxyServer = await proxyServerWithRule(rule); proxyPort = proxyServer.proxyPort; proxyHost = `http://localhost:${proxyPort}`; }); afterAll(() => { return proxyServer && proxyServer.close(); }); it('should get error', async () => { const url = 'https://intercept.anyproxy_not_exists.io/some_path'; const result = await basicProxyRequest(proxyHost, 'GET', url); const proxyRes = result.response; const body = result.body; expect(proxyRes.statusCode).toBe(200); expect(body).toBe(ERROR_PAGE_IN_RULE); expect(jestMockErrorFn.mock.calls.length).toBe(1); }); it('should get connec error', async () => { const url = 'https://anyproxy_not_exists.io/do_not_intercept'; let e; try { await basicProxyRequest(proxyHost, 'GET', url); } catch (err) { e = err; } expect(e).not.toBeUndefined(); expect(jestMockConnectErrorFn.mock.calls.length).toBe(1); }); }); ================================================ FILE: test/util.js ================================================ const request = require('request'); const assert = require('assert'); // TODO const { freshRequire, getFreePort } = require('../lib/util.js'); function basicProxyRequest(proxyHost, method, url, headers, qs, payload) { assert(method && url, 'method and url are required'); assert(proxyHost, 'proxyHost is required'); headers = Object.assign({ 'via-anyproxy': 'true', }, headers || {}); const requestOpt = { method, url, headers, followRedirect: false, rejectUnauthorized: false, qs, proxy: proxyHost, }; return new Promise((resolve, reject) => { const callback = (error, response, body) => { if (error) { reject(error); } else { resolve({ response, body, }); } }; if (payload) { payload.pipe(request(requestOpt, callback)); } else { request(requestOpt, callback); } }); } const DEFAULT_OPTIONS = { type: 'http', port: 8001, webInterface: false, wsIntercept: true, // throttle: 10000, // optional, speed limit in kb/s forceProxyHttps: true, // intercept https as well dangerouslyIgnoreUnauthorized: true, silent: false //optional, do not print anything into terminal. do not set it when you are still debugging. }; async function proxyServerWithRule(rule, overrideConfig) { const AnyProxy = freshRequire('../proxy.js'); const freeportA = await getFreePort(); const freeportB = await getFreePort(); const options = Object.assign(DEFAULT_OPTIONS, { port: freeportA, webInterface: { enable: true, webPort: freeportB, } }, overrideConfig || {}); options.rule = rule; return new Promise((resolve, reject) => { const instance = new AnyProxy.ProxyServer(options); instance.on('error', reject); instance.on('ready', () => { resolve(instance); }); instance.start(); }); } module.exports = { basicProxyRequest, proxyServerWithRule, }; ================================================ FILE: test/web/curlUtil.spec.js ================================================ const { curlify } = require('../../web/src/common/curlUtil'); describe('Test the curlify function', () => { it('request with headers', () => { const requestDetail = { method: 'POST', url: 'https://localhost:3001/test', reqHeader: { 'via-proxy': 'true', }, }; const result = 'curl \'https://localhost:3001/test\' -X POST -H \'via-proxy: true\''; expect(curlify(requestDetail)).toBe(result); }); it('request with JSON body', () => { const requestDetail = { method: 'POST', url: 'https://localhost:3001/test', reqHeader: { 'content-type': 'application/json; charset=utf-8', }, reqBody: '{"action":1,"method":"test"}', }; const result = `curl '${requestDetail.url}' -X POST -H 'content-type: application/json; charset=utf-8' -d '${requestDetail.reqBody}'`; expect(curlify(requestDetail)).toBe(result); }); it('accpet gzip encoding with compressed flag', () => { const requestDetail = { method: 'GET', url: 'https://localhost:3001/test', reqHeader: { Host: 'localhost', 'Accept-Encoding': 'gzip', }, }; const result = 'curl \'https://localhost:3001/test\' -H \'Host: localhost\' -H \'Accept-Encoding: gzip\' --compressed'; expect(curlify(requestDetail)).toBe(result); }); it('escape url character', () => { const requestDetail = { method: 'GET', url: 'https://localhost:3001/test?a[]=1', }; const result = 'curl \'https://localhost:3001/test?a\\[\\]=1\''; expect(curlify(requestDetail)).toBe(result); }); }); ================================================ FILE: test/web/webInterface.spec.js ================================================ const WebInterface = require('../../lib/webInterface.js'); const Recorder = require('../../lib/recorder'); const urllib = require('urllib'); describe('WebInterface server', () => { let webServer = null; const webHost = 'http://127.0.0.1:8002' beforeAll(async () => { const recorder = new Recorder(); webServer = new WebInterface({ webPort: 8002, }, recorder); await webServer.start(); }); afterAll(async () => { await webServer.close(); }); it('should response qrcode string in /getQrCode', async () => { const response = await urllib.request(`${webHost}/api/getQrCode`); const body = JSON.parse(response.res.data); expect(body.qrImgDom).toMatch('
      ================================================ FILE: web/postcss.config.js ================================================ module.exports = { plugins: [ require('autoprefixer') ] } ================================================ FILE: web/src/action/globalStatusAction.js ================================================ export const STOP_RECORDING = 'STOP_RECORDING'; export const RESUME_RECORDING = 'RESUME_RECORDING'; export const SHOW_FILTER = 'SHOW_FILTER'; export const HIDE_FILTER = 'HIDE_FILTER'; export const UPDATE_FILTER = 'UPDATE_FILTER'; export const SHOW_MAP_LOCAL = 'SHOW_MAP_LOCAL'; export const HIDE_MAP_LOCAL = 'HIDE_MAP_LOCAL'; export const FETCH_DIRECTORY = 'FETCH_DIRECTORY'; // fetch the directory export const UPDATE_LOCAL_DIRECTORY = 'UPDATE_LOCAL_DIRECTORY'; export const FETCH_MAPPED_CONFIG = 'FETCH_MAPPED_CONFIG'; export const UPDATE_LOCAL_MAPPED_CONFIG = 'UPDATE_LOCAL_MAPPED_CONFIG'; export const UPDATE_REMOTE_MAPPED_CONFIG = 'UPDATE_REMOTE_MAPPED_CONFIG'; export const UPDATE_ACTIVE_RECORD_ITEM = 'UPDATE_ACTIVE_RECORD_ITEM'; export const TOGGLE_REMOTE_INTERCEPT_HTTPS = 'TOGGLE_REMOTE_INTERCEPT_HTTPS'; export const UPDATE_LOCAL_INTERCEPT_HTTPS_FLAG = 'UPDATE_LOCAL_INTERCEPT_HTTPS_FLAG'; export const TOGGLE_REMORE_GLOBAL_PROXY_FLAG = 'TOGGLE_REMORE_GLOBAL_PROXY_FLAG'; export const UPDATE_LOCAL_GLOBAL_PROXY_FLAG = 'UPDATE_LOCAL_GLOBAL_PROXY_FLAG'; export const SHOW_ROOT_CA = 'SHOW_ROOT_CA'; export const HIDE_ROOT_CA = 'HIDE_ROOT_CA'; export const UPDATE_CAN_LOAD_MORE = 'UPDATE_CAN_LOAD_MORE'; export const INCREASE_DISPLAY_RECORD_LIST = 'INCREASE_DISPLAY_RECORD_LIST'; export const UPDATE_SHOULD_CLEAR_RECORD = 'UPDATE_SHOULD_CLEAR_RECORD'; export const UPDATE_APP_VERSION = 'UPDATE_APP_VERSION'; export const UPDATE_IS_ROOTCA_EXISTS = 'UPDATE_IS_ROOTCA_EXISTS'; // should we display the tip for new record export const UPDATE_SHOW_NEW_RECORD_TIP = 'UPDATE_SHOW_NEW_RECORD_TIP'; // update if currently loading the record from server export const UPDATE_FETCHING_RECORD_STATUS = 'UPDATE_FETCHING_RECORD_STATUS'; export function stopRecording() { return { type: STOP_RECORDING }; } export function resumeRecording() { return { type: RESUME_RECORDING }; } export function showFilter() { return { type: SHOW_FILTER }; } export function hideFilter() { return { type: HIDE_FILTER }; } export function updateFilter(filterStr) { return { type: UPDATE_FILTER, data: filterStr }; } export function showMapLocal() { return { type: SHOW_MAP_LOCAL }; } export function hideMapLocal() { return { type: HIDE_MAP_LOCAL }; } export function fetchDirectory(path) { return { type: FETCH_DIRECTORY, data: path }; } export function updateLocalDirectory(path, sub) { return { type: UPDATE_LOCAL_DIRECTORY, data: { path, sub } }; } export function fetchMappedConfig() { return { type: FETCH_MAPPED_CONFIG }; } export function updateLocalMappedConfig(config) { return { type: UPDATE_LOCAL_MAPPED_CONFIG, data: config }; } export function updateRemoteMappedConfig(config) { return { type: UPDATE_REMOTE_MAPPED_CONFIG, data: config }; } export function updateActiveRecordItem(id) { return { type: UPDATE_ACTIVE_RECORD_ITEM, data: id }; } export function updateLocalInterceptHttpsFlag(flag) { return { type: UPDATE_LOCAL_INTERCEPT_HTTPS_FLAG, data: flag }; } export function toggleRemoteInterceptHttpsFlag(flag) { return { type: TOGGLE_REMOTE_INTERCEPT_HTTPS, data: flag }; } export function toggleRemoteGlobalProxyFlag(flag) { return { type: TOGGLE_REMORE_GLOBAL_PROXY_FLAG, data: flag }; } export function updateLocalGlobalProxyFlag(flag) { return { type: UPDATE_LOCAL_GLOBAL_PROXY_FLAG, data: flag }; } export function showRootCA() { return { type: SHOW_ROOT_CA }; } export function hideRootCA() { return { type: HIDE_ROOT_CA }; } export function updateCanLoadMore(canLoadMore) { return { type: UPDATE_CAN_LOAD_MORE, data: canLoadMore }; } export function increaseDisplayRecordLimit(moreToAdd) { return { type: INCREASE_DISPLAY_RECORD_LIST, data: moreToAdd }; } export function updateShouldClearRecord(shouldClear) { return { type: UPDATE_SHOULD_CLEAR_RECORD, data: shouldClear }; } export function updateLocalAppVersion(version) { return { type: UPDATE_APP_VERSION, data: version }; } export function updateShowNewRecordTip(shouldShow) { return { type: UPDATE_SHOW_NEW_RECORD_TIP, data: shouldShow }; } export function updateIsRootCAExists(exists) { return { type: UPDATE_IS_ROOTCA_EXISTS, data: exists }; } export function updateFechingRecordStatus(isFetching) { return { type: UPDATE_FETCHING_RECORD_STATUS, data: isFetching } } ================================================ FILE: web/src/action/recordAction.js ================================================ export const FETCH_REQUEST_LOG = 'FETCH_REQUEST_LOG'; export const UPDATE_WHOLE_REQUEST = 'UPDATE_WHOLE_REQUEST'; export const UPDATE_SINGLE_RECORD = 'UPDATE_SINGLE_RECORD'; export const CLEAR_ALL_RECORD = 'CLEAR_ALL_RECORD'; export const CLEAR_ALL_LOCAL_RECORD = 'CLEAR_ALL_LOCAL_RECORD'; export const FETCH_RECORD_DETAIL = 'FETCH_RECORD_DETAIL'; export const SHOW_RECORD_DETAIL = 'SHOW_RECORD_DETAIL'; export const HIDE_RECORD_DETAIL = 'HIDE_RECORD_DETAIL'; export const UPDATE_MULTIPLE_RECORDS = 'UPDATE_MULTIPLE_RECORDS'; export function fetchRequestLog() { return { type: FETCH_REQUEST_LOG }; } export function updateWholeRequest(data) { return { type: UPDATE_WHOLE_REQUEST, data: data }; } export function updateRecord(record) { return { type: UPDATE_SINGLE_RECORD, data: record }; } export function clearAllRecord () { return { type: CLEAR_ALL_RECORD }; } export function clearAllLocalRecord () { return { type: CLEAR_ALL_LOCAL_RECORD }; } export function fetchRecordDetail (recordId) { return { type: FETCH_RECORD_DETAIL, data: recordId }; } export function showRecordDetail (record) { return { type: SHOW_RECORD_DETAIL, data: record }; } export function hideRecordDetail () { return { type: HIDE_RECORD_DETAIL }; } export function updateMultipleRecords (records) { return { type: UPDATE_MULTIPLE_RECORDS, data: records }; } ================================================ FILE: web/src/common/ApiUtil.js ================================================ /** * AJAX操作工具类 */ import PromiseUtil from './promiseUtil'; export function getJSON(url, data) { const d = PromiseUtil.defer(); fetch(url + serializeQuery(data)) .then((data) => { d.resolve(data.json()); }) .catch((error) => { console.error(error); d.reject(error); }); return d.promise; } export function postJSON(url, data) { const d = PromiseUtil.defer(); fetch(url, { method: 'POST', headers: { 'Accept': 'application/json', 'Content-Type': 'application/json' }, body: JSON.stringify(data) }) .then((data) => { d.resolve(data.json()); }) .catch((error) => { console.error(error); d.reject(error); }); return d.promise; } function serializeQuery (data = {}) { data['__t'] = Date.now();// disable the cache const queryArray = []; for (let key in data) { queryArray.push(`${key}=${data[key]}`); } const queryStr = queryArray.join('&'); return queryStr ? '?' + queryStr : ''; } export function isApiSuccess (response) { return response.status === 'success'; } const apiUtil = { getJSON, postJSON, isApiSuccess }; export default apiUtil; ================================================ FILE: web/src/common/Constant.js ================================================ /** * define all constant variables here */ module.exports.MenuKeyMap = { RECORD_FILTER: 'RECORD_FILTER', MAP_LOCAL: 'MAP_LOCAL', ROOT_CA: 'ROOT_CA' }; ================================================ FILE: web/src/common/WsUtil.js ================================================ /* * Utility for websocket * */ import { message } from 'antd'; /** * Initiate a ws connection. * The default path `do-not-proxy` means the ws do not need to be proxied. * This is very important for AnyProxy‘s own server, such as WEB UI, * and the websocket detail panel in it, to prevent a recursive proxy. * @param {wsPort} wsPort the port of websocket * @param {key} path the path of the ws url * */ export function initWs(wsPort = location.port, path = 'do-not-proxy') { if(!WebSocket){ throw (new Error('WebSocket is not supported on this browser')); } const wsClient = new WebSocket(`ws://${location.hostname}:${wsPort}/${path}`); wsClient.onerror = (error) => { console.error(error); message.error('error happened when setup websocket'); }; wsClient.onopen = (e) => { console.info('websocket opened: ', e); }; wsClient.onclose = (e) => { console.info('websocket closed: ', e); }; return wsClient; } export default { initWs: initWs }; ================================================ FILE: web/src/common/apiUtil.js ================================================ /** * AJAX操作工具类 */ import PromiseUtil from './promiseUtil'; export function getJSON(url, data) { const d = PromiseUtil.defer(); fetch(url + serializeQuery(data)) .then((data) => { d.resolve(data.json()); }) .catch((error) => { console.error(error); d.reject(error); }); return d.promise; } export function postJSON(url, data) { const d = PromiseUtil.defer(); fetch(url, { method: 'POST', headers: { 'Accept': 'application/json', 'Content-Type': 'application/json' }, body: JSON.stringify(data) }) .then((data) => { d.resolve(data.json()); }) .catch((error) => { console.error(error); d.reject(error); }); return d.promise; } function serializeQuery (data = {}) { data['__t'] = Date.now();// disable the cache const queryArray = []; for (let key in data) { queryArray.push(`${key}=${data[key]}`); } const queryStr = queryArray.join('&'); return queryStr ? '?' + queryStr : ''; } export function isApiSuccess (response) { return response.status === 'success'; } const apiUtil = { getJSON, postJSON, isApiSuccess }; export default apiUtil; ================================================ FILE: web/src/common/commonUtil.js ================================================ /* * 存放常用工具类 */ /* * 格式化日期 * @param date Date or timestamp * @param formatter yyyyMMddHHmmss */ export function formatDate(date, formatter) { if (typeof date !== 'object') { date = new Date(date); } const transform = function(value) { return value < 10 ? '0' + value : value; }; return formatter.replace(/^YYYY|MM|DD|hh|mm|ss|ms/g, function(match) { switch (match) { case 'YYYY': return transform(date.getFullYear()); case 'MM': return transform(date.getMonth() + 1); case 'mm': return transform(date.getMinutes()); case 'DD': return transform(date.getDate()); case 'hh': return transform(date.getHours()); case 'ss': return transform(date.getSeconds()); case 'ms': return transform(date.getMilliseconds()); } }); } export function selectText(element) { let range, selection; if (window.getSelection) { selection = window.getSelection(); range = document.createRange(); range.selectNodeContents(element); selection.removeAllRanges(); selection.addRange(range); } else if (document.body.createTextRange) { range = document.body.createTextRange(); range.moveToElementText(element); range.select(); } } export function getQueryParameter (name) { var results = new RegExp('[\?&]' + name + '=([^&#]*)').exec(window.location.href); if (results == null) { return ''; } else { return results[1] || ''; } } const CommonUtil = { formatDate, selectText, getQueryParameter }; export default CommonUtil; ================================================ FILE: web/src/common/constant.js ================================================ /** * define all constant variables here */ module.exports.MenuKeyMap = { RECORD_FILTER: 'RECORD_FILTER', MAP_LOCAL: 'MAP_LOCAL', ROOT_CA: 'ROOT_CA' }; ================================================ FILE: web/src/common/curlUtil.js ================================================ module.exports = { curlify(recordDetail) { const headers = { ...recordDetail.reqHeader }; const acceptEncoding = headers['Accept-Encoding'] || headers['accept-encoding']; // escape reserve character in url const url = recordDetail.url.replace(/([\[\]])/g, '\\$1'); const curlified = ['curl', `'${url}'`]; if (recordDetail.method.toUpperCase() !== 'GET') { curlified.push('-X', recordDetail.method); } Object.keys(headers).forEach((key) => { curlified.push('-H', `'${key}: ${headers[key]}'`); }); if (recordDetail.reqBody) { curlified.push('-d', `'${recordDetail.reqBody}'`); } if (/deflate|gzip/.test(acceptEncoding)) { curlified.push('--compressed'); } return curlified.join(' '); } }; ================================================ FILE: web/src/common/promiseUtil.js ================================================ /* * Promise的工具类 */ export function defer() { const d = {}; d.promise = new Promise((resolve, reject) => { d.resolve = resolve; d.reject = reject; }); return d; } export default { defer }; ================================================ FILE: web/src/common/wsUtil.js ================================================ /* * Utility for websocket * */ import { message } from 'antd'; /** * Initiate a ws connection. * The default path `do-not-proxy` means the ws do not need to be proxied. * This is very important for AnyProxy‘s own server, such as WEB UI, * and the websocket detail panel in it, to prevent a recursive proxy. * @param {wsPort} wsPort the port of websocket * @param {key} path the path of the ws url * */ export function initWs(wsPort = location.port, path = 'do-not-proxy') { if(!WebSocket){ throw (new Error('WebSocket is not supported on this browser')); } const wsClient = new WebSocket(`ws://${location.hostname}:${wsPort}/${path}`); wsClient.onerror = (error) => { console.error(error); message.error('error happened when setup websocket'); }; wsClient.onopen = (e) => { console.info('websocket opened: ', e); }; wsClient.onclose = (e) => { console.info('websocket closed: ', e); }; return wsClient; } export default { initWs: initWs }; ================================================ FILE: web/src/component/download-root-ca.jsx ================================================ /** * The panel to edit the filter * */ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import ClassBind from 'classnames/bind'; import { connect } from 'react-redux'; import { message, Button, Spin } from 'antd'; import ResizablePanel from 'component/resizable-panel'; import { hideRootCA, updateIsRootCAExists } from 'action/globalStatusAction'; import { MenuKeyMap } from 'common/constant'; import { getJSON, ajaxGet, postJSON } from 'common/apiUtil'; import Style from './download-root-ca.less'; import CommonStyle from '../style/common.less'; class DownloadRootCA extends React.Component { constructor () { super(); this.state = { loadingCAQr: false, generatingCA: false }; this.onClose = this.onClose.bind(this); this.getQrCodeContent = this.getQrCodeContent.bind(this); } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object } fetchData () { this.setState({ loadingCAQr: true }); getJSON('/api/getQrCode') .then((response) => { this.setState({ loadingCAQr: false, CAQrCodeImageDom: response.qrImgDom, isRootCAFileExists: response.isRootCAFileExists, url: response.url }); }) .catch((error) => { console.error(error); message.error(error.errorMsg || 'Failed to get the QR code of RootCA path.'); }); } onClose () { this.props.dispatch(hideRootCA()); } getQrCodeContent () { const imgDomContent = { __html: this.state.CAQrCodeImageDom }; const content = (
      Scan to download rootCA to your Phone
      ); const spin = ; return this.state.loadingCAQr ? spin : content; } getGenerateRootCADiv () { const doToggleRemoteIntercept = () => { postJSON('/api/generateRootCA') .then((result) => { this.setState({ generateRootCA: false, isRootCAFileExists: true }); this.props.dispatch(updateIsRootCAExists(true)); }) .catch((error) => { this.setState({ generatingCA: false }); message.error('生成根证书失败,请重试'); }); }; return (
      RootCA
      Your RootCA has not been generated yet, please click the button to generate before you download it. Please install and trust the generated RootCA.
      ); } getDownloadDiv () { return (
      RootCA
      {this.getQrCodeContent()}
      Or click the button to download.
      ); } componentDidMount () { this.fetchData(); } render() { const panelVisible = this.props.globalStatus.activeMenuKey === MenuKeyMap.ROOT_CA; return ( {this.props.globalStatus.isRootCAFileExists ? this.getDownloadDiv() : this.getGenerateRootCADiv()} ); } } function select (state) { return { globalStatus: state.globalStatus }; } export default connect(select)(DownloadRootCA); ================================================ FILE: web/src/component/download-root-ca.less ================================================ @import '../style/constant.less'; .wrapper { position: relative; height: 100%; width: 100%; padding: 10px 13px 15px; color: @tip-color; text-align: center; :global { .ant-btn { width: 100%; font-weight: 400; } } } .title { font-size: @middlepanel-font-size; text-align: left; font-weight: 200; color: @hint-color; margin-bottom: 12px; } .fullHeightWrapper { height: 100%; padding-bottom: 100px; margin-bottom: -100px; min-height: 500px; } .arCodeDivWrapper { margin-top: 63px; } .generateRootCaTip { padding-top: 100px; .strongColor { color: @default-color; padding-top: 10px; font-weight: 500; } span { display: block; } } .generateCAButton { margin-top: 50px; } .buttons { width: 100%; .tipSpan { margin-top: 18px; display: block; } } ================================================ FILE: web/src/component/header-menu.jsx ================================================ /* * 页面顶部菜单的组件 */ import React, { PropTypes } from 'react'; import ClassBind from 'classnames/bind'; import { connect } from 'react-redux'; import InlineSVG from 'svg-inline-react'; import { message, Modal, Popover, Button } from 'antd'; import { getQueryParameter } from 'common/commonUtil'; import { MenuKeyMap } from 'common/constant'; import { resumeRecording, stopRecording, updateLocalInterceptHttpsFlag, updateLocalGlobalProxyFlag, toggleRemoteInterceptHttpsFlag, toggleRemoteGlobalProxyFlag, updateShouldClearRecord, updateIsRootCAExists, showFilter, updateLocalAppVersion } from 'action/globalStatusAction'; const { RECORD_FILTER: RECORD_FILTER_MENU_KEY } = MenuKeyMap; import { getJSON } from 'common/apiUtil'; import Style from './header-menu.less'; const StyleBind = ClassBind.bind(Style); class HeaderMenu extends React.Component { constructor() { super(); this.state = { ruleSummary: '', inAppMode: getQueryParameter('in_app_mode'), runningDetailVisible: false }; this.stopRecording = this.stopRecording.bind(this); this.resumeRecording = this.resumeRecording.bind(this); this.clearAllRecord = this.clearAllRecord.bind(this); this.initEvent = this.initEvent.bind(this); this.fetchData = this.fetchData.bind(this); this.togglerHttpsIntercept = this.togglerHttpsIntercept.bind(this); this.showRunningInfo = this.showRunningInfo.bind(this); this.handleRuningInfoVisibleChange = this.handleRuningInfoVisibleChange.bind(this); this.toggleGlobalProxyFlag = this.toggleGlobalProxyFlag.bind(this); this.showFilter = this.showFilter.bind(this); } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object, resumeRefreshFunc: PropTypes.func } stopRecording() { this.props.dispatch(stopRecording()); } resumeRecording() { console.info('Resuming...'); this.props.dispatch(resumeRecording()); } clearAllRecord() { this.props.dispatch(updateShouldClearRecord(true)); this.props.resumeRefreshFunc && this.props.resumeRefreshFunc(); } handleRuningInfoVisibleChange(visible) { this.setState({ runningDetailVisible: visible }); } showRunningInfo() { this.setState({ runningDetailVisible: true }); } showFilter() { this.props.dispatch(showFilter()); } togglerHttpsIntercept() { const self = this; // if no rootCA exists, inform the user about trust the root if (!this.props.globalStatus.isRootCAFileExists) { Modal.info({ title: 'AnyProxy is about to generate the root CA for you', content: (
      Trust the root CA before AnyProxy can do HTTPS proxy for you. They will be located in {' ' + this.state.rootCADirPath}
      ), width: 500, onOk() { doToggleRemoteIntercept(); this.props.dispatch(updateIsRootCAExists(true)); } }); } else { doToggleRemoteIntercept(); } function doToggleRemoteIntercept() { const currentHttpsFlag = self.props.globalStatus.interceptHttpsFlag; self.props.dispatch(toggleRemoteInterceptHttpsFlag(!currentHttpsFlag)); } } toggleGlobalProxyFlag() { const currentGlobalProxyFlag = this.props.globalStatus.globalProxyFlag; this.props.dispatch(toggleRemoteGlobalProxyFlag(!currentGlobalProxyFlag)); } initEvent() { document.addEventListener('keyup', (e) => { if (e.keyCode === 88 && e.ctrlKey) { this.clearAllRecord(); } }); } fetchData() { getJSON('/api/getInitData') .then((response) => { this.setState({ ruleSummary: response.ruleSummary, rootCADirPath: response.rootCADirPath, ipAddress: response.ipAddress, port: response.port }); this.props.dispatch(updateLocalInterceptHttpsFlag(response.currentInterceptFlag)); this.props.dispatch(updateLocalGlobalProxyFlag(response.currentGlobalProxyFlag)); this.props.dispatch(updateLocalAppVersion(response.appVersion)); this.props.dispatch(updateIsRootCAExists(response.rootCAExists)); }) .catch((error) => { console.error(error); message.error('Failed to get rule summary'); }); } componentDidMount() { this.fetchData(); this.initEvent(); } render() { const { globalStatus } = this.props; const { activeMenuKey } = globalStatus; const { ipAddress, inAppMode } = this.state; const stopMenuStyle = StyleBind('menuItem', { disabled: globalStatus.recording !== true }); const resumeMenuStyle = StyleBind('menuItem', { disabled: globalStatus.recording === true }); const runningTipStyle = StyleBind('menuItem', 'rightMenuItem', { active: this.state.runningDetailVisible }); const showFilterMenuStyle = StyleBind('menuItem', { active: activeMenuKey === RECORD_FILTER_MENU_KEY }); const addressDivs = ipAddress ? ( this.state.ipAddress.map((singleIpAddress) => { return
      {singleIpAddress}
      ; })) : null; const runningInfoDiv = (
      • Active Rule: {this.state.ruleSummary}
      • Host Address: {addressDivs}
      • Listening on: {this.state.port}
      • Proxy Protocol: HTTP
      ); const stopRecordingMenu = (
      Stop
      ); const resumeRecordingMenu = (
      Resume
      ); const filterMenu = (
      Filter
      ); return (
      {globalStatus.recording ? stopRecordingMenu : resumeRecordingMenu} Clear {inAppMode ? filterMenu : null}
      Proxy Info
      ); } } function select(state) { return { globalStatus: state.globalStatus }; } export default connect(select)(HeaderMenu) ================================================ FILE: web/src/component/header-menu.less ================================================ @import '../style/constant.less'; @svg-default-color: #3A3A3A; .wrapper { width: 100%; } .menuList { overflow: hidden; } .menuItem { color: @default-color; font-size: @font-size-xs; float: left; cursor: pointer; padding: 0 5px; opacity: 0.87; min-width: 95px; padding: 0 25px; text-align: center; -webkit-app-region: no-drag; -webkit-user-select: text; &:focus { text-decoration: none; } i { display: block; } .playIcon, .stopIcon { svg { width: 18px; } } .eyeIcon { svg { width: 27px; } } .tipIcon { svg { width: 22px; } } svg { width: 23px; height: 21px; cursor: pointer; polyline { fill: @svg-default-color; stroke: @svg-default-color; } g { fill: @svg-default-color; opacity: 1; } g > use { fill: @svg-default-color; } } span { display: block; line-height: 30px; color: @top-menu-span-color; } &:hover { color: @primary-color; span { color: @primary-color; } svg { polyline { fill: @primary-color; stroke: @primary-color; } g { fill: @primary-color; } use { fill: @primary-color; } } } &.disabled { color: @tip-color; } &.active { color: @primary-color; span { color: @primary-color; } svg { polyline { fill: @primary-color; stroke: @primary-color; } // fill: @primary-color; // stroke: @primary-color; g { fill: @primary-color; } use { fill: @primary-color; } } } &.rightMenuItem { float: right; padding-right: 0; margin-right: 35px; } } .menuItemSpliter { display: block; float: left; width: 1px; background-color: @top-menu-spliter-color; height: 30px; margin: 5px 20px 0; } .ruleTip { color: @tip-color; line-height: 26px; i { padding-right: 5px; } } .modalInfo { li { font-size: @font-size-reg; overflow: hidden; strong { font-weight: normal; float: left; } span { display: block; overflow: hidden; padding-left: 10px; color: @tip-color; } } :global { .ant-modal-title { color: @default-color; font-weight: 400; } .ant-btn { font-weight: normal; } } } .runningInfoDivWrapper { line-height: 1.5; li { overflow: hidden; strong { float: left; } span { overflow: hidden; padding-left: 5px; display: block; } } :global { .ant-btn { margin-top: 10px; } .ant-popover { font-size: @font-size-reg; } .ant-popover-title { color: @default-color; font-weight: bold; padding-left: 20px; } .ant-popover-inner-content { padding-bottom: 25px; padding-left: 20px; } } } ================================================ FILE: web/src/component/json-viewer.jsx ================================================ /* * A copoment to display content in the a modal */ import React, { PropTypes } from 'react'; import { Menu } from 'antd'; import ReactDOM from 'react-dom'; import JSONTree from 'react-json-tree'; import Style from './json-viewer.less'; const PageIndexMap = { 'JSON_STRING': 'JSON_STRING', 'JSON_TREE': 'JSON_TREE' }; const theme = { scheme: 'google', author: 'seth wright (http://sethawright.com)', base00: '#1d1f21', base01: '#282a2e', base02: '#373b41', base03: '#969896', base04: '#b4b7b4', base05: '#c5c8c6', base06: '#e0e0e0', base07: '#ffffff', base08: '#CC342B', base09: '#F96A38', base0A: '#FBA922', base0B: '#198844', base0C: '#3971ED', base0D: '#3971ED', base0E: '#A36AC7', base0F: '#3971ED' }; class JsonViewer extends React.Component { constructor () { super(); this.state = { pageIndex: PageIndexMap.JSON_STRING }; this.getMenuDiv = this.getMenuDiv.bind(this); this.handleMenuClick = this.handleMenuClick.bind(this); } static propTypes = { data: PropTypes.string } handleMenuClick(e) { this.setState({ pageIndex: e.key, }); } getMenuDiv () { return ( Source Preview ); } render () { if (!this.props.data) { return null; } let jsonTreeDiv =
      {this.props.data}
      ; try { // In an invalid JSON string returned, handle the exception const jsonObj = JSON.parse(this.props.data); jsonTreeDiv = ; } catch (e) { console.warn('Failed to get JSON Tree:', e); } const jsonStringDiv =
      {this.props.data}
      ; return (
      {this.getMenuDiv()}
      {this.state.pageIndex === PageIndexMap.JSON_STRING ? jsonStringDiv : jsonTreeDiv}
      ); } } export default JsonViewer; ================================================ FILE: web/src/component/json-viewer.less ================================================ @import '../style/constant.less'; .wrapper { border: 1px solid #d9d9d9; } .contentDiv { padding: 20px 25px; background: #fff; } ================================================ FILE: web/src/component/left-menu.jsx ================================================ /* * A copoment to for left main menu */ import React, { PropTypes } from 'react'; import { connect } from 'react-redux'; import InlineSVG from 'svg-inline-react'; import { getQueryParameter } from 'common/commonUtil'; import Style from './left-menu.less'; import ClassBind from 'classnames/bind'; import { showFilter, showRootCA } from 'action/globalStatusAction'; import { MenuKeyMap } from 'common/constant'; const StyleBind = ClassBind.bind(Style); const { RECORD_FILTER: RECORD_FILTER_MENU_KEY, ROOT_CA: ROOT_CA_MENU_KEY } = MenuKeyMap; class LeftMenu extends React.Component { constructor() { super(); this.state = { inAppMode: getQueryParameter('in_app_mode') }; // this.showMapLocal = this.showMapLocal.bind(this); this.showFilter = this.showFilter.bind(this); this.showRootCA = this.showRootCA.bind(this); } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object } // showMapLocal() { // this.props.dispatch(showMapLocal()); // } showFilter() { this.props.dispatch(showFilter()); } showRootCA() { this.props.dispatch(showRootCA()); } render() { const { filterStr, activeMenuKey, recording } = this.props.globalStatus; const filterMenuStyle = StyleBind('menuItem', { working: filterStr.length > 0, active: activeMenuKey === RECORD_FILTER_MENU_KEY }); const rootCAMenuStyle = StyleBind('menuItem', { active: activeMenuKey === ROOT_CA_MENU_KEY }); const wrapperStyle = StyleBind('wrapper', { inApp: this.state.inAppMode }); const circleStyle = StyleBind('circles', { active: recording, stop: !recording }); return (
      Any Proxy
      AnyProxy.io
      Version {this.props.globalStatus.appVersion}
      ); } } function select(state) { return { globalStatus: state.globalStatus }; } export default connect(select)(LeftMenu); ================================================ FILE: web/src/component/left-menu.less ================================================ @import '../style/constant.less'; @keyframes lightBulb{ from { opacity: 1 } to { opacity: 0.1 } } @total-duration: 5s; @single-duration: 0.7s; .wrapper { position: relative; background-color: @left-menu-background-color; width: 100%; height: 100%; padding-top: 14px; &.inApp { padding-top: 20px; } } .systemTitleButton { overflow: hidden; padding-left: 10px; } .logo { text-align: center; margin-top: 7px; -webkit-app-region: drag; -webkit-user-select: none; .brand { -webkit-app-region: no-drag; font-size: @logo-font-size; font-weight: 100; color: #fff; letter-spacing: 1px; } .any { font-family: "PingFangSC-Semibold", "Helvetica Neue", Helvetica, "Hiragino Sans GB", "Microsoft YaHei", Arial, sans-serif !important; } .proxy { font-family: "PingFangSC-Ultralight", "Helvetica Neue", Helvetica, "Hiragino Sans GB", "Microsoft YaHei", Arial, sans-serif !important; } .circles { display: inline-block; span { display: block; float: left; width: 3px; height: 3px; background-color: #10A1FF; margin: 0 4.7px; border-radius: 50%; opacity: 0.2; } &.stop { .circle1, .circle7 { opacity: 0.2; } .circle2, .circle6 { opacity: 0.4; } .circle3, .circle5 { opacity: 0.8; } .circle4 { opacity: 1; } } &.active { span { animation: lightBulb @total-duration infinite; } .circle7 { animation-delay: (1*@single-duration); } .circle6 { animation-delay: (2*@single-duration); } .circle5 { animation-delay: (3*@single-duration); } .circle4 { animation-delay: (4*@single-duration); } .circle3 { animation-delay: (5*@single-duration); } .circle2 { animation-delay: (6*@single-duration); } .circle1 { animation-delay: (7*@single-duration); } } } } .menuList { overflow: hidden; margin-top: 50px; display: block; font-size: @left-menu-font-size; font-weight: 200; -webkit-app-region: no-drag; a:hover { color: #fff; } } .menuItem { color: @left-menu-color; float: left; width: 100%; line-height: 47px; border-left: 2px solid transparent; display: flex; align-items: center; .filterIcon, .downloadIcon { width: 51px; display: inline-block; text-align: center; svg { width: 15px; } } .retweetIcon { width: 51px; display: inline-block; text-align: center; svg { width: 23px; } } svg { width: 23px; height: auto; cursor: pointer; display: inline-block; } &:hover { background-color: #252630; color: #fff; svg { polyline { fill: #fff; stroke: #fff; } g { fill: #fff; opacity: 1; } use { fill: #fff; } } } &.active { border-color: @primary-color; color: #fff; background-color: #252630; svg { polyline { fill: #fff; stroke: #fff; } g { fill: #fff; opacity: 1; } use { fill: #fff; } } } &.working { // color: #fff; } i { width: 35px; } } .bottom { position: absolute; bottom: 0; left: 0; right: 0; height: 60px; font-size: 12px; padding-left: 16px; color: @left-menu-color; .bottomItem { display: inline-block; margin-top: 7px; line-height: 1.5; color: @left-menu-color; } .bottomBorder { overflow: hidden; display: block; position: relative; margin-top: -3.52px; } .bottomBorder1 { float: left; width: 16px; margin-right: 2px; border-top: 1px solid; } .bottomBorder2 { float: left; width: 29px; margin-right: 2px; border-top: 1px solid; } .bottomBorder3 { float: left; width: 16px; border-top: 1px solid; } } ================================================ FILE: web/src/component/map-local.jsx ================================================ /** * The panel map request to local * */ import React, { PropTypes } from 'react'; import { connect } from 'react-redux'; import { Tree, Form, Input, Button } from 'antd'; import ResizablePanel from 'component/resizable-panel'; import PromiseUtil from 'common/promiseUtil'; import { fetchDirectory, hideMapLocal, fetchMappedConfig, updateRemoteMappedConfig } from 'action/globalStatusAction'; import { MenuKeyMap } from 'common/constant'; import Style from './map-local.less'; import CommonStyle from '../style/common.less'; const TreeNode = Tree.TreeNode; const createForm = Form.create; const FormItem = Form.Item; class MapLocal extends React.Component { constructor () { super(); this.state = { selectedLocalPath: '' }; this.loadTreeNode = this.loadTreeNode.bind(this); this.onClose = this.onClose.bind(this); this.getFormDiv = this.getFormDiv.bind(this); this.onNodeSelect = this.onNodeSelect.bind(this); this.getMappedConfigDiv = this.getMappedConfigDiv.bind(this); this.loadMappedConfig = this.loadMappedConfig.bind(this); this.addMappedConfig = this.addMappedConfig.bind(this); this.removeMappedConfig = this.removeMappedConfig.bind(this); } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object, form: PropTypes.object } loadTreeNode(node) { const d = PromiseUtil.defer(); const key = node ? node.props.eventKey : ''; this.props.dispatch(fetchDirectory(key)); setTimeout(function() { d.resolve(); }, 500); return d.promise; } loadMappedConfig() { this.props.dispatch(fetchMappedConfig()); } addMappedConfig () { const config = this.props.globalStatus.mappedConfig.slice(); this.props.form.validateFieldsAndScroll((error, value) => { config.push({ keyword: value.keyword, local: value.local }); this.props.dispatch(updateRemoteMappedConfig(config)); }); } removeMappedConfig (index) { const config = this.props.globalStatus.mappedConfig.slice(); config.splice(index, 1); this.props.dispatch(updateRemoteMappedConfig(config)); } loopTreeNode(nodes) { const treeNodes = nodes.map((item) => { if (item.children) { return ( {this.loopTreeNode(item.children)} ); } else { return ; } }); return treeNodes; } onClose () { this.props.dispatch(hideMapLocal()); } onNodeSelect (selectedKeys, { selected, selectedNodes }) { const node = selectedNodes[0]; // Only a file will be mapped if (node && node.props.isLeaf) { this.setState({ selectedLocalPath: selectedKeys[0] }); } } getFormDiv () { const { getFieldDecorator, getFieldError } = this.props.form; const formItemLayout = { labelCol: { span: 6 }, wrapperCol: { span: 18 }, }; const keywordProps = getFieldDecorator('keyword', { initialValue: '', validate: [ { trigger: 'onBlur', rules: [ { type: 'string', whitespace: true, required: true, message: '请录入需要映射的url匹配' } ] } ] }); const localProps = getFieldDecorator('local', { initialValue: this.state.selectedLocalPath, validate: [ { trigger: 'onBlur', rules: [ { type: 'string', whitespace: true, required: true, message: '请输入本地文件路径' } ] } ] }); return (
      {keywordProps()} {localProps()}
      ); } getMappedConfigDiv () { const { mappedConfig } = this.props.globalStatus; const mappedLiDiv = mappedConfig.map((item, index) => { return (
    • {item.keyword} Remove
      {item.local}
    • ); }); return (
      Current Configuration
        {mappedLiDiv}
      ); } componentDidMount () { this.loadTreeNode(); this.loadMappedConfig(); } render() { const treeNodes = this.loopTreeNode(this.props.globalStatus.directory); const panelVisible = this.props.globalStatus.activeMenuKey === MenuKeyMap.MAP_LOCAL; return (
      Map Local
      {this.getMappedConfigDiv()}
      Add Local Map
      {this.getFormDiv()}
      {treeNodes}
      ); } } function select (state) { return { globalStatus: state.globalStatus }; } export default connect(select)(createForm()(MapLocal)); ================================================ FILE: web/src/component/map-local.less ================================================ @import '../style/constant.less'; @panel-width: 100%; .mapLocalWrapper { padding: 10px 15px 30px; } .title { font-size: @middlepanel-font-size; text-align: left; font-weight: 200; margin-bottom: 12px; color: @hint-color; } .treeWrapper { height: 310px; width: @panel-width; overflow: auto; border: 1px solid @light-border-color; border-radius: 4px; } .form { width: @panel-width; label { font-weight: 600; } } .mappedKeyDiv { position: relative; strong { display: block; width: 230px; word-wrap: break-word; } a { overflow: hidden; position: absolute; top: 1px; right: 8px; } } .mappedList { padding-left: 25px; padding-bottom: 15px; li { list-style-type: disc; } } .operations { margin-top: 10px; button { width: @panel-width; } } ================================================ FILE: web/src/component/modal-panel.jsx ================================================ /* * A copoment to display content in the a modal */ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import { Icon } from 'antd'; import Style from './modal-panel.less'; import ClassBind from 'classnames/bind'; const StyleBind = ClassBind.bind(Style); class ModalPanel extends React.Component { constructor () { super(); this.state = { dragBarLeft: '', contentLeft: '' }; this.onDragbarMoveUp = this.onDragbarMoveUp.bind(this); this.onDragbarMove = this.onDragbarMove.bind(this); this.onDragbarMoveDown = this.onDragbarMoveDown.bind(this); this.onClose = this.onClose.bind(this); this.doClose = this.doClose.bind(this); this.onKeyUp = this.onKeyUp.bind(this); this.addKeyEvent = this.addKeyEvent.bind(this); this.removeKeyEvent = this.removeKeyEvent.bind(this); } static propTypes = { children: PropTypes.element, onClose: PropTypes.func, visible: PropTypes.bool, hideBackModal: PropTypes.bool, left: PropTypes.string } onDragbarMove (event) { this.setState({ dragBarLeft: event.pageX }); } onKeyUp (e) { if (e.keyCode == 27) { this.doClose(); } } addKeyEvent () { document.addEventListener('keyup', this.onKeyUp); } removeKeyEvent () { document.removeEventListener('keyup', this.onKeyUp); } onDragbarMoveUp (event) { this.setState({ contentLeft: event.pageX }); document.removeEventListener('mousemove', this.onDragbarMove); document.removeEventListener('mouseup', this.onDragbarMoveUp); } onDragbarMoveDown (event) { document.addEventListener('mousemove', this.onDragbarMove); document.addEventListener('mouseup', this.onDragbarMoveUp); } onClose (event) { if (event.target === event.currentTarget) { this.props.onClose && this.props.onClose(); } } doClose () { this.props.onClose && this.props.onClose(); } render () { // will not remove the dom but hidden it, so the dom will not be relayouted let renderLeft = '100%'; if (!this.props.visible) { this.removeKeyEvent(); // return null; } else { const { dragBarLeft, contentLeft } = this.state; const propsLeft = this.props.left; renderLeft = dragBarLeft || propsLeft; this.addKeyEvent(); } // const dragBarStyle = dragBarLeft || propsLeft ? { 'left': dragBarLeft || propsLeft } : null; // const contentStyle = contentLeft || propsLeft ? { 'left': contentLeft || propsLeft } : null; const dragBarStyle = { 'left': renderLeft }; const modalStyle = { 'left': renderLeft }; // const modalStyle = this.props.hideBackModal ? contentStyle : { 'left': 0 }; return (
      {this.props.children}
      ); } } export default ModalPanel; ================================================ FILE: web/src/component/modal-panel.less ================================================ @import '../style/constant.less'; .wrapper { display: block; position: absolute; z-index: 1000; top: 0; bottom: 0; left: 60%; right: 0; background-color: @opacity-background-color; overflow: hidden; -webkit-box-shadow: -3px 0px 6px 0px rgba(128,128,128,0.56); -moz-box-shadow: -3px 0px 6px 0px rgba(128,128,128,0.56); box-shadow: -3px 0px 6px 0px rgba(128,128,128,0.56); will-change: left; } .relativeWrapper { position: relative; width: 100%; height: 100%; } .closeIcon { position: absolute; z-index: 2000; top: 0; right: 0; cursor: pointer; padding: 15px; i { font-size: @font-size-big; } &:hover { color: @primary-color; } } .relativeWrapper { width: 100%; height: 100%; position: relative; } .contentWrapper { color: @default-color; position: relative; width: 100%; height: 100%; overflow: hidden; } .dragBar { position: fixed; left: 60%; bottom: 0; top: 0; width: 5px; background-color: @hint-color; opacity: 0; z-index: 2000; cursor: col-resize; &:hover { opacity: 0.87; } } .content { background: #fff; overflow: auto; width: 100%; height: 100%; position: relative; } ================================================ FILE: web/src/component/record-detail.jsx ================================================ /** * The panel to display the detial of the record * */ import React, { PropTypes } from 'react'; import ClassBind from 'classnames/bind'; import { Menu, Spin } from 'antd'; import ModalPanel from 'component/modal-panel'; import RecordRequestDetail from 'component/record-request-detail'; import RecordResponseDetail from 'component/record-response-detail'; import RecordWsMessageDetail from 'component/record-ws-message-detail'; import { hideRecordDetail } from 'action/recordAction'; import Style from './record-detail.less'; const StyleBind = ClassBind.bind(Style); const PageIndexMap = { REQUEST_INDEX: 'REQUEST_INDEX', RESPONSE_INDEX: 'RESPONSE_INDEX', WEBSOCKET_INDEX: 'WEBSOCKET_INDEX' }; class RecordDetail extends React.Component { constructor() { super(); this.onClose = this.onClose.bind(this); this.state = { pageIndex: PageIndexMap.REQUEST_INDEX }; this.onMenuChange = this.onMenuChange.bind(this); } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object, requestRecord: PropTypes.object } onClose() { this.props.dispatch(hideRecordDetail()); } onMenuChange(e) { this.setState({ pageIndex: e.key, }); } hasWebSocket (recordDetail = {}) { return recordDetail && recordDetail.method && recordDetail.method.toLowerCase() === 'websocket'; } getRequestDiv(recordDetail) { return ; } getResponseDiv(recordDetail) { return ; } getWsMessageDiv(recordDetail) { return ; } getRecordContentDiv(recordDetail = {}, fetchingRecord) { const getMenuBody = () => { let menuBody = null; switch (this.state.pageIndex) { case PageIndexMap.REQUEST_INDEX: { menuBody = this.getRequestDiv(recordDetail); break; } case PageIndexMap.RESPONSE_INDEX: { menuBody = this.getResponseDiv(recordDetail); break; } case PageIndexMap.WEBSOCKET_INDEX: { menuBody = this.getWsMessageDiv(recordDetail); break; } default: { menuBody = this.getRequestDiv(recordDetail); break; } } return menuBody; } const websocketMenu = ( WebSocket ); return (
      Request Response {this.hasWebSocket(recordDetail) ? websocketMenu : null}
      {fetchingRecord ? this.getLoaingDiv() : getMenuBody()}
      ); } getLoaingDiv() { return (
      LOADING...
      ); } getRecordDetailDiv() { const { requestRecord, globalStatus } = this.props; const recordDetail = requestRecord.recordDetail; const fetchingRecord = globalStatus.fetchingRecord; if (!recordDetail && !fetchingRecord) { return null; } return this.getRecordContentDiv(recordDetail, fetchingRecord); } componentWillReceiveProps(nextProps) { const { requestRecord } = nextProps; const { pageIndex } = this.state; // if this is not websocket, reset the index to RESPONSE_INDEX if (!this.hasWebSocket(requestRecord.recordDetail) && pageIndex === PageIndexMap.WEBSOCKET_INDEX) { this.setState({ pageIndex: PageIndexMap.RESPONSE_INDEX }); } } render() { return ( {this.getRecordDetailDiv()} ); } } export default RecordDetail; ================================================ FILE: web/src/component/record-detail.less ================================================ @import '../style/constant.less'; .wrapper { padding: 5px 15px; height: 100%; word-wrap: break-word; } .loading { text-align: center; padding-top: 100px; .loadingText { margin-top: 15px; color: @primary-color; font-size: @font-size-big; } } .detailWrapper { position: relative; min-height: 100%; padding: 5px; } .section { padding: 10px 0; font-size: @font-size-xs; border-bottom: 1px solid @border-color-base; &.noBorder { border: none; } } .okStatus { color: @success-color; } .reqBody, .resBody { min-width: 200px; padding-left: 15px; } .imageBody { max-width: 100%; max-height: 400px; } .ulItem { padding-left: 15px; overflow: hidden; } .liItem { overflow: hidden; strong { float: left; // min-width: 125px; // text-align: right; opacity: 0.8; } span { display: block; overflow: hidden; opacity: 0.87; padding-left: 15px; } } .cookieWrapper { padding-top: 15px; :global { .ant-table-middle .ant-table-thead > tr > th, .ant-table-middle .ant-table-tbody > tr > td { padding: 5px 8px; background-color: transparent; } } } .noCookes { text-align: center; color: @tip-color; padding: 7px 0 8px; border-bottom: 1px solid @light-border-color; } .odd { background-color: @light-background-color; } .codeWrapper { overflow: auto; padding: 15px; background-color: @info-bkg-color; border: 1px solid @light-border-color; } ================================================ FILE: web/src/component/record-filter.jsx ================================================ /** * The panel to edit the filter * */ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import ClassBind from 'classnames/bind'; import { connect } from 'react-redux'; import { Input, Alert } from 'antd'; import ResizablePanel from 'component/resizable-panel'; import { hideFilter, updateFilter } from 'action/globalStatusAction'; import { MenuKeyMap } from 'common/constant'; import Style from './record-filter.less'; import CommonStyle from '../style/common.less'; class RecordFilter extends React.Component { constructor () { super(); this.onChange = this.onChange.bind(this); this.onClose = this.onClose.bind(this); this.filterTimeoutId = null; } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object } onChange (event) { this.props.dispatch(updateFilter(event.target.value)); } onClose () { this.props.dispatch(hideFilter()); } render() { const description = (
      • Multiple filters supported, write them in a single line.
      • Each line will be treaded as a Reg expression.
      • The result will be an 'OR' of the filters.
      • All the filters will be tested against the URL.
      ); const panelVisible = this.props.globalStatus.activeMenuKey === MenuKeyMap.RECORD_FILTER; return (
      Filter
      ); } } function select (state) { return { globalStatus: state.globalStatus }; } export default connect(select)(RecordFilter); ================================================ FILE: web/src/component/record-filter.less ================================================ @import '../style/constant.less'; .filterWrapper { padding: 10px 15px 15px; } .title { font-size: @middlepanel-font-size; text-align: left; font-weight: 200; color: @hint-color; margin-bottom: 12px; } .filterInput { } .filterTip { margin-top: 20px; } .tipList { color: @tip-color; li { list-style-type: decimal; padding-left: 15px; } } ================================================ FILE: web/src/component/record-list-diff-worker.jsx ================================================ /* * A webworker to identify whether the component need to be re-rendered */ const getFilterReg = function (filterStr) { let filterReg = null; if (filterStr) { let regFilterStr = filterStr .replace(/\r\n/g, '\n') .replace(/\n\n/g, '\n'); // remove the last /\n$/ in case an accidential br regFilterStr = regFilterStr.replace(/\n$/, ''); if (regFilterStr[0] === '/' && regFilterStr[regFilterStr.length - 1] === '/') { regFilterStr = regFilterStr.substring(1, regFilterStr.length - 2); } regFilterStr = regFilterStr.replace(/((.+)\n|(.+)$)/g, (matchStr, $1, $2) => { // if there is '\n' in the string if ($2) { return `(${$2})|`; } else { return `(${$1})`; } }); try { filterReg = new RegExp(regFilterStr); } catch (e) { console.error(e); } } return filterReg; }; self.addEventListener('message', (e) => { const data = JSON.parse(e.data); const { limit, currentData, nextData, filterStr } = data; const filterReg = getFilterReg(filterStr); const filterdRecords = []; const length = nextData.length; // mark if the component need to be refreshed let shouldUpdate = false; // filtered out the records for (let i = 0; i < length; i++) { const item = nextData[i]; if (filterReg && filterReg.test(item.url)) { filterdRecords.push(item); } if (!filterReg) { filterdRecords.push(item); } if (filterdRecords.length >= limit) { break; } } const newDataLength = filterdRecords.length; const currentDataLength = currentData.length; if (newDataLength !== currentDataLength) { shouldUpdate = true; } else { // only the two with same index and the `_render` === true then we'll need to render for (let i = 0; i < currentData.length; i++) { const item = currentData[i]; const targetItem = filterdRecords[i]; if (item.id !== targetItem.id || targetItem._render === true) { shouldUpdate = true; break; } } } self.postMessage(JSON.stringify({ shouldUpdate, data: filterdRecords })); }); ================================================ FILE: web/src/component/record-panel.jsx ================================================ /* * A copoment for the request log table */ import React, { PropTypes } from 'react'; import { Icon } from 'antd'; import RecordRow from 'component/record-row'; import Style from './record-panel.less'; import ClassBind from 'classnames/bind'; import { fetchRecordDetail } from 'action/recordAction'; const StyleBind = ClassBind.bind(Style); class RecordPanel extends React.Component { constructor() { super(); this.state = { }; this.wsClient = null; this.getRecordDetail = this.getRecordDetail.bind(this); this.onKeyUp = this.onKeyUp.bind(this); this.addKeyEvent = this.addKeyEvent.bind(this); } static propTypes = { dispatch: PropTypes.func, data: PropTypes.array, lastActiveRecordId: PropTypes.number, currentActiveRecordId: PropTypes.number, loadingNext: PropTypes.bool, loadingPrev: PropTypes.bool, stopRefresh: PropTypes.func } getRecordDetail(id) { this.props.dispatch(fetchRecordDetail(id)); this.props.stopRefresh(); } // get next detail with cursor, to go previous and next getNextDetail(cursor) { const currentId = this.props.currentActiveRecordId; this.props.dispatch(fetchRecordDetail(currentId + cursor)); } onKeyUp(e) { if (typeof this.props.currentActiveRecordId === 'number') { // up arrow if (e.keyCode === 38) { this.getNextDetail(-1); } // down arrow if (e.keyCode === 40) { this.getNextDetail(1); } } } addKeyEvent() { document.addEventListener('keyup', this.onKeyUp); } getTrs() { const trs = []; const { lastActiveRecordId, currentActiveRecordId } = this.props; const { data: recordList } = this.props; const length = recordList.length; for (let i = 0; i < length; i++) { // only display records less than max limit if (i >= this.state.maxAllowedRecords) { break; } const item = recordList[i]; const tableRowStyle = StyleBind('row', { lightBackgroundColor: item.id % 2 === 1, lightColor: item.statusCode === '', activeRow: currentActiveRecordId === item.id }); if (currentActiveRecordId === item.id || lastActiveRecordId === item.id) { item._render = true; } trs.push( ); } return trs; } getLoadingPreviousDiv() { if (!this.props.loadingPrev) { return null; } return ( 正在加载... ); //
      正在加载...
      ; } getLoadingNextDiv() { if (!this.props.loadingNext) { return null; } return ( 正在加载... ); } shouldComponentUpdate(nextProps) { const { currentActiveRecordId, loadingNext, loadingPrev } = this.props; const shouldUpdate = nextProps.data !== this.props.data || nextProps.loadingNext !== loadingNext || nextProps.loadingPrev !== loadingPrev || nextProps.currentActiveRecordId !== currentActiveRecordId; // console.info(nextProps.data.length, this.props.data.length, shouldUpdate, Date.now()); return shouldUpdate; } componentDidMount() { this.addKeyEvent(); } render() { if (!this.props.data) { return null; } return (
      {this.getLoadingPreviousDiv()} {this.getTrs()}
      # Method Code Host Path Mime Start
      ); } } export default RecordPanel; ================================================ FILE: web/src/component/record-panel.less ================================================ @import '../style/constant.less'; .wrapper { position: relative; -webkit-user-select: none; :global { .ant-table { border: none; border-radius: 0; border-bottom: 0; padding-bottom: 50px; .ant-table-body { height: calc(100% - 114px); overflow: auto; } } th { padding-top: 15px !important; padding-bottom: 12px !important; border: none !important; border: none !important; } } .firstRow { padding-left: 20px; text-align: center; } .leftRow { text-align: left; } .centerRow { text-align: center; } .pathRow { min-width: 300px; } } .row { cursor: pointer; font-size: @font-size-xs; td { padding-top: 5px !important; padding-bottom: 5px !important; border: none; } } .lightBackgroundColor { background: @light-background-color; } .lightColor { color: @tip-color; } .activeRow { background: @active-color !important; color: #fff; :global { td { background-color: transparent !important; } } } .okStatus { color: @ok-color; } tr.loading { text-align: center; color: @tip-color; i { margin-right: 5px; } } ================================================ FILE: web/src/component/record-request-detail.jsx ================================================ /** * The panel to display the detial of the record * */ import React, { PropTypes } from 'react'; import ClassBind from 'classnames/bind'; import { Menu, Table, notification, Spin } from 'antd'; import clipboard from 'clipboard-js' import JsonViewer from 'component/json-viewer'; import ModalPanel from 'component/modal-panel'; import { hideRecordDetail } from 'action/recordAction'; import { selectText } from 'common/commonUtil'; import { curlify } from 'common/curlUtil'; import Style from './record-detail.less'; import CommonStyle from '../style/common.less'; const StyleBind = ClassBind.bind(Style); const PageIndexMap = { REQUEST_INDEX: 'REQUEST_INDEX', RESPONSE_INDEX: 'RESPONSE_INDEX' }; // the maximum length of the request body to decide whether to offer a download link for the request body const MAXIMUM_REQ_BODY_LENGTH = 10000; class RecordRequestDetail extends React.Component { constructor() { super(); this.state = { }; this.copyCurlCmd = this.copyCurlCmd.bind(this); } static propTypes = { requestRecord: PropTypes.object } onSelectText(e) { selectText(e.target); } getLiDivs(targetObj) { const liDom = Object.keys(targetObj).map((key) => { return (
    • {key} : {targetObj[key]}
    • ); }); return liDom; } getCookieDiv(cookies) { let cookieArray = []; if (cookies) { const cookieStringArray = cookies.split(';'); cookieArray = cookieStringArray.map((cookieString) => { const cookie = cookieString.split('='); return { name: cookie[0], value: cookie.slice(1).join('=') // cookie的值本身可能含有"=", 此处进行修正 }; }); } else { return
      No Cookies
      ; } const columns = [ { title: 'Name', dataIndex: 'name', width: 300 }, { title: 'Value', dataIndex: 'value' } ]; const rowClassFunc = function (record, index) { // return index % 2 === 0 ? null : Style.odd; return null; }; const locale = { emptyText: 'No Cookies' }; return (
      ); } getReqBodyDiv() { const { recordDetail } = this.props; const requestBody = recordDetail.reqBody; const reqDownload = download; const getReqBodyContent = () => { const bodyLength = requestBody.length; if (bodyLength > MAXIMUM_REQ_BODY_LENGTH) { return reqDownload; } else { return
      {requestBody}
      } } return (
      {getReqBodyContent()}
      ); } notify(message, type = 'info', duration = 1.6, opts = {}) { notification[type]({ message, duration, ...opts }) } copyCurlCmd() { const recordDetail = this.props.recordDetail clipboard .copy(curlify(recordDetail)) .then(() => this.notify('COPY SUCCESS', 'success')) .catch(() => this.notify('COPY FAILED', 'error')) } getRequestDiv() { const recordDetail = this.props.recordDetail; const reqHeader = Object.assign({}, recordDetail.reqHeader); const cookieString = reqHeader.cookie || reqHeader.Cookie; delete reqHeader.cookie; // cookie will be displayed seperately const { protocol, host, path } = recordDetail; return (
      General
      • Method: {recordDetail.method}
      • URL: {`${protocol}://${host}${path}`}
      • Protocol: HTTP/1.1
      Header
        {this.getLiDivs(reqHeader)}
      Cookies
      {this.getCookieDiv(cookieString)}
      Body
      {this.getReqBodyDiv()}
      ); } render() { return this.getRequestDiv(); } } export default RecordRequestDetail; ================================================ FILE: web/src/component/record-response-detail.jsx ================================================ /** * The panel to display the response detial of the record * */ import React, { PropTypes } from 'react'; import ClassBind from 'classnames/bind'; import { Menu, Table, notification, Spin } from 'antd'; import JsonViewer from 'component/json-viewer'; import ModalPanel from 'component/modal-panel'; import Style from './record-detail.less'; import CommonStyle from '../style/common.less'; const StyleBind = ClassBind.bind(Style); const PageIndexMap = { REQUEST_INDEX: 'REQUEST_INDEX', RESPONSE_INDEX: 'RESPONSE_INDEX' }; // the maximum length of the request body to decide whether to offer a download link for the request body const MAXIMUM_REQ_BODY_LENGTH = 10000; class RecordResponseDetail extends React.Component { constructor() { super(); this.state = { }; } static propTypes = { recordDetail: PropTypes.object } onSelectText(e) { selectText(e.target); } getLiDivs(targetObj) { const liDom = Object.keys(targetObj).map((key) => { return (
    • {key} : {targetObj[key]}
    • ); }); return liDom; } getImageBody(recordDetail) { return ; } getJsonBody(recordDetail) { return ; } getResBodyDiv() { const { recordDetail } = this.props; const self = this; let reqBodyDiv =
      {recordDetail.resBody} 
      ; switch (recordDetail.type) { case 'image': { reqBodyDiv =
      {self.getImageBody(recordDetail)}
      ; break; } case 'json': { reqBodyDiv = self.getJsonBody(recordDetail); break; } default: { if (!recordDetail.resBody && recordDetail.ref) { reqBodyDiv = {recordDetail.fileName}; } break; } } return (
      {reqBodyDiv}
      ); } getResponseDiv(recordDetail) { const statusStyle = StyleBind({ okStatus: recordDetail.statusCode === 200 }); return (
      General
      • Status Code: {recordDetail.statusCode}
      Header
        {this.getLiDivs(recordDetail.resHeader)}
      Body
      {this.getResBodyDiv()}
      ); } render() { return this.getResponseDiv(this.props.recordDetail); } } export default RecordResponseDetail; ================================================ FILE: web/src/component/record-row.jsx ================================================ /* * A copoment for the request log table */ import React, { PropTypes } from 'react'; import { formatDate } from 'common/commonUtil'; import Style from './record-row.less'; import CommonStyle from '../style/common.less'; import ClassBind from 'classnames/bind'; const StyleBind = ClassBind.bind(Style); class RecordRow extends React.Component { constructor() { super(); this.state = { }; } static propTypes = { data: PropTypes.object, detailHanlder: PropTypes.func, className: PropTypes.string } getMethodDiv(item) { const httpsIcon =
      ; return
      {item.method}
      {item.protocol === 'https' ? httpsIcon : null}
      ; } getCodeDiv(item) { const statusCode = parseInt(item.statusCode, 10); const className = StyleBind({ okStatus: statusCode === 200, errorStatus: statusCode >= 400 }); return {item.statusCode}; } shouldComponentUpdate(nextProps) { if (nextProps.data._render) { nextProps.data._render = false; return true; } else { return false; } } render() { const data = this.props.data; if (!data) { return null; } return (
      ); } } export default RecordRow; ================================================ FILE: web/src/component/record-row.less ================================================ @import '../style/constant.less'; .tableRow { display: block; cursor: pointer; font-size: @font-size-sm; td { padding-top: 5px !important; padding-bottom: 5px !important; } } .lightBackgroundColor { background: @light-background-color; } .id { padding-left: 20px; text-align: center; } .method { text-align: left; } .code { text-align: center; } // .host { // width: 200px; // } .path { max-width: 0; min-width: 300px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } .mime { min-width: 160px; max-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } // .time { // width: 100px; // } .okStatus { color: @ok-color; } .errorStatus { color: @error-color; } .https { background: url('../assets/https.png'); width: 11px; height: 11px; margin-left: 1px; background-size: contain; display: inline-block; } ================================================ FILE: web/src/component/record-worker.jsx ================================================ /* * A webworker to identify whether the component need to be re-rendered * * the core idea is that, we do the caclulation here, compare the new filtered record with current one. * if they two are same, we'll send no update event out. * otherwise, will send out a full filtered record list, to replace the current one. * * The App itself will just need to display all the filtered records, the filter and max-limit logic are handled here. */ let recordList = []; // store all the filtered record, so there will be no need to re-calculate the filtere record fully through all records. self.FILTERED_RECORD_LIST = []; const defaultLimit = 500; self.currentStateData = []; // the data now used by state self.filterStr = ''; self.canLoadMore = false; self.updateQueryTimer = null; self.refreshing = true; self.beginIndex = 0; self.endIndex = self.beginIndex + defaultLimit - 1; self.IN_DIFF = false; // mark if currently in diff working const getFilterReg = function (filterStr) { let filterReg = null; if (filterStr) { let regFilterStr = filterStr .replace(/\r\n/g, '\n') .replace(/\n\n/g, '\n'); // remove the last /\n$/ in case an accidential br regFilterStr = regFilterStr.replace(/\n*$/, ''); if (regFilterStr[0] === '/' && regFilterStr[regFilterStr.length - 1] === '/') { regFilterStr = regFilterStr.substring(1, regFilterStr.length - 2); } regFilterStr = regFilterStr.replace(/((.+)\n|(.+)$)/g, (matchStr, $1, $2) => { // if there is '\n' in the string if ($2) { return `(${$2})|`; } else { return `(${$1})`; } }); try { filterReg = new RegExp(regFilterStr); } catch (e) { console.error(e); } } return filterReg; }; self.resetDisplayRecordIndex = function () { self.beginIndex = 0; self.endIndex = self.beginIndex + defaultLimit - 1; }; self.getFilteredRecords = function () { // const filterReg = getFilterReg(self.filterStr); // const filterdRecords = []; // const length = recordList.length; // // filtered out the records // for (let i = 0 ; i < length; i++) { // const item = recordList[i]; // if (filterReg && filterReg.test(item.url)) { // filterdRecords.push(item); // } // if (!filterReg) { // filterdRecords.push(item); // } // } // return filterdRecords; return self.FILTERED_RECORD_LIST; }; /* * calculate the filtered records, at each time the origin list is updated * @param isFullyCalculate bool, whether to calculate the filtered record fully, if ture, will do a fully calculation; Otherwise, will only calculate the "listForThisTime", this usually means there are some updates for the 'filtered' list, or some new records arrived * @param listForThisTime object, the list which to be calculated for this time, usually the from the new event, contains some update to exist list, or some new records */ self.calculateFilteredRecords = function (isFullyCalculate, listForThisTime = []) { const filterReg = getFilterReg(self.filterStr); if (isFullyCalculate) { self.FILTERED_RECORD_LIST = []; const length = recordList.length; // filtered out the records for (let i = 0; i < length; i++) { const item = recordList[i]; if (!filterReg || (filterReg && filterReg.test(item.url))) { self.FILTERED_RECORD_LIST.push(item); } } } else { listForThisTime.forEach((item) => { const index = self.FILTERED_RECORD_LIST.findIndex((record) => { return item.id === record.id; }); if (index >= 0) { self.FILTERED_RECORD_LIST[index] = item; } else if (!filterReg || (filterReg && filterReg.test(item.url))) { self.FILTERED_RECORD_LIST.push(item); } }); } }; // diff the record, so when the refreshing is stoped, the page will not be updated // cause the filtered records will be unchanged self.diffRecords = function () { if (self.IN_DIFF) { return; } self.IN_DIFF = true; // mark if the component need to be refreshed let shouldUpdateRecord = false; const filterdRecords = self.getFilteredRecords(); if (self.refreshing) { self.beginIndex = filterdRecords.length - 1 - defaultLimit; self.endIndex = filterdRecords.length - 1; } else { if (self.endIndex > filterdRecords.length) { self.endIndex = filterdRecords.length; } } const newStateRecords = filterdRecords.slice(self.beginIndex, self.endIndex + 1); const currentDataLength = self.currentStateData.length; const newDataLength = newStateRecords.length; if (newDataLength !== currentDataLength) { shouldUpdateRecord = true; } else { // only the two with same index and the `_render` === true then we'll need to render for (let i = 0; i < currentDataLength; i++) { const item = self.currentStateData[i]; const targetItem = newStateRecords[i]; if (item.id !== targetItem.id || targetItem._render === true) { shouldUpdateRecord = true; break; } } } self.currentStateData = newStateRecords; self.postMessage(JSON.stringify({ type: 'updateData', shouldUpdateRecord, recordList: newStateRecords })); self.IN_DIFF = false; }; // check if there are many new records arrivied self.checkNewRecordsTip = function () { if (self.IN_DIFF) { return; } const newRecordLength = self.getFilteredRecords().length; self.postMessage(JSON.stringify({ type: 'updateTip', data: (newRecordLength - self.endIndex) > 0 })); }; self.updateSingle = function (record) { recordList.forEach((item) => { item._render = false; }); const index = recordList.findIndex((item) => { return item.id === record.id; }); if (index >= 0) { // set the mark to ensure the item get re-rendered record._render = true; recordList[index] = record; } else { recordList.push(record); } self.calculateFilteredRecords(false, [record]); }; self.updateMultiple = function (records) { recordList.forEach((item) => { item._render = false; }); records.forEach((record) => { const index = recordList.findIndex((item) => { return item.id === record.id; }); if (index >= 0) { // set the mark to ensure the item get re-rendered record._render = true; recordList[index] = record; } else { recordList.push(record); } }); self.calculateFilteredRecords(false, records); }; self.addEventListener('message', (e) => { const data = JSON.parse(e.data); switch (data.type) { case 'diff' : { self.diffRecords(); break; } case 'updateQuery': { // if filterStr or limit changed if (data.filterStr !== self.filterStr) { self.updateQueryTimer && clearTimeout(self.updateQueryTimer); self.updateQueryTimer = setTimeout(() => { self.resetDisplayRecordIndex(); self.filterStr = data.filterStr; self.calculateFilteredRecords(true); self.diffRecords(); }, 150); } break; } case 'updateSingle': { self.updateSingle(data.data); if (self.refreshing) { self.diffRecords(); } else { self.checkNewRecordsTip(); } break; } case 'updateMultiple': { self.updateMultiple(data.data); if (self.refreshing) { self.diffRecords(); } else { self.checkNewRecordsTip(); } break; } case 'initRecord': { recordList = data.data; self.calculateFilteredRecords(true); self.diffRecords(); break; } case 'clear': { recordList = []; self.calculateFilteredRecords(true); self.diffRecords(); break; } case 'loadMore': { if (self.IN_DIFF) { return; } self.refreshing = false; if (data.data > 0) { self.endIndex += data.data; } else { self.beginIndex = Math.max(self.beginIndex + data.data, 0); } self.diffRecords(); break; } case 'updateRefreshing': { if (typeof data.refreshing === 'boolean') { self.refreshing = data.refreshing; if (self.refreshing) { self.diffRecords(); } } break; } default: { break; } } }); ================================================ FILE: web/src/component/record-ws-message-detail.jsx ================================================ /** * The panel to display the detial of the record * */ import React, { PropTypes } from 'react'; import { message, Button, Icon } from 'antd'; import { formatDate } from 'common/commonUtil'; import { initWs } from 'common/wsUtil'; import ClassBind from 'classnames/bind'; import Style from './record-ws-message-detail.less'; import CommonStyle from '../style/common.less'; const ToMessage = (props) => { const { message: wsMessage } = props; return (
      {formatDate(wsMessage.time, 'hh:mm:ss:ms')}
      {wsMessage.message}
      ); } const FromMessage = (props) => { const { message: wsMessage } = props; return (
      {formatDate(wsMessage.time, 'hh:mm:ss:ms')}
      {wsMessage.message}
      ); } class RecordWsMessageDetail extends React.Component { constructor() { super(); this.state = { stateCheck: false, // a prop only to trigger state check autoRefresh: true, socketMessages: [] // the messages from websocket listening }; this.updateStateRef = null; // a timeout ref to reduce the calling of update state this.wsClient = null; // ref to the ws client this.onMessageHandler = this.onMessageHandler.bind(this); this.receiveNewMessage = this.receiveNewMessage.bind(this); this.toggleRefresh = this.toggleRefresh.bind(this); } static propTypes = { recordDetail: PropTypes.object } toggleRefresh () { const { autoRefresh } = this.state; this.state.autoRefresh = !autoRefresh; this.setState({ stateCheck: true }); } receiveNewMessage (message) { this.state.socketMessages.push(message); this.updateStateRef && clearTimeout(this.updateStateRef); this.updateStateRef = setTimeout(() => { this.setState({ stateCheck: true }); }, 100); } getMessageList () { const { recordDetail } = this.props; const { socketMessages } = this.state; const { wsMessages = [] } = recordDetail; const targetMessage = wsMessages.concat(socketMessages); return targetMessage.map((messageItem, index) => { return messageItem.isToServer ? : ; }); } refreshPage () { const { autoRefresh } = this.state; if (autoRefresh && this.messageRef && this.messageContentRef) { this.messageRef.scrollTop = this.messageContentRef.scrollHeight; } } onMessageHandler (event) { const { recordDetail } = this.props; const data = JSON.parse(event.data); const content = data.content; if (data.type === 'updateLatestWsMsg' ) { if (recordDetail.id === content.id) { this.receiveNewMessage(content.message); } } } componentDidUpdate () { this.refreshPage(); } componentWillUnmount () { this.wsClient && this.wsClient.removeEventListener('message', this.onMessageHandler); } componentDidMount () { const { recordDetail } = this.props; this.refreshPage(); this.wsClient = initWs(); this.wsClient.addEventListener('message', this.onMessageHandler); } render() { const { recordDetail } = this.props; const { autoRefresh } = this.state; if (!recordDetail) { return null; } const playIcon = ; const pauseIcon = ; return (
      this.messageRef = _ref}>
      this.messageContentRef = _ref}> {this.getMessageList()}
      {autoRefresh ? pauseIcon : playIcon}
      ); } } export default RecordWsMessageDetail; ================================================ FILE: web/src/component/record-ws-message-detail.less ================================================ @import '../style/constant.less'; .wrapper { position: absolute; top: 0; left: 0; width: 100%; height: 100%; overflow: auto; } .contentWrapper { overflow: hidden; } .toMessage { float: right; clear: both; margin: 5px auto; .content { background-color: @primary-color; } } .fromMessage { float: left; clear: both; max-width: 40%; margin: 5px auto; .content { background: @success-color; } } .time { font-size: @font-size-xs; color: @tip-color; } .content { clear: both; border-radius: @border-radius-base; color: #fff; padding: 7px 8px; font-size: @font-size-sm; word-wrap: break-word; word-break: break-all; } .refreshBtn { position: fixed; right: 20px; bottom: 5px; opacity: 0.53; font-size: @font-size-large; cursor: pointer; } ================================================ FILE: web/src/component/resizable-panel.jsx ================================================ /* * A copoment to display content in the a resizable panel */ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import { Icon } from 'antd'; import Style from './resizable-panel.less'; import ClassBind from 'classnames/bind'; const StyleBind = ClassBind.bind(Style); class ResizablePanel extends React.Component { constructor () { super(); this.state = { dragBarLeft: '', contentLeft: '' }; this.onDragbarMoveUp = this.onDragbarMoveUp.bind(this); this.onDragbarMove = this.onDragbarMove.bind(this); this.onDragbarMoveDown = this.onDragbarMoveDown.bind(this); this.doClose = this.doClose.bind(this); this.onKeyUp = this.onKeyUp.bind(this); this.addKeyEvent = this.addKeyEvent.bind(this); this.removeKeyEvent = this.removeKeyEvent.bind(this); } static propTypes = { children: PropTypes.element, onClose: PropTypes.func, visible: PropTypes.bool } onDragbarMove (event) { this.setState({ dragBarLeft: event.pageX }); } onKeyUp (e) { if (e.keyCode == 27) { this.doClose(); } } addKeyEvent () { document.addEventListener('keyup', this.onKeyUp); } removeKeyEvent () { document.removeEventListener('keyup', this.onKeyUp); } onDragbarMoveUp (event) { this.setState({ contentLeft: event.pageX }); document.removeEventListener('mousemove', this.onDragbarMove); document.removeEventListener('mouseup', this.onDragbarMoveUp); } onDragbarMoveDown (event) { document.addEventListener('mousemove', this.onDragbarMove); document.addEventListener('mouseup', this.onDragbarMoveUp); } doClose () { this.props.onClose && this.props.onClose(); } render () { if (!this.props.visible) { this.removeKeyEvent(); return null; } this.addKeyEvent(); const { dragBarLeft, contentLeft } = this.state; const propsLeft = this.props.left; const dragBarStyle = dragBarLeft || propsLeft ? { 'left': dragBarLeft || propsLeft } : null; const contentStyle = contentLeft || propsLeft ? { 'left': contentLeft || propsLeft } : null; const modalStyle = this.props.hideBackModal ? contentStyle : { 'left': 0 }; return (
      {this.props.children}
      ); } } export default ResizablePanel; ================================================ FILE: web/src/component/resizable-panel.less ================================================ @import '../style/constant.less'; .wrapper { display: block; background-color: #fff; position: relative; width: 360px; height: 100%; -webkit-box-shadow: 10px 0px 21px 0px rgba(97,95,97,0.15); -moz-box-shadow: 10px 0px 21px 0px rgba(97,95,97,0.15); box-shadow: 10px 0px 21px 0px rgba(97,95,97,0.15); overflow-y: auto; overflow-x: hidden; z-index: 1; } .contentWrapper, .content { height: 100%; } .dragBar { width: 1px; background-color: @hint-color; z-index: 2000; cursor: col-resize; } .closeIcon { position: absolute; top: 0; right: 0; cursor: pointer; padding: 15px; i { font-size: @font-size-large; } &:hover { color: @primary-color; } } ================================================ FILE: web/src/component/table-panel.jsx ================================================ /* * A copoment for the request log table */ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import { Table } from 'antd'; import { formatDate } from 'common/commonUtil'; import Style from './table-panel.less'; import ClassBind from 'classnames/bind'; import CommonStyle from '../style/common.less'; const StyleBind = ClassBind.bind(Style); class TablePanel extends React.Component { constructor () { super(); this.state = { active: true }; } static propTypes = { data: PropTypes.array } getTr () { } render () { const httpsIcon = ; const columns = [ { title: '#', width: 50, dataIndex: 'id' }, { title: 'Method', width:100, dataIndex: 'method', render (text, item) { return {text} {item.protocol === 'https' ? httpsIcon : null}; } }, { title: 'Code', width: 70, dataIndex: 'statusCode', render(text) { const className = StyleBind({ 'okStatus': text === '200' }); return {text}; } }, { title: 'Host', width: 200, dataIndex: 'host' }, { title: 'Path', dataIndex: 'path' }, { title: 'MIME', width: 150, dataIndex: 'mime' }, { title: 'Start', width: 100, dataIndex: 'startTime', render (text) { const timeStr = formatDate(text, 'hh:mm:ss'); return {timeStr}; } } ]; function rowClassFunc (record, index) { return StyleBind('row', { 'lightBackgroundColor': index % 2 === 1 }); } return (
      {data.id} {this.getMethodDiv(data)} {this.getCodeDiv(data)} {data.host} {data.path} {data.mime} {formatDate(data.startTime, 'hh:mm:ss')}
      ); } } export default TablePanel; ================================================ FILE: web/src/component/table-panel.less ================================================ @import '../style/constant.less'; .tableWrapper { clear: both; margin-top: 30px; :global { th { padding-top: 5px !important; padding-bottom: 5px !important; background: @background-color !important; border-top: 1px solid @hint-color; border-bottom: 1px solid @hint-color; } } } .row { cursor: pointer; font-size: @font-size-sm; td { padding-top: 5px !important; padding-bottom: 5px !important; } } .lightBackgroundColor { background: @light-background-color; } .okStatus { color: @ok-color; } ================================================ FILE: web/src/component/title-bar.jsx ================================================ /** * The panel to edit the filter * */ import React, { PropTypes } from 'react'; import { Icon } from 'antd'; import { getQueryParameter } from 'common/commonUtil'; import Style from './title-bar.less'; class TitleBar extends React.Component { constructor () { super(); this.state = { inMaxWindow: false, inApp: getQueryParameter('in_app_mode') // will only show the bar when in app }; } static propTypes = { } render() { if (this.state.inApp !== 'true') { return null; } // the buttons with normal window size const normalButton = (
      ); const maxmizeButton = (
      ); return this.state.inMaxWindow ? maxmizeButton : normalButton; } } export default TitleBar; ================================================ FILE: web/src/component/title-bar.less ================================================ @import '../style/constant.less'; .wrapper { position: relative; } .iconButtons { overflow: hidden; span { position: relative; display: block; float: left; width: 12.44px; height: 12.84px; border-radius: 50%; margin: 0 7px; } .close { background-color: #EB3131; } .disabled { background-color: #aaa; } .minimize { background-color: #FFBF11; } .maxmize { background-color: #09CE26; } i { display: none; position: absolute; left: 0px; top: 1px; } &:hover { i { display: block; } } } ================================================ FILE: web/src/component/ws-listener.jsx ================================================ /* * listen on the websocket event * */ import React, { PropTypes } from 'react'; import { message } from 'antd'; import { initWs } from 'common/wsUtil'; import { updateWholeRequest } from 'action/recordAction'; import { updateShouldClearRecord, updateShowNewRecordTip } from 'action/globalStatusAction'; const RecordWorker = require('worker-loader?inline!./record-worker.jsx'); import { getJSON } from 'common/apiUtil'; const myRecordWorker = new RecordWorker(window.URL.createObjectURL(new Blob([RecordWorker.toString()]))); const fetchLatestLog = function () { getJSON('/latestLog') .then((data) => { const msg = { type: 'initRecord', data }; myRecordWorker.postMessage(JSON.stringify(msg)); }) .catch((error) => { console.error(error); message.error(error.errorMsg || 'Failed to load latest log'); }); }; class WsListener extends React.Component { constructor() { super(); this.state = { wsInited: false } this.initWs = this.initWs.bind(this); this.onWsMessage = this.onWsMessage.bind(this); this.loadNext = this.loadNext.bind(this); this.loadPrevious = this.loadPrevious.bind(this); this.stopPanelRefreshing = this.stopPanelRefreshing.bind(this); fetchLatestLog(); this.refreshing = true; this.loadingNext = false; } static propTypes = { dispatch: PropTypes.func, globalStatus: PropTypes.object } loadPrevious() { this.stopPanelRefreshing(); myRecordWorker.postMessage(JSON.stringify({ type: 'loadMore', data: -500 })); } loadNext() { this.loadingNext = true; myRecordWorker.postMessage(JSON.stringify({ type: 'loadMore', data: 500 })); } stopPanelRefreshing() { this.refreshing = false; myRecordWorker.postMessage(JSON.stringify({ type: 'updateRefreshing', refreshing: false })); } resumePanelRefreshing() { this.refreshing = true; this.loadingNext = false; this.props.dispatch(updateShowNewRecordTip(false)); myRecordWorker.postMessage(JSON.stringify({ type: 'updateRefreshing', refreshing: true })); } onWsMessage(event) { try { const data = JSON.parse(event.data); switch (data.type) { case 'update': { const record = data.content; // stop update the record when it's turned off if (this.props.globalStatus.recording) { // this.props.dispatch(updateRecord(record)); const msg = { type: 'updateSingle', data: record }; myRecordWorker.postMessage(JSON.stringify(msg)); } break; } case 'updateMultiple': { const records = data.content; // stop update the record when it's turned off if (this.props.globalStatus.recording) { // // only in multiple mode we consider there are new records // if (!this.refreshing && !this.loadingNext) { // console.info(`==> this.loadingNext`, this.loadingNext) // const hasNew = records.some((item) => { // return (typeof item.id !== 'undefined'); // }); // hasNew && this.props.dispatch(updateShowNewRecordTip(true)); // } const msg = { type: 'updateMultiple', data: records }; myRecordWorker.postMessage(JSON.stringify(msg)); } break; } default : { break; } } } catch (error) { console.error(error); console.error('Failed to parse the websocket data with message: ', event.data); } } initWs() { if (this.state.wsInited) { return; } this.state.wsInited = true; const wsClient = initWs(); wsClient.onmessage = this.onWsMessage; } componentDidMount() { myRecordWorker.addEventListener('message', (e) => { const data = JSON.parse(e.data); this.loadingNext = false; switch (data.type) { case 'updateData': { if (data.shouldUpdateRecord) { this.props.dispatch(updateWholeRequest(data.recordList)); } break; } case 'updateTip': { this.props.dispatch(updateShowNewRecordTip(data.data)); break; } default: { break; } } }); } componentWillReceiveProps(nextProps) { const { shouldClearAllRecord: nextShouldClearAllRecord } = nextProps.globalStatus; // if it's going to clear the record, if (nextShouldClearAllRecord) { const msg = { type: 'clear' }; myRecordWorker.postMessage(JSON.stringify(msg)); this.props.dispatch(updateShouldClearRecord(false)); } } render() { this.initWs(); const { displayRecordLimit: limit, filterStr } = this.props.globalStatus; const msg = { type: 'updateQuery', limit, filterStr }; myRecordWorker.postMessage(JSON.stringify(msg)); return
      ; } } export default WsListener; ================================================ FILE: web/src/index.jsx ================================================ import React, { PropTypes } from 'react'; import ReactDOM from 'react-dom'; import { createStore, applyMiddleware } from 'redux'; import { Provider, connect } from 'react-redux'; import { LocaleProvider } from 'antd'; import enUS from 'antd/lib/locale-provider/en_US'; import createSagaMiddleware from 'redux-saga'; import rootSaga from 'saga/rootSaga'; import { MenuKeyMap } from 'common/constant'; import { getQueryParameter } from 'common/commonUtil'; import reducer from 'reducer/rootReducer'; import HeaderMenu from 'component/header-menu'; import RecordPanel from 'component/record-panel'; import RecordFilter from 'component/record-filter'; import MapLocal from 'component/map-local'; import WsListener from 'component/ws-listener'; import RecordDetail from 'component/record-detail'; import LeftMenu from 'component/left-menu'; import DownloadRootCA from 'component/download-root-ca'; require('./style/antd-reset.global.less'); import Style from './index.less'; import CommonStyle from './style/common.less'; const { RECORD_FILTER: RECORD_FILTER_MENU_KEY, MAP_LOCAL: MAP_LOCAL_MENU_KEY, ROOT_CA: ROOT_CA_MENU_KEY } = MenuKeyMap; const sagaMiddleware = createSagaMiddleware(); const store = createStore(reducer, applyMiddleware(sagaMiddleware)); sagaMiddleware.run(rootSaga); class App extends React.Component { constructor() { super(); this.state = { showResizePanel: false, panelIndex: '', inAppMode: getQueryParameter('in_app_mode'), refreshing: true }; this.onResizePanelClose = this.onResizePanelClose.bind(this); this.onRecordScroll = this.onRecordScroll.bind(this); this.stopRefresh = this.stopRefresh.bind(this); this.resumeFresh = this.resumeFresh.bind(this); this.detectIfToStopRefreshing = this.detectIfToStopRefreshing.bind(this); this.scrollHandler = this.scrollHandler.bind(this); this.initRecrodPanelWrapperRef = this.initRecrodPanelWrapperRef.bind(this); this.recordTableRef = null; this.wsListenerRef = null; this.lastScrollTop = 0; this.scrollHandlerTimeout = null; this.stopRefreshTimout = null; this.stopRefreshTokenScrollTop = null; // the token used to decide the move distance } static propTypes = { dispatch: PropTypes.func, requestRecord: PropTypes.object, globalStatus: PropTypes.object } stopRefresh() { this.wsListenerRef && this.wsListenerRef.stopPanelRefreshing(); this.state.refreshing = false; this.setState({ refreshing: false }); } resumeFresh() { this.wsListenerRef && this.wsListenerRef.resumePanelRefreshing(); this.state.refreshing = true; this.setState({ refreshing: true }); } onResizePanelClose() { this.setState({ showResizePanel: false }); } // if is scrolling up during refresh, will stop the refresh detectIfToStopRefreshing(currentScrollTop) { if (!this.stopRefreshTokenScrollTop) { this.stopRefreshTokenScrollTop = currentScrollTop; } this.stopRefreshTimout = setTimeout(() => { // if the scrollbar is scrolled up more than 50px, stop refreshing if ((this.stopRefreshTokenScrollTop - currentScrollTop) > 50) { this.stopRefresh(); this.stopRefreshTokenScrollTop = null; } }, 50); } initRecrodPanelWrapperRef(ref) { this.recordTableRef = ref; ref.addEventListener('wheel', this.onRecordScroll, { passive: true }); } scrollHandler() { if (!this.recordTableRef || !this.wsListenerRef) { return; } const self = this; const scrollTop = this.recordTableRef.scrollTop; if (scrollTop < this.lastScrollTop || (this.lastScrollTop === 0)) { this.detectIfToStopRefreshing(scrollTop); // load more previous record when scrolled to top if (scrollTop < 10) { self.state.loadingPrev = true; self.setState({ loadingPrev: true }); //TODO: hide the loading stauts after 1000 ms, a lazy way to hide it when there is no previous records setTimeout(() => { self.state.loadingPrev = false; self.setState({ loadingPrev: false }); }, 1000); this.wsListenerRef.loadPrevious(); } } else if (scrollTop >= this.lastScrollTop) { const recordPanelHeight = this.recordTableRef.firstChild.clientHeight; const tableHeight = this.recordTableRef.clientHeight; // when close to bottom in less than 30, load more next records if (scrollTop + tableHeight + 30 > recordPanelHeight) { this.state.loadNext = true; this.setState({ loadingNext: true }); this.wsListenerRef.loadNext(); } } this.lastScrollTop = scrollTop; } onRecordScroll() { this.scrollHandlerTimeout && clearTimeout(this.scrollHandlerTimeout); this.scrollHandlerTimeout = setTimeout(() => { this.scrollHandler(); }, 60); } getResumeFreshDiv() { if (!this.props.globalStatus.showNewRecordTip) { return null; } return (
      New Records Detected.
      ); } getMiddlePanel() { const { activeMenuKey } = this.props.globalStatus; let middlePanel = null; // TODO: move the logic of resizepanel out to here, keep each panel component independant switch (activeMenuKey) { case RECORD_FILTER_MENU_KEY: { middlePanel = ; break; } case MAP_LOCAL_MENU_KEY: { middlePanel = ; break; } case ROOT_CA_MENU_KEY: { middlePanel = ; break; } default: { middlePanel = null; break; } } return middlePanel; } componentWillReceiveProps(nextProps) { const { recordList: nextRecordList } = nextProps.requestRecord; const { recordList: currentRecordList } = this.props.requestRecord; // if there are new data, reset the status of loadingNext and loadingPrev if (nextRecordList !== currentRecordList) { // scroll the window to last remembered position, when in loading pre mode if (this.state.loadingPrev) { const nextBeginId = nextRecordList[0].id; const currentBeginId = currentRecordList[0].id; if (nextBeginId < currentBeginId) { // each line is limited to 29px const scrollPosition = 29 * (nextRecordList.length - currentRecordList.length); if (this.recordTableRef) { setTimeout(() => { this.recordTableRef.scrollTop = scrollPosition; }, 200); } } this.state.loadingPrev = false; } this.setState({ loadingNext: false, loadingPrev: false }); } } componentDidUpdate() { if (this.state.refreshing && this.recordTableRef && !this.state.loadingPrev) { this.recordTableRef.scrollTop = this.recordTableRef.scrollHeight; } } componentDidMount() { if (this.state.refreshing && this.recordTableRef) { this.recordTableRef.scrollTop = this.recordTableRef.scrollHeight; } } render() { const { lastActiveRecordId, currentActiveRecordId } = this.props.globalStatus; const leftMenuDiv = (
      ); return (
      {this.state.inAppMode ? null : leftMenuDiv}
      {this.getMiddlePanel()}
      {this.getResumeFreshDiv()}
      { this.wsListenerRef = ref; }} dispatch={this.props.dispatch} globalStatus={this.props.globalStatus} />
      ); } } function select(state) { return { requestRecord: state.requestRecord, globalStatus: state.globalStatus }; } const ReduxApp = connect(select)(App); ReactDOM.render( , document.getElementById('root')); ================================================ FILE: web/src/index.less ================================================ @import './style/constant.less'; .indexWrapper { display: block; position: relative; height: 100%; } .leftPanel { width: 158px; height: 100%; min-height: 300px; float: left; -webkit-app-region: drag; -webkit-user-select: none; } .middlePanel { float: left; height: 100%; } .rightPanel { overflow: hidden; position: relative; height: 100%; z-index: 0; } .headerWrapper { border-bottom: 1px solid @hint-color; padding-bottom: 5px; padding-top: 25px; -webkit-app-region: drag; -webkit-user-select: none; } .tableWrapper { clear: both; position: absolute; top: 89px; left: 0; right: 0; bottom: 0; overflow: auto; transform: translateZ(0); } .resumeTip { display: block; background-color: @primary-color; color: #fff; position: absolute; right: 30px; bottom: 15px; padding: 10px 10px; opacity: 0.9; border-radius: 4px; cursor: pointer; -webkit-box-shadow: 1px 1px 9px 0px rgba(0,0,0,0.37); -moz-box-shadow: 1px 1px 9px 0px rgba(0,0,0,0.37); box-shadow: 1px 1px 9px 0px rgba(0,0,0,0.37); &:hover { opacity: 1; } } .arrowDown { display: block; position: absolute; bottom: -15px; right: 60px; width: 0; height: 0; border-left: 5px solid transparent; border-right: 5px solid transparent; border-top: 5px solid @primary-color; opacity: 0.9; } ================================================ FILE: web/src/reducer/globalStatusReducer.js ================================================ const defaultStatus = { recording: true, panelRefreshing: true, // indicate whether the record panel should be refreshing showFilter: false, // if the filter panel is showing showMapLocal: false, activeMenuKey: '', canLoadMore: false, interceptHttpsFlag: false, globalProxyFlag: false, // is global proxy now filterStr: '', directory: [], lastActiveRecordId: -1, currentActiveRecordId: -1, shouldClearAllRecord: false, appVersion: '', panelLoadingNext: false, panelLoadingPrev: false, showNewRecordTip: false, isRootCAFileExists: false, fetchingRecord: false, mappedConfig:[] // configured map config }; import { MenuKeyMap } from 'common/constant'; import { STOP_RECORDING, RESUME_RECORDING, SHOW_FILTER, HIDE_FILTER, UPDATE_FILTER, UPDATE_LOCAL_DIRECTORY, SHOW_MAP_LOCAL, HIDE_MAP_LOCAL, UPDATE_LOCAL_MAPPED_CONFIG, UPDATE_ACTIVE_RECORD_ITEM, UPDATE_LOCAL_INTERCEPT_HTTPS_FLAG, UPDATE_LOCAL_GLOBAL_PROXY_FLAG, HIDE_ROOT_CA, SHOW_ROOT_CA, UPDATE_CAN_LOAD_MORE, INCREASE_DISPLAY_RECORD_LIST, UPDATE_SHOULD_CLEAR_RECORD, UPDATE_APP_VERSION, UPDATE_IS_ROOTCA_EXISTS, UPDATE_SHOW_NEW_RECORD_TIP, UPDATE_FETCHING_RECORD_STATUS } from 'action/globalStatusAction'; // The map to save the mapping relationships of the path and it's location in the tree node const directoryNodeMap = {}; // The map to store all the directory in a tree way let direcotryList = []; function getTreeMap(path, sub) { const children = []; sub.directory.forEach((item) => { if (!(item.name.indexOf('.') === 0)) { item.isLeaf = false; directoryNodeMap[item.fullPath] = item; children.push(item); } }); sub.file.forEach((item) => { if (!(item.name.indexOf('.') === 0)) { item.isLeaf = true; directoryNodeMap[item.fullPath] = item; children.push(item); } }); if (!path) { direcotryList = children; } else { directoryNodeMap[path].children = children; } return direcotryList; } function requestListReducer(state = defaultStatus, action) { switch (action.type) { case STOP_RECORDING: { const newState = Object.assign({}, state); newState.recording = false; return newState; } case RESUME_RECORDING: { const newState = Object.assign({}, state); newState.recording = true; return newState; } case SHOW_FILTER: { const newState = Object.assign({}, state); newState.activeMenuKey = MenuKeyMap.RECORD_FILTER; return newState; } case HIDE_FILTER: { const newState = Object.assign({}, state); newState.activeMenuKey = ''; return newState; } case UPDATE_FILTER: { const newState = Object.assign({}, state); newState.filterStr = action.data; return newState; } case SHOW_MAP_LOCAL: { const newState = Object.assign({}, state); newState.activeMenuKey = MenuKeyMap.MAP_LOCAL; return newState; } case HIDE_MAP_LOCAL: { const newState = Object.assign({}, state); newState.activeMenuKey = ''; return newState; } case UPDATE_LOCAL_DIRECTORY: { const newState = Object.assign({}, state); const { path, sub } = action.data; newState.directory = getTreeMap(path, sub); return newState; } case UPDATE_LOCAL_MAPPED_CONFIG: { const newState = Object.assign({}, state); newState.mappedConfig = action.data; return newState; } case UPDATE_ACTIVE_RECORD_ITEM: { const newState = Object.assign({}, state); newState.lastActiveRecordId = state.currentActiveRecordId; newState.currentActiveRecordId = action.data; return newState; } case UPDATE_LOCAL_INTERCEPT_HTTPS_FLAG: { const newState = Object.assign({}, state); newState.interceptHttpsFlag = action.data; return newState; } case UPDATE_LOCAL_GLOBAL_PROXY_FLAG: { const newState = Object.assign({}, state); newState.globalProxyFlag = action.data; return newState; } case SHOW_ROOT_CA: { const newState = Object.assign({}, state); newState.activeMenuKey = MenuKeyMap.ROOT_CA; return newState; } case HIDE_ROOT_CA: { const newState = Object.assign({}, state); newState.activeMenuKey = ''; return newState; } case UPDATE_CAN_LOAD_MORE: { const newState = Object.assign({}, state); newState.canLoadMore = action.data; return newState; } case UPDATE_SHOULD_CLEAR_RECORD: { const newState = Object.assign({}, state); newState.shouldClearAllRecord = action.data; return newState; } case INCREASE_DISPLAY_RECORD_LIST: { const newState = Object.assign({}, state); newState.displayRecordLimit += action.data; return newState; } case UPDATE_APP_VERSION: { const newState = Object.assign({}, state); newState.appVersion = action.data; return newState; } case UPDATE_SHOW_NEW_RECORD_TIP: { const newState = Object.assign({}, state); newState.showNewRecordTip = action.data; return newState; } case UPDATE_IS_ROOTCA_EXISTS: { const newState = Object.assign({}, state); newState.isRootCAFileExists = action.data; return newState; } case UPDATE_FETCHING_RECORD_STATUS: { const newState = Object.assign({}, state); newState.fetchingRecord = action.data; return newState; } default: { return state; } } } export default requestListReducer; ================================================ FILE: web/src/reducer/requestRecordReducer.js ================================================ const defaultState = { recordList: [], recordDetail: null }; import { UPDATE_WHOLE_REQUEST, UPDATE_SINGLE_RECORD, CLEAR_ALL_LOCAL_RECORD, UPDATE_MULTIPLE_RECORDS, SHOW_RECORD_DETAIL, HIDE_RECORD_DETAIL } from 'action/recordAction'; const getRecordInList = function (recordId, recordList) { const newRecordList = recordList.slice(); for (let i = 0; i< newRecordList.length ; i++) { const record = newRecordList[i]; if (record.id === recordId) { return record; } } }; function requestListReducer (state = defaultState, action) { switch (action.type) { case UPDATE_WHOLE_REQUEST: { const newState = Object.assign({}, state); newState.recordList = action.data.slice(); return newState; } case UPDATE_SINGLE_RECORD: { const newState = Object.assign({}, state); const list = newState.recordList.slice(); list.forEach((item) => { item._render = false; }); const record = action.data; const index = list.findIndex((item) => { return item.id === record.id; }); if (index >= 0) { // set the mark to ensure the item get re-rendered record._render = true; list[index] = record; } else { list.push(record); } newState.recordList = list; return newState; } case UPDATE_MULTIPLE_RECORDS: { const newState = Object.assign({}, state); const list = newState.recordList.slice(); list.forEach((item) => { item._render = false; }); const records = action.data; records.forEach((record) => { const index = list.findIndex((item) => { return item.id === record.id; }); if (index >= 0) { // set the mark to ensure the item get re-rendered record._render = true; list[index] = record; } else { list.push(record); } }); newState.recordList = list; return newState; } case CLEAR_ALL_LOCAL_RECORD: { const newState = Object.assign({}, state); newState.recordList = []; return newState; } case SHOW_RECORD_DETAIL: { const newState = Object.assign({}, state); const responseBody = action.data; const originRecord = getRecordInList(responseBody.id, newState.recordList); // 只在id存在的时候,才更新, 否则取消 if (originRecord) { newState.recordDetail = Object.assign(responseBody, originRecord); } else { newState.recordDetail = null; } return newState; } case HIDE_RECORD_DETAIL: { const newState = Object.assign({}, state); newState.recordDetail = null; return newState; } default: { return state; } } } export default requestListReducer; ================================================ FILE: web/src/reducer/rootReducer.js ================================================ import requestRecordReducer from './requestRecordReducer'; import globalStatusReducer from './globalStatusReducer'; const defaultState = { }; export default function(state = defaultState, action) { return { requestRecord: requestRecordReducer(state.requestRecord, action), globalStatus: globalStatusReducer(state.globalStatus, action) }; } ================================================ FILE: web/src/saga/rootSaga.js ================================================ import { take, put, call, fork } from 'redux-saga/effects'; import { message } from 'antd'; import { FETCH_REQUEST_LOG, CLEAR_ALL_RECORD, FETCH_RECORD_DETAIL, clearAllLocalRecord, updateWholeRequest, showRecordDetail } from 'action/recordAction'; import { FETCH_DIRECTORY, FETCH_MAPPED_CONFIG, UPDATE_REMOTE_MAPPED_CONFIG, TOGGLE_REMOTE_INTERCEPT_HTTPS, TOGGLE_REMORE_GLOBAL_PROXY_FLAG, updateLocalDirectory, updateLocalMappedConfig, updateActiveRecordItem, updateLocalInterceptHttpsFlag, updateFechingRecordStatus, updateLocalGlobalProxyFlag } from 'action/globalStatusAction'; import { getJSON, postJSON, isApiSuccess } from 'common/apiUtil'; function* doFetchRequestList() { const data = yield call(getJSON, '/latestLog'); yield put(updateWholeRequest(data)); } function* doFetchDirectory(path = '') { const sub = yield call(getJSON, '/filetree', { root: path }); yield put(updateLocalDirectory(path, sub)); } function* doFetchMappedConfig() { const config = yield call(getJSON, '/getMapConfig'); yield put(updateLocalMappedConfig(config)); } function* doFetchRecordBody(recordId) { // const recordBody = { id: recordId }; yield put(updateFechingRecordStatus(true)); const recordBody = yield call(getJSON, '/fetchBody', { id: recordId }); if (recordBody.method && recordBody.method.toLowerCase() === 'websocket') { recordBody.wsMessages = yield call(getJSON, '/fetchWsMessages', { id: recordId}); } recordBody.id = parseInt(recordBody.id, 10); yield put(updateFechingRecordStatus(false)); yield put(updateActiveRecordItem(recordId)); yield put(showRecordDetail(recordBody)); } function* doUpdateRemoteMappedConfig(config) { const newConfig = yield call(postJSON, '/setMapConfig', config); yield put(updateLocalMappedConfig(newConfig)); } function * doToggleRemoteInterceptHttps(flag) { yield call(postJSON, '/api/toggleInterceptHttps', { flag: flag }); yield put(updateLocalInterceptHttpsFlag(flag)); } function * doToggleRemoteGlobalProxy(flag) { const result = yield call(postJSON, '/api/toggleGlobalProxy', { flag: flag }); const windowsMessage = 'Successfully turned on, it may take up to 1 min to take effect.'; const linuxMessage = 'Successfully turned on.'; const turnDownMessage = 'Global proxy has been turned down.'; if (isApiSuccess(result)) { const tipMessage = result.isWindows ? windowsMessage : linuxMessage; message.success(flag ? tipMessage : turnDownMessage, 3); yield put(updateLocalGlobalProxyFlag(flag)); } else { message.error(result.errorMsg, 3); } } function * fetchRequestSaga() { while (true) { yield take(FETCH_REQUEST_LOG); yield fork(doFetchRequestList); } } function * clearRequestRecordSaga() { while (true) { yield take(CLEAR_ALL_RECORD); yield put(clearAllLocalRecord()); } } function * fetchDirectorySaga() { while (true) { const action = yield take(FETCH_DIRECTORY); yield fork(doFetchDirectory, action.data); } } function * fetchMappedConfigSaga() { while (true) { yield take(FETCH_MAPPED_CONFIG); yield fork(doFetchMappedConfig); } } function * updateRemoteMappedConfigSaga() { while (true) { const action = yield take(UPDATE_REMOTE_MAPPED_CONFIG); yield fork(doUpdateRemoteMappedConfig, action.data); } } function * fetchRecordBodySaga() { while (true) { const action = yield take(FETCH_RECORD_DETAIL); yield fork(doFetchRecordBody, action.data); } } function * toggleRemoteInterceptHttpsSaga() { while (true) { const action = yield take(TOGGLE_REMOTE_INTERCEPT_HTTPS); yield fork(doToggleRemoteInterceptHttps, action.data); } } function * toggleRemoteGlobalProxySaga() { while (true) { const action = yield take(TOGGLE_REMORE_GLOBAL_PROXY_FLAG); yield fork(doToggleRemoteGlobalProxy, action.data); } } export default function* root() { yield fork(fetchRequestSaga); yield fork(clearRequestRecordSaga); yield fork(fetchDirectorySaga); yield fork(fetchMappedConfigSaga); yield fork(updateRemoteMappedConfigSaga); yield fork(fetchRecordBodySaga); yield fork(toggleRemoteInterceptHttpsSaga); yield fork(toggleRemoteGlobalProxySaga); } ================================================ FILE: web/src/style/animate.less ================================================ @keyframes rotation { 0% { -webkit-transform: rotate(0deg); opacity: 0.1; } 20% { opacity: 0.1; } 40% { opacity: 0.25; } 80% { opacity: 0.1; } 100% { -webkit-transform: rotate(359deg); opacity: 0.1; } } .rotation { animation: rotation 1.2s infinite cubic-bezier(.63,.33,.46,.71); } ================================================ FILE: web/src/style/antd-constant.less ================================================ @primary-color : #2db7f5; @info-color : #2db7f5; @success-color : #87d068; @error-color : #f50; @warning-color : #fa0; // ------ Base & Require ------ @body-background : #fff; @font-family : "Helvetica Neue",Helvetica,"PingFang SC","Hiragino Sans GB","Microsoft YaHei","微软雅黑",Arial,sans-serif; @code-family : Consolas,Menlo,Courier,monospace; @text-color : #666; @font-size-base : 12px; @line-height-base : 1.5; @line-height-computed : floor((@font-size-base * @line-height-base)); @border-radius-base : 6px; @border-radius-sm : 4px; // LINK @link-color : #2db7f5; @link-hover-color : tint(@link-color, 20%); @link-active-color : shade(@link-color, 5%); @link-hover-decoration : none; // Disabled cursor for form controls and buttons. @cursor-disabled : not-allowed; // Animation @ease-out : cubic-bezier(0.215, 0.61, 0.355, 1); @ease-in : cubic-bezier(0.55, 0.055, 0.675, 0.19); @ease-in-out : cubic-bezier(0.645, 0.045, 0.355, 1); @ease-out-back : cubic-bezier(0.12, 0.4, 0.29, 1.46); @ease-in-back : cubic-bezier(0.71, -0.46, 0.88, 0.6); @ease-in-out-back : cubic-bezier(0.71, -0.46, 0.29, 1.46); @ease-out-circ : cubic-bezier(0.08, 0.82, 0.17, 1); @ease-in-circ : cubic-bezier(0.6, 0.04, 0.98, 0.34); @ease-in-out-circ : cubic-bezier(0.78, 0.14, 0.15, 0.86); @ease-out-quint : cubic-bezier(0.23, 1, 0.32, 1); @ease-in-quint : cubic-bezier(0.755, 0.05, 0.855, 0.06); @ease-in-out-quint : cubic-bezier(0.86, 0, 0.07, 1); // Border color @border-color-base : #d9d9d9; // base border outline a component @border-color-split : #e9e9e9; // split border inside a component // Shadow @shadow-color : rgba(100, 100, 100, .2); @box-shadow-base : @shadow-1-down; @shadow-1-up : 0 -1px 6px @shadow-color; @shadow-1-down : 0 1px 6px @shadow-color; @shadow-1-left : -1px 0 6px @shadow-color; @shadow-1-right : 1px 0 6px @shadow-color; @shadow-2 : 0 1px 8px @shadow-color; // Buttons @btn-font-weight : 500; @btn-primary-color : #fff; @btn-primary-bg : @primary-color; @btn-group-border : shade(@primary-color, 5%); @btn-default-color : @text-color; @btn-default-bg : #f7f7f7; @btn-default-border : @border-color-base; @btn-ghost-color : @text-color; @btn-ghost-bg : transparent; @btn-ghost-border : @border-color-base; @btn-disable-color : #ccc; @btn-disable-bg : #f7f7f7; @btn-disable-border : @border-color-base; @btn-padding-base : 4px 15px; @btn-font-size-lg : 14px; @btn-padding-lg : 4px 15px 5px 15px; @btn-padding-sm : 1px 7px; @btn-circle-size : 28px; @btn-circle-size-lg : 32px; @btn-circle-size-sm : 22px; // Media queries breakpoints // Extra small screen / phone @screen-xs : 480px; @screen-xs-min : @screen-xs; @screen-xs-max : (@screen-xs-min - 1); // Small screen / tablet @screen-sm : 768px; @screen-sm-min : @screen-sm; @screen-sm-max : (@screen-sm-min - 1); // Medium screen / desktop @screen-md : 992px; @screen-md-min : @screen-md; @screen-md-max : (@screen-md-min - 1); // Large screen / wide desktop @screen-lg : 1200px; @screen-lg-min : @screen-lg; @screen-lg-max : (@screen-lg-min - 1); // Layout and Grid system @grid-columns : 24; @grid-gutter-width : 0; // Container sizes @container-sm : (720px + @grid-gutter-width); @container-md : (940px + @grid-gutter-width); @container-lg : (1140px + @grid-gutter-width); // z-index list @zindex-affix : 10; @zindex-back-top : 10; @zindex-modal-mask : 1000; @zindex-modal : 1000; @zindex-notification : 1010; @zindex-message : 1010; @zindex-popover : 1030; @zindex-picker : 1050; @zindex-dropdown : 1050; @zindex-tooltip : 1060; // Form // -------------------------------- // Legend @legend-color : #999; @legend-border-color : @border-color-base; // Label @label-required-color : #f50; @label-color : @text-color; // Input @input-height-base: 28px; @input-height-lg: 32px; @input-height-sm: 22px; @input-padding-horizontal : 7px; @input-padding-vertical-base : 4px; @input-padding-vertical-sm : 1px; @input-padding-vertical-lg : 6px; @input-placeholder-color : #ccc; @input-color : @text-color; @input-border-color : @border-color-base; @input-bg : #fff; @input-hover-border-color : @primary-color; @input-focus-border-color : @primary-color; @input-disabled-bg : #f7f7f7; @form-item-margin-bottom : 24px; ================================================ FILE: web/src/style/antd-reset.global.less ================================================ @import "~antd/lib/style/themes/default.less"; @import "./constant.less"; @import "~antd/lib/style/core/index.less"; @import "~antd/lib/style/components.less"; // .ant-input { // border-radius: 0; // height: @form-input-height; // } // .has-error .ant-input:focus { // box-shadow: 0 0 0 2px rgba(224, 21, 21, 0.2); // border-color: #e01515; // } // .ant-menu-inline > .ant-menu-item{ // font-size: @font-size-reg; // line-height: 44px; // height: 44px; // } // .ant-form-explain { // margin-top:5px; // } // // 图片上传 // .ant-upload-list-picture-card .ant-upload-list-item { // width:87px; // height:87px; // } // // menu // // .ant-menu-inline > .ant-menu-item { // // border-bottom:1px solid @light-border-color; // // } // .ant-alert { // margin-bottom:0; // } // .ant-modal-body { // padding:15px 30px 20px; // } // .ant-checkbox-wrapper + span, .ant-checkbox + span { // margin-right:0; // } // .ant-checkbox-wrapper { // margin-bottom:0; // } // // 分页插件的页码输入 // .ant-pagination-options-quick-jumper input { // border-radius: 0; // height: @form-input-height; // line-height: @form-input-height; // } // // 下拉选择的组件 // .ant-select-selection--single { // height: @form-input-height; // } // .ant-select-selection { // border-radius: 0; // } // .ant-select-selection__rendered { // height: 30px; // line-height: 30px; // } // .ant-select-dropdown { // border-radius: 0; // } // .ant-pagination-prev, .ant-pagination-next, .ant-pagination-jump-prev, .ant-pagination-jump-next { // height: @form-input-height; // line-height: @form-input-height; // min-width: @form-input-height; // } // .ant-pagination-prev a:after, .ant-pagination-next a:after { // height: 28px; // line-height: 28px; // } // .ant-pagination-item { // height: @form-input-height; // line-height: @form-input-height; // min-width: @form-input-height; // } ================================================ FILE: web/src/style/common.less ================================================ @import './constant.less'; @import './animate.less'; body { line-height: 1.5; font-size: @font-size-reg; position: relative; font-family: "PingFangSC-Regular", "Helvetica Neue", Helvetica, "Hiragino Sans GB", "Microsoft YaHei", Arial, sans-serif !important; word-wrap: break-word; min-height: 500px; } .sectionTitle { display: block; font-size: @font-size-reg; border-left: 3px solid @primary-color; line-height: 1; padding-left: 5px; } .whiteSpace10 { display: block; width: 100%; height: 10px; } .whiteSpace20 { display: block; width: 100%; height: 20px; } .whiteSpace30 { display: block; width: 100%; height: 30px; } .left { float: left; } .right { float: right; } .topAlign { display: flex; justify-content: flex-start; align-items: flex-start; div { line-height: 1; } } :global { // .ant-btn { // min-width: 100px; // } } .relativeWrapper { position: relative; width: 100%; height: 100%; } ================================================ FILE: web/src/style/constant.less ================================================ @import "./antd-constant"; @primary-color: #108ee9; @default-color: #1a1a1a; @tip-color: #999; @hint-color: #ddd; @error-color: #EA2020; @step-head-color:#666666; @background-color: #eaeaea; @light-border-color: #e0e0e0; @warn-color: #fac450; @number-color: #FF6600; @font-size-large: 24px; @font-size-big: 18px; @font-size-base: 12px; @text-color :#1a1a1a; @light-background-color: #f6f6f6; @ok-color: #2FD000; @active-color: #2196f3; @info-bkg-color: #ecf6fd; @success-color: #87d068; @info-color: #2db7f5; @opacity-background-color: rgba(102, 102, 102, 0.37); @font-size-reg: 14px; @font-size-sm: 13px; @font-size-xs: 12px; @border-radius-base: 4px; @border-radius-sm: 4px; @link-color: #108ee9; @notice-success-bacolor: #e5f5ff; @notice-success-borcolor: #cbd7e3; @form-input-height: 32px; @left-menu-background-color: #2E303C; @left-menu-color: rgba(255, 255, 255, 0.5); @left-menu-font-size: 16px; @middlepanel-font-size: 48px; @top-menu-span-color: #A9A9A9; @top-menu-spliter-color: #DDD; @menu-icon-font-size: 19px; @logo-font-size: 29.48px; ================================================ FILE: web/webpack.config.js ================================================ const webpack = require('webpack'); const path = require('path'); const autoprefixer = require('autoprefixer'); const ExtractTextPlugin = require('extract-text-webpack-plugin'); const UglifyJsPlugin = webpack.optimize.UglifyJsPlugin; const extractCss = new ExtractTextPlugin('[name].css', { disable: false, allChunks: true }); // a plugin to set the environment const defineProperty = new webpack.DefinePlugin({ 'process.env': { NODE_ENV: JSON.stringify(process.env.NODE_ENV || 'test') } }); module.exports = { entry: ['whatwg-fetch', 'babel-polyfill', path.join(__dirname, './src/index.jsx')], output: { path: path.join(__dirname, 'dist'), filename: 'main.js' }, resolve: { modules: [ 'node_modules', path.join(__dirname, 'src') ], extensions: ['.', '.js', '.jsx'] }, module: { rules: [{ test: /\.js$/, exclude: /node_modules/, loader: 'babel-loader', options: { presets: ['es2015', 'stage-0'] } }, { test: /\.jsx$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['es2015', 'stage-0', 'react'], plugins: [['import', { libraryName: 'antd', style: true }]] } } }, { test: function (filePath) { return (/antd\/.*\.less$/.test(filePath) || /\.global\.less$/.test(filePath)); }, use: ExtractTextPlugin.extract({use: 'css-loader!postcss-loader!less-loader'}) }, { test: function (filePath) { return (/\.less$/.test(filePath) && !/\.global\.less$/.test(filePath) && !/antd\/.*\.less$/.test(filePath)); }, use: ExtractTextPlugin.extract({use: 'css-loader?modules&localIdentName=[local]___[hash:base64:5]!postcss-loader!less-loader'}) }, { test: /\.css$/, use: ExtractTextPlugin.extract({use:'css-loader'}) }, { test: /\.png(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=image/png' } }, { test: /\.woff(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=application/font-woff' } }, { test: /\.woff2(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=application/font-woff' } }, { test: /\.ttf(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=application/octet-stream' } }, { test: /\.eot(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=application/octet-stream' } }, { test: /font\.svg(\?v=\d+\.\d+\.\d+)?$/, use: { loader: 'url-loader?limit=10000&mimetype=image/svg+xml' } }] }, plugins: [ extractCss, defineProperty, new UglifyJsPlugin() ], stats: { children: false } };