diff --git a/blog/zh-cn/2.3.0-beta-release.md b/blog/zh-cn/2.3.0-beta-release.md new file mode 100644 index 00000000000..8d2f3ee009a --- /dev/null +++ b/blog/zh-cn/2.3.0-beta-release.md @@ -0,0 +1,163 @@ +--- +title: Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用 +keywords: 2.3.0 +description: Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用 +date: 2023-10-24 +--- +# Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用 + +## 新版本发布 + +经过4个多月社区的大量贡献,Nacos 2.3.0版本终于进入的Beta发布阶段,在经历了2.2.X的多个安全风险修复的版本后,又一次迎来了功能性上的更新版本。 + +### Nacos 反脆弱插件 + +2.2.0版本加入ALPHA版本的反脆弱插件后,社区经过多次重构和修改,提升抽象度和稳定性,终于在2.3.0版本中正式公布了Nacos的反脆弱插件,用于避免Nacos集群在大量请求和连接时导致的容量问题而引发更大规模的故障。 + +默认反脆弱实现插件的使用以及自定义反脆弱插件的开发可参考文档[反脆弱插件](/zh-cn/docs/v2/plugin/control-plugin.html). + +### Nacos 配置变更插件 + +Nacos配置管理作为Nacos的核心功能,一直受到开发人员、运维人员和审计人员的关注,不少运维人员和审计人员向社区提出诉求,希望Nacos能够提供配置变更的审计、变更通知等功能;同时开发人员也希望Nacos提供一定的格式校验功能,避免修改配置时出现错误导致应用故障。 + +由于Nacos配置管理本身对配置内容是不感知的,且各个用户公司的审计、通知系统也各不相同,因此Nacos为了能够更广阔的适配各个用户的诉求,在去年的开源之夏中制定了配置变更插件的课题,希望以插件化的形式,来满足对配置变更过程的介入。 + +经过同学和导师长期的努力,该插件在2.3.0版本发布,用户可以开发自定义插件,在配置变更前和变更完成后,分别对接公司的审计系统和通知系统,并在变更前进行更多的校验操作,如格式校验,配置名字的规范化等。同时Nacos社区也提供了默认插件的实现,可接入webhook进行配置变更的通知以及导入配置的文件名校验等;详情可参考文档[配置变更插件](/zh-cn/docs/v2/plugin/config-change-plugin.html) + +### Nacos 请求参数校验 + +2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。 + +在2.3.0版本中,Nacos明确了参数校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据校验规则对客户端向服务端发送的请求进行校验。用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。 + +详细情况可参考文档[参数校验规则](/zh-cn/docs/v2/guide/user/parameters-check.html) + +### Nacos 能力协商 + +随着Nacos功能越来越多,版本越来越多,客户端和服务端之间的兼容性愈发重要,如果继续保持尝试性的兼容,只会兼容能力愈发的困难。因此Nacos社区在去年开源之夏中,发布了课题,通过增加客户端和服务端之间的能力协商机制,在客户端连接到服务端时,让连接双方知道对方分别支持的功能,在支持对应功能的情况下,开启对应功能。避免通过尝试而增加通信开销。 + +在2.3.0版本中,该能力也被集成进来,为之后Nacos3.0支持更多优化功能提供基础。 + +### 其他重要改动 + +除了上述改动,Nacos2.3.0版本还支持了:客户端异常的metrics、grpc长连接的SSL功能、关闭开源控制台等功能,更多细节可以查看变更列表,欢迎试用: + +```markdown +## feature +[#5698] Support nacos control plugin. +[#8458] Support ability negotiations between server and clients. +[#8460] Support config change hook plugin. +[#10117] Support metrics for nacos client request server exception. +[#10150] Support SSL for grpc connection. +[#10223] Support auto build instance id when client request instance id is null. +[#10288] Support get more module state and switches in console. +[#10734] Support validate most of request parameters. +[#10774] Support toml format for configuration in console ui. +[#10831] Support batch deregister instances for service. +[#10971] Support disable console ui and support add guide information. + +## Enhancement&Refactor +[#6819] Add page size selector in service details page. +[#8107][#9109][#10169][#10176] Enhance hint when console ui session expired for default auth plugin. +[#9085] Add the Reachability Metadata required by native-image. +[#9821] Enhance datasource plugin to make more datasource implementation easier. +[#9881] Enhance configuration page to supports folding when editing configuration. +[#10067] Enhance Windows compatibility for configuration snapshot. +[#10155] Enhance hints for grpc request when request timeout. +[#10343] Use CMS as default GC when jdk less 9. +[#10361] Refactor module switches to make only load specified module but not only close in console ui. +[#10520] Validate for namespace show name when create new namespace. +[#10521] Enhance the hints for `No DataSourceSet` error by validate datasource after construction. +[#10539] Enhance logs when opeation configuration failed. +[#10730] Link to v2 document for console ui. +[#10811] Enhance compatibility for colorful service healthy status in console ui. +[#10891] Support setting maximum number of push retries. +[#10930] Forward compatible old version secretKey for default auth plugin. +[#11129] Remove the namespace information from the node list page. +[#11231] Optimize the handleSpringBinder method in PropertiesUtil. + +## BugFix +[#10056] Fix loss revision of client for distro sync. +[#10128] Fix wrong judgement in raft stateMachine. +[#10149] Fix dead lock on sending connection reset request on server over limit. +[#10271] Fix nacos-client failover switch file path. +[#10318] Fix import configuration problem. +[#10347] Fix only admin role user can register service into default namespace when enabled default auth plugin. +[#10406] Fix jraft install leader snapshot error after disconnection. +[#10427] Fix nacos client no response when handle server request with exception. +[#10464] Fix NPE when concurrent operations for client. +[#10470] Fix some missed i18n for console ui. +[#10509] Fix out data connection not be disconnect problem. +[#10548] Fix switch domain might not load snapshot after restart. +[#10556] Fix index loss for client and service in extreme scenarios. +[#10583] Fix some new API loss auth check. +[#10585] Fix selectInstances and selectOneHealthyInstance methods will not subscribe service problem. +[#10593] Fix invalid create `file:` dir under `nacos.home`. +[#10598] Fix nacos-client not random get server address when using address. +[#10606] Fix memory leak for nacos client when user create and shutdown client frequently. +[#10657] Fix NPE when using derby datasource for cluster mode. +[#10935] Fix startsWith judgement wrong when ignoreCase is true. +[#11056] Fix Batch register count size wrong, when batch register sereval time. +[#11059] Fix RPC_CLIENT_TLS_PROTOCOLS setting error. +[#11192] Fix batchRegisterInstance not recalculate revision prblem. +[#11197] Fix frequent do query service when hit protect empty. + +## Dependency +[#7698] Remove httpasyncclient version dependency management to avoid version conflicts. +[#10416] Upgrade console yaml editor. +[#10648] Optimize Guava Dependency. +[#10893] Upgrade spring boot to 2.7.15. +[#11199] Upgrade grpc version to 1.57.2. +``` + +## 开源之夏OSPP 2023以及编程夏令营GLCC 2023 + +经过夏天如火如荼的努力,Nacos今年的开源之夏也接近尾声,各项目进展稳步推进,均取得了不小的进展。 + +### 开源之夏 OSPP 2023 + +Nacos在2023年度的[开源之夏](https://summer-ospp.ac.cn/org/orgdetail/ab188e59-fab8-468f-bc89-bdc2bd8b5e64?lang=zh) 中,发布了9个项目,其中6个项目成功匹配同学,并基本完成了项目课题,其中不乏大量社区期待的新增功能和优化,将在后续版本中加入。 + +入选项目列表 + +- 客户端订阅者合并和数据选择功能 +- 为nacos-spring-boot-starter支持适配spring boot 3 +- ConfigMap到Nacos配置中心自动同步工具 +- 客户端可观测性指标建设 +- 支持Nacos注册中心的模糊订阅能力 +- 实现其他部分xDS协议 + +感谢导师和同学们的辛苦付出,也感谢主办方中科院软件所提供的优秀平台和活动。 + +### Nacos 荣获编程夏令营 GLCC 2023优秀社区 + +Nacos在2023年度的[编程夏令营](https://www.gitlink.org.cn/glcc/2023/projects) 申报了一个Nacos3.0中呼声较高的一个功能作为课题 -- 支持分布式锁。 目前已在导师和同学的共同努力下,初步完成了功能,预计会在Nacos3.0的Alpha版本中在社区公布。 + +由于导师和同学的努力,Nacos社区荣获CCF Gitlink 颁发的开源编程夏令营的优秀社区奖,感谢主办方Gitlink为社区提供的平台和活动。 + +|![奖杯](https://cdn.nlark.com/yuque/0/2023/jpeg/1577777/1698197262198-f88bab75-a8f8-41f8-9675-9bfa916642c7.jpeg#averageHue=%23786456&clientId=ueeb4d907-433f-4&from=drop&id=udede56df&originHeight=3024&originWidth=4032&originalType=binary&ratio=2&rotation=0&showTitle=false&size=877771&status=done&style=none&taskId=u6da77a56-e076-4a34-ae9b-c1e8afb010b&title=)|![现场](https://cdn.nlark.com/yuque/0/2023/jpeg/1577777/1698197100242-e440717c-80ec-4825-93c1-2daca1f289c4.jpeg#averageHue=%236664b2&clientId=u14301240-344a-4&from=drop&id=ua9e02282&originHeight=854&originWidth=1280&originalType=binary&ratio=2&rotation=0&showTitle=false&size=204483&status=done&style=none&taskId=u7637d2eb-a17e-431e-a637-231a61f9e96&title=)| +|----|----| + +## 展望 +### 2.X 后续计划 + +从2021年3月 2.0.0正式版发布至今,2.X版本已经走了接近2年时间,如今2.3.0版本发布,完成了大部分功能的插件化提炼,在之后的2.3.X版本中,会主要对当前版本的问题进行修复,并做出小范围的功能优化。同时对于2.4.0版本,会作为一个Nacos3.0的过度版本,对大量代码进行优化重构,在提升稳定性、健壮性的同时,提升易用性和可观测性,向Nacos3.0版本平稳过度。 + +### 3.0 计划 + +Nacos社区同时也开启了关于[Nacos3.0](https://mp.weixin.qq.com/s/8UwwD_WxSJINP8Qr_1wogg) 的畅想和规划,Nacos将会从统一控制面、支持国产化、存储计算分离等方向进一步演进Nacos的功能和架构,欢迎社区积极参与到新版本的建设中。 + +![image.png](https://cdn.nlark.com/yuque/0/2023/png/1577777/1698198629123-af9f1216-f996-4ac2-81bf-436048823d21.png) + +![image.png](https://cdn.nlark.com/yuque/0/2022/png/1577777/1660125280551-a2e881fe-d25e-4ebb-a28f-8e56683deef1.png#clientId=uf10cb19a-105c-4&crop=0&crop=0&crop=1&crop=1&from=url&id=Z9to1&margin=%5Bobject%20Object%5D&name=image.png&originHeight=794&originWidth=1650&originalType=binary&ratio=1&rotation=0&showTitle=false&size=185821&status=done&style=none&taskId=u63849e10-1dae-45cb-b559-04d106ebe86&title=#crop=0&crop=0&crop=1&crop=1&id=rUihF&originHeight=794&originWidth=1650&originalType=binary&ratio=1&rotation=0&showTitle=false&status=done&style=none&title=) + +## About Nacos + +Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。 + +Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。 Nacos 是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施。 + +最后欢迎大家扫码加入Nacos社区群 + +![image.png](https://cdn.nlark.com/yuque/0/2023/png/1577777/1679276899363-83081d59-67c6-4501-9cf8-0d84ba7c6d7e.png#averageHue=%23c1c2c2&clientId=u9dfeac18-3281-4&from=paste&height=551&id=ubcf45e51&name=image.png&originHeight=1102&originWidth=854&originalType=binary&ratio=2&rotation=0&showTitle=false&size=155261&status=done&style=none&taskId=ud6bea1fe-b003-441b-a810-84435d2aeff&title=&width=427) + diff --git a/build/documentation.js b/build/documentation.js index 248085c8f58..cf03cc5ce9f 100644 --- a/build/documentation.js +++ b/build/documentation.js @@ -3,4 +3,4 @@ Licensed under the MIT License (MIT), see http://jedwatson.github.io/classnames */ -!function(){"use strict";function n(){for(var e=[],t=0;t1&&void 0!==arguments[1]?arguments[1]:t.key)+arguments[2]})}},{key:"key",get:function(){return this.childDescriptor.key}},{key:"parentNotation",get:function(){return this.parentKlass.constructor.name+"#"+this.parentPropertySignature}},{key:"childNotation",get:function(){return this.childKlass.constructor.name+"#"+this.childPropertySignature}},{key:"parentTopic",get:function(){return this._getTopic(this.parentDescriptor)}},{key:"childTopic",get:function(){return this._getTopic(this.childDescriptor)}},{key:"parentPropertySignature",get:function(){return this._extractTopicSignature(this.parentTopic)}},{key:"childPropertySignature",get:function(){return this._extractTopicSignature(this.childTopic)}}]),h(e,[{key:"assert",value:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"";!0!==e&&this.error("{child} does not properly override {parent}"+t)}},{key:"error",value:function(e){var t=this;throw e=e.replace("{parent}",function(e){return t.parentNotation}).replace("{child}",function(e){return t.childNotation}),new SyntaxError(e)}}]),e}(),g=[function(e){return e.toLowerCase()},function(e){return e.toUpperCase()},function(e){return e+"s"},function(e){return e.slice(0,-1)},function(e){return e.slice(1,e.length)}]},function(e,t,n){"use strict";function r(e,t,r,c){var s=l(c,3),f=s[0],d=void 0===f?null:f,p=s[1],h=void 0!==p&&p,m=s[2],y=void 0===m?u:m;if(!o.__enabled)return o.__warned||(y.warn("console.profile is not supported. All @profile decorators are disabled."),o.__warned=!0),r;var g=r.value;if(null===d&&(d=e.constructor.name+"."+t),"function"!=typeof g)throw new SyntaxError("@profile can only be used on functions, not: "+g);return a({},r,{value:function(){var e=Date.now(),t=n.i(i.c)(this);(!0===h&&!t.profileLastRan||!1===h||"number"==typeof h&&e-t.profileLastRan>h||"function"==typeof h&&h.apply(this,arguments))&&(y.profile(d),t.profileLastRan=e);try{return g.apply(this,arguments)}finally{y.profileEnd(d)}}})}function o(){for(var e=arguments.length,t=Array(e),o=0;o1){if(i=e({path:"/"},r.defaults,i),"number"==typeof i.expires){var l=new Date;l.setMilliseconds(l.getMilliseconds()+864e5*i.expires),i.expires=l}i.expires=i.expires?i.expires.toUTCString():"";try{a=JSON.stringify(o),/^[\{\[]/.test(a)&&(o=a)}catch(e){}o=n.write?n.write(o,t):encodeURIComponent(String(o)).replace(/%(23|24|26|2B|3A|3C|3E|3D|2F|3F|40|5B|5D|5E|60|7B|7D|7C)/g,decodeURIComponent),t=encodeURIComponent(String(t)),t=t.replace(/%(23|24|26|2B|5E|60|7C)/g,decodeURIComponent),t=t.replace(/[\(\)]/g,escape);var u="";for(var c in i)i[c]&&(u+="; "+c,!0!==i[c]&&(u+="="+i[c]));return document.cookie=t+"="+o+u}t||(a={});for(var s=document.cookie?document.cookie.split("; "):[],f=/(%[0-9A-Z]{2})+/g,d=0;d=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n}function i(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t1?t-1:0),r=1;r0&&(H.apply(void 0,t),L&&((0,N.default)("called ga('arguments');"),(0,N.default)("with arguments: "+JSON.stringify(t)))),window.ga}function f(e,t){return e?"object"!==(void 0===e?"undefined":w(e))?void(0,A.default)("Expected `fieldsObject` arg to be an Object"):(0===Object.keys(e).length&&(0,A.default)("empty `fieldsObject` given to .set()"),l(t,"set",e),void(L&&((0,N.default)("called ga('set', fieldsObject);"),(0,N.default)("with fieldsObject: "+JSON.stringify(e))))):void(0,A.default)("`fieldsObject` is required in .set()")}function d(e,t){l(t,"send",e),L&&((0,N.default)("called ga('send', fieldObject);"),(0,N.default)("with fieldObject: "+JSON.stringify(e)),(0,N.default)("with trackers: "+JSON.stringify(t)))}function p(e,t,n){if(!e)return void(0,A.default)("path is required in .pageview()");var r=(0,j.default)(e);if(""===r)return void(0,A.default)("path cannot be an empty string in .pageview()");var o={};if(n&&(o.title=n),"function"==typeof s&&(l(t,"send",b({hitType:"pageview",page:r},o)),L)){(0,N.default)("called ga('send', 'pageview', path);");var i="";n&&(i=" and title: "+n),(0,N.default)("with path: "+r+i)}}function h(e,t){if(!e)return void(0,A.default)("modalName is required in .modalview(modalName)");var n=(0,E.default)((0,j.default)(e));if(""===n)return void(0,A.default)("modalName cannot be an empty string or a single / in .modalview()");if("function"==typeof s){var r="/modal/"+n;l(t,"send","pageview",r),L&&((0,N.default)("called ga('send', 'pageview', path);"),(0,N.default)("with path: "+r))}}function m(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=e.category,n=e.variable,r=e.value,o=e.label,i=arguments[1];if("function"==typeof s){if(!t||!n||!r||"number"!=typeof r)return void(0,A.default)("args.category, args.variable AND args.value are required in timing() AND args.value has to be a number");var l={hitType:"timing",timingCategory:a(t),timingVar:a(n),timingValue:r};o&&(l.timingLabel=a(o)),d(l,i)}}function y(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments[1],n=e.category,r=e.action,i=e.label,l=e.value,u=e.nonInteraction,c=e.transport,f=o(e,["category","action","label","value","nonInteraction","transport"]);if("function"==typeof s){if(!n||!r)return void(0,A.default)("args.category AND args.action are required in event()");var p={hitType:"event",eventCategory:a(n),eventAction:a(r)};i&&(p.eventLabel=a(i)),void 0!==l&&("number"!=typeof l?(0,A.default)("Expected `args.value` arg to be a Number."):p.eventValue=l),void 0!==u&&("boolean"!=typeof u?(0,A.default)("`args.nonInteraction` must be a boolean."):p.nonInteraction=u),void 0!==c&&("string"!=typeof c?(0,A.default)("`args.transport` must be a string."):(-1===["beacon","xhr","image"].indexOf(c)&&(0,A.default)("`args.transport` must be either one of these values: `beacon`, `xhr` or `image`"),p.transport=c)),Object.keys(f).filter(function(e){return"dimension"===e.substr(0,"dimension".length)}).forEach(function(e){p[e]=f[e]}),Object.keys(f).filter(function(e){return"metric"===e.substr(0,"metric".length)}).forEach(function(e){p[e]=f[e]}),d(p,t)}}function g(e,t){var n=e.description,r=e.fatal;if("function"==typeof s){var o={hitType:"exception"};n&&(o.exDescription=a(n)),void 0!==r&&("boolean"!=typeof r?(0,A.default)("`args.fatal` must be a boolean."):o.exFatal=r),d(o,t)}}function v(e,t,n){if("function"!=typeof t)return void(0,A.default)("hitCallback function is required");if("function"==typeof s){if(!e||!e.label)return void(0,A.default)("args.label is required in outboundLink()");var r={hitType:"event",eventCategory:"Outbound",eventAction:"Click",eventLabel:a(e.label)},o=!1,i=function(){o=!0,t()},l=setTimeout(i,250),u=function(){clearTimeout(l),o||t()};r.hitCallback=u,d(r,n)}else setTimeout(t,0)}Object.defineProperty(t,"__esModule",{value:!0}),t.testModeAPI=t.OutboundLink=t.plugin=void 0;var b=Object.assign||function(e){for(var t=1;t0&&t+e.length!==n.length&&e.search(a)>-1&&":"!==n.charAt(t-2)&&("-"!==n.charAt(t+e.length)||"-"===n.charAt(t-1))&&n.charAt(t-1).search(/[^\s-]/)<0?e.toLowerCase():e.substr(1).search(/[A-Z]|\../)>-1?e:e.charAt(0).toUpperCase()+e.substr(1)})}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r;var o=n(1),i=function(e){return e&&e.__esModule?e:{default:e}}(o),a=/^(a|an|and|as|at|but|by|en|for|if|in|nor|of|on|or|per|the|to|vs?\.?|via)$/i},function(e,t,n){"use strict";function r(e){return"/"===e.substring(0,1)?e.substring(1):e}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default=function(e){!function(e,t,n,r,o,i,a){e.GoogleAnalyticsObject=o,e[o]=e[o]||function(){(e[o].q=e[o].q||[]).push(arguments)},e[o].l=1*new Date,i=t.createElement(n),a=t.getElementsByTagName(n)[0],i.async=1,i.src=r,a.parentNode.insertBefore(i,a)}(window,document,"script",e&&e.gaAddress?e.gaAddress:"https://www.google-analytics.com/analytics.js","ga")}},function(e,t,n){"use strict";function r(e){console.info("[react-ga]",e)}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=t.gaCalls=[];t.default={calls:r,ga:function(){for(var e=arguments.length,t=Array(e),n=0;n=Math.floor(a)&&s=Math.floor(l),p=n.getActiveLink();d&&(o===p&&n.setActiveLink(void 0),e.props.hashSpy&&w.default.getHash()===o&&w.default.changeHash(),e.props.spy&&e.state.active&&(e.setState({active:!1}),e.props.onSetInactive&&e.props.onSetInactive(o,i))),!f||p===o&&!1!==e.state.active||(n.setActiveLink(o),e.props.hashSpy&&w.default.changeHash(o),e.props.spy&&(e.setState({active:!0}),e.props.onSetActive&&e.props.onSetActive(o,i)))}}};return r.propTypes=k,r.defaultProps={offset:0},r}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=n(51),o=function(e){return e&&e.__esModule?e:{default:e}}(r),i=n(40),a=function(e){return(0,o.default)(e,66)},l={spyCallbacks:[],spySetState:[],scrollSpyContainers:[],mount:function(e){if(e){var t=a(function(t){l.scrollHandler(e)});l.scrollSpyContainers.push(e),(0,i.addPassiveEventListener)(e,"scroll",t)}},isMounted:function(e){return-1!==l.scrollSpyContainers.indexOf(e)},currentPositionY:function(e){if(e===document){var t=void 0!==window.pageXOffset,n="CSS1Compat"===(document.compatMode||"");return t?window.pageYOffset:n?document.documentElement.scrollTop:document.body.scrollTop}return e.scrollTop},scrollHandler:function(e){(l.scrollSpyContainers[l.scrollSpyContainers.indexOf(e)].spyCallbacks||[]).forEach(function(t){return t(l.currentPositionY(e))})},addStateHandler:function(e){l.spySetState.push(e)},addSpyHandler:function(e,t){var n=l.scrollSpyContainers[l.scrollSpyContainers.indexOf(t)];n.spyCallbacks||(n.spyCallbacks=[]),n.spyCallbacks.push(e),e(l.currentPositionY(t))},updateStates:function(){l.spySetState.forEach(function(e){return e()})},unmount:function(e,t){l.scrollSpyContainers.forEach(function(e){return e.spyCallbacks&&e.spyCallbacks.length&&e.spyCallbacks.splice(e.spyCallbacks.indexOf(t),1)}),l.spySetState&&l.spySetState.length&&l.spySetState.splice(l.spySetState.indexOf(e),1),document.removeEventListener("scroll",l.scrollHandler)},update:function(){return l.scrollSpyContainers.forEach(function(e){return l.scrollHandler(e)})}};t.default=l},function(e,t){},function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}Object.defineProperty(t,"__esModule",{value:!0});var o=Object.assign||function(e){for(var t=1;t=o.duration?1:t(o.progress/o.duration),o.currentPositionY=o.startPositionY+Math.ceil(o.deltaTop*o.percent),o.containerElement&&o.containerElement!==document&&o.containerElement!==document.body?o.containerElement.scrollTop=o.currentPositionY:window.scrollTo(0,o.currentPositionY),o.percent<1){var i=e.bind(null,t,n);return void m.call(window,i)}f.default.registered.end&&f.default.registered.end(o.to,o.target,o.currentPositionY)},w=function(e){e.data.containerElement=e?e.containerId?document.getElementById(e.containerId):e.container&&e.container.nodeType?e.container:document:null},k=function(e,t,n,r){if(t.data=t.data||y(),window.clearTimeout(t.data.delayTimeout),c.default.subscribe(function(){t.data.cancel=!0}),w(t),t.data.start=null,t.data.cancel=!1,t.data.startPositionY=g(t),t.data.targetPositionY=t.absolute?e:e+t.data.startPositionY,t.data.startPositionY===t.data.targetPositionY)return void(f.default.registered.end&&f.default.registered.end(t.data.to,t.data.target,t.data.currentPositionY));t.data.deltaTop=Math.round(t.data.targetPositionY-t.data.startPositionY),t.data.duration=p(t.duration)(t.data.deltaTop),t.data.duration=isNaN(parseFloat(t.data.duration))?1e3:parseFloat(t.data.duration),t.data.to=n,t.data.target=r;var o=d(t),i=b.bind(null,o,t);if(t&&t.delay>0)return void(t.data.delayTimeout=window.setTimeout(function(){m.call(window,i)},t.delay));m.call(window,i)},O=function(e){return e=o({},e),e.data=e.data||y(),e.absolute=!0,e},_=function(e){k(0,O(e))},E=function(e,t){k(e,O(t))},S=function(e){e=O(e),w(e),k(v(e),e)},j=function(e,t){t=O(t),w(t),k(g(t)+e,t)};t.default={animateTopScroll:k,getAnimationType:d,scrollToTop:_,scrollToBottom:S,scrollTo:E,scrollMore:j}},function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}function o(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function i(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t}function a(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}Object.defineProperty(t,"__esModule",{value:!0});var l=Object.assign||function(e){for(var t=1;t=0;r--){var o=e[r];"."===o?e.splice(r,1):".."===o?(e.splice(r,1),n++):n&&(e.splice(r,1),n--)}if(t)for(;n--;n)e.unshift("..");return e}function r(e,t){if(e.filter)return e.filter(t);for(var n=[],r=0;r=-1&&!o;i--){var a=i>=0?arguments[i]:e.cwd();if("string"!=typeof a)throw new TypeError("Arguments to path.resolve must be strings");a&&(t=a+"/"+t,o="/"===a.charAt(0))}return t=n(r(t.split("/"),function(e){return!!e}),!o).join("/"),(o?"/":"")+t||"."},t.normalize=function(e){var o=t.isAbsolute(e),i="/"===a(e,-1);return e=n(r(e.split("/"),function(e){return!!e}),!o).join("/"),e||o||(e="."),e&&i&&(e+="/"),(o?"/":"")+e},t.isAbsolute=function(e){return"/"===e.charAt(0)},t.join=function(){var e=Array.prototype.slice.call(arguments,0);return t.normalize(r(e,function(e,t){if("string"!=typeof e)throw new TypeError("Arguments to path.join must be strings");return e}).join("/"))},t.relative=function(e,n){function r(e){for(var t=0;t=0&&""===e[n];n--);return t>n?[]:e.slice(t,n-t+1)}e=t.resolve(e).substr(1),n=t.resolve(n).substr(1);for(var o=r(e.split("/")),i=r(n.split("/")),a=Math.min(o.length,i.length),l=a,u=0;u-1?t:e}function p(e,t){t=t||{};var n=t.body;if(e instanceof p){if(e.bodyUsed)throw new TypeError("Already read");this.url=e.url,this.credentials=e.credentials,t.headers||(this.headers=new o(e.headers)),this.method=e.method,this.mode=e.mode,n||null==e._bodyInit||(n=e._bodyInit,e.bodyUsed=!0)}else this.url=String(e);if(this.credentials=t.credentials||this.credentials||"omit",!t.headers&&this.headers||(this.headers=new o(t.headers)),this.method=d(t.method||this.method||"GET"),this.mode=t.mode||this.mode||null,this.referrer=null,("GET"===this.method||"HEAD"===this.method)&&n)throw new TypeError("Body not allowed for GET or HEAD requests");this._initBody(n)}function h(e){var t=new FormData;return e.trim().split("&").forEach(function(e){if(e){var n=e.split("="),r=n.shift().replace(/\+/g," "),o=n.join("=").replace(/\+/g," ");t.append(decodeURIComponent(r),decodeURIComponent(o))}}),t}function m(e){var t=new o;return e.replace(/\r?\n[\t ]+/g," ").split(/\r?\n/).forEach(function(e){var n=e.split(":"),r=n.shift().trim();if(r){var o=n.join(":").trim();t.append(r,o)}}),t}function y(e,t){t||(t={}),this.type="default",this.status=void 0===t.status?200:t.status,this.ok=this.status>=200&&this.status<300,this.statusText="statusText"in t?t.statusText:"OK",this.headers=new o(t.headers),this.url=t.url||"",this._initBody(e)}if(!e.fetch){var g={searchParams:"URLSearchParams"in e,iterable:"Symbol"in e&&"iterator"in Symbol,blob:"FileReader"in e&&"Blob"in e&&function(){try{return new Blob,!0}catch(e){return!1}}(),formData:"FormData"in e,arrayBuffer:"ArrayBuffer"in e};if(g.arrayBuffer)var v=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],b=function(e){return e&&DataView.prototype.isPrototypeOf(e)},w=ArrayBuffer.isView||function(e){return e&&v.indexOf(Object.prototype.toString.call(e))>-1};o.prototype.append=function(e,r){e=t(e),r=n(r);var o=this.map[e];this.map[e]=o?o+","+r:r},o.prototype.delete=function(e){delete this.map[t(e)]},o.prototype.get=function(e){return e=t(e),this.has(e)?this.map[e]:null},o.prototype.has=function(e){return this.map.hasOwnProperty(t(e))},o.prototype.set=function(e,r){this.map[t(e)]=n(r)},o.prototype.forEach=function(e,t){for(var n in this.map)this.map.hasOwnProperty(n)&&e.call(t,this.map[n],n,this)},o.prototype.keys=function(){var e=[];return this.forEach(function(t,n){e.push(n)}),r(e)},o.prototype.values=function(){var e=[];return this.forEach(function(t){e.push(t)}),r(e)},o.prototype.entries=function(){var e=[];return this.forEach(function(t,n){e.push([n,t])}),r(e)},g.iterable&&(o.prototype[Symbol.iterator]=o.prototype.entries);var k=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];p.prototype.clone=function(){return new p(this,{body:this._bodyInit})},f.call(p.prototype),f.call(y.prototype),y.prototype.clone=function(){return new y(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new o(this.headers),url:this.url})},y.error=function(){var e=new y(null,{status:0,statusText:""});return e.type="error",e};var O=[301,302,303,307,308];y.redirect=function(e,t){if(-1===O.indexOf(t))throw new RangeError("Invalid status code");return new y(null,{status:t,headers:{location:e}})},e.Headers=o,e.Request=p,e.Response=y,e.fetch=function(e,t){return new Promise(function(n,r){var o=new p(e,t),i=new XMLHttpRequest;i.onload=function(){var e={status:i.status,statusText:i.statusText,headers:m(i.getAllResponseHeaders()||"")};e.url="responseURL"in i?i.responseURL:e.headers.get("X-Request-URL");var t="response"in i?i.response:i.responseText;n(new y(t,e))},i.onerror=function(){r(new TypeError("Network request failed"))},i.ontimeout=function(){r(new TypeError("Network request failed"))},i.open(o.method,o.url,!0),"include"===o.credentials?i.withCredentials=!0:"omit"===o.credentials&&(i.withCredentials=!1),"responseType"in i&&g.blob&&(i.responseType="blob"),o.headers.forEach(function(e,t){i.setRequestHeader(t,e)}),i.send(void 0===o._bodyInit?null:o._bodyInit)})},e.fetch.polyfill=!0}}("undefined"!=typeof self?self:this)},function(e,t,n){(function(t){function n(e,t,n){function r(t){var n=m,r=y;return m=y=void 0,E=t,v=e.apply(r,n)}function i(e){return E=e,b=setTimeout(s,t),S?r(e):v}function a(e){var n=e-w,r=e-E,o=t-n;return j?O(o,g-r):o}function c(e){var n=e-w,r=e-E;return void 0===w||n>=t||n<0||j&&r>=g}function s(){var e=_();if(c(e))return f(e);b=setTimeout(s,a(e))}function f(e){return b=void 0,P&&m?r(e):(m=y=void 0,v)}function d(){void 0!==b&&clearTimeout(b),E=0,m=w=y=b=void 0}function p(){return void 0===b?v:f(_())}function h(){var e=_(),n=c(e);if(m=arguments,y=this,w=e,n){if(void 0===b)return i(w);if(j)return b=setTimeout(s,t),r(w)}return void 0===b&&(b=setTimeout(s,t)),v}var m,y,g,v,b,w,E=0,S=!1,j=!1,P=!0;if("function"!=typeof e)throw new TypeError(u);return t=l(t)||0,o(n)&&(S=!!n.leading,j="maxWait"in n,g=j?k(l(n.maxWait)||0,t):g,P="trailing"in n?!!n.trailing:P),h.cancel=d,h.flush=p,h}function r(e,t,r){var i=!0,a=!0;if("function"!=typeof e)throw new TypeError(u);return o(r)&&(i="leading"in r?!!r.leading:i,a="trailing"in r?!!r.trailing:a),n(e,t,{leading:i,maxWait:t,trailing:a})}function o(e){var t=typeof e;return!!e&&("object"==t||"function"==t)}function i(e){return!!e&&"object"==typeof e}function a(e){return"symbol"==typeof e||i(e)&&w.call(e)==s}function l(e){if("number"==typeof e)return e;if(a(e))return c;if(o(e)){var t="function"==typeof e.valueOf?e.valueOf():e;e=o(t)?t+"":t}if("string"!=typeof e)return 0===e?e:+e;e=e.replace(f,"");var n=p.test(e);return n||h.test(e)?m(e.slice(2),n?2:8):d.test(e)?c:+e}var u="Expected a function",c=NaN,s="[object Symbol]",f=/^\s+|\s+$/g,d=/^[-+]0x[0-9a-f]+$/i,p=/^0b[01]+$/i,h=/^0o[0-7]+$/i,m=parseInt,y="object"==typeof t&&t&&t.Object===Object&&t,g="object"==typeof self&&self&&self.Object===Object&&self,v=y||g||Function("return this")(),b=Object.prototype,w=b.toString,k=Math.max,O=Math.min,_=function(){return v.Date.now()};e.exports=r}).call(t,n(59))},function(e,t){function n(){throw new Error("setTimeout has not been defined")}function r(){throw new Error("clearTimeout has not been defined")}function o(e){if(s===setTimeout)return setTimeout(e,0);if((s===n||!s)&&setTimeout)return s=setTimeout,setTimeout(e,0);try{return s(e,0)}catch(t){try{return s.call(null,e,0)}catch(t){return s.call(this,e,0)}}}function i(e){if(f===clearTimeout)return clearTimeout(e);if((f===r||!f)&&clearTimeout)return f=clearTimeout,clearTimeout(e);try{return f(e)}catch(t){try{return f.call(null,e)}catch(t){return f.call(this,e)}}}function a(){m&&p&&(m=!1,p.length?h=p.concat(h):y=-1,h.length&&l())}function l(){if(!m){var e=o(a);m=!0;for(var t=h.length;t;){for(p=h,h=[];++y1)for(var n=1;n=Math.floor(a)&&f=Math.floor(l),m=n.getActiveLink();return h?(o===m&&n.setActiveLink(void 0),e.props.hashSpy&&d.getHash()===o&&d.changeHash(),e.props.spy&&e.state.active&&(e.setState({active:!1}),e.props.onSetInactive&&e.props.onSetInactive()),c.updateStates()):p&&m!==o?(n.setActiveLink(o),e.props.hashSpy&&d.changeHash(o),e.props.spy&&(e.setState({active:!0}),e.props.onSetActive&&e.props.onSetActive(o)),c.updateStates()):void 0}}};return f.propTypes=p,f.defaultProps={offset:0},f},Element:function(e){console.warn("Helpers.Element is deprecated since v1.7.0");var t=function(t){function n(e){r(this,n);var t=o(this,(n.__proto__||Object.getPrototypeOf(n)).call(this,e));return t.childBindings={domNode:null},t}return i(n,t),l(n,[{key:"componentDidMount",value:function(){if("undefined"==typeof window)return!1;this.registerElems(this.props.name)}},{key:"componentWillReceiveProps",value:function(e){this.props.name!==e.name&&this.registerElems(e.name)}},{key:"componentWillUnmount",value:function(){if("undefined"==typeof window)return!1;s.unregister(this.props.name)}},{key:"registerElems",value:function(e){s.register(e,this.childBindings.domNode)}},{key:"render",value:function(){return u.createElement(e,a({},this.props,{parentBindings:this.childBindings}))}}]),n}(u.Component);return t.propTypes={name:f.string,id:f.string},t}};e.exports=h},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=n(40),o=["mousedown","mousewheel","touchmove","keydown"];t.default={subscribe:function(e){return"undefined"!=typeof document&&o.forEach(function(t){return(0,r.addPassiveEventListener)(document,t,e)})}}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={defaultEasing:function(e){return e<.5?Math.pow(2*e,2)/2:1-Math.pow(2*(1-e),2)/2},linear:function(e){return e},easeInQuad:function(e){return e*e},easeOutQuad:function(e){return e*(2-e)},easeInOutQuad:function(e){return e<.5?2*e*e:(4-2*e)*e-1},easeInCubic:function(e){return e*e*e},easeOutCubic:function(e){return--e*e*e+1},easeInOutCubic:function(e){return e<.5?4*e*e*e:(e-1)*(2*e-2)*(2*e-2)+1},easeInQuart:function(e){return e*e*e*e},easeOutQuart:function(e){return 1- --e*e*e*e},easeInOutQuart:function(e){return e<.5?8*e*e*e*e:1-8*--e*e*e*e},easeInQuint:function(e){return e*e*e*e*e},easeOutQuint:function(e){return 1+--e*e*e*e*e},easeInOutQuint:function(e){return e<.5?16*e*e*e*e*e:1+16*--e*e*e*e*e}}},function(e,t){var n;n=function(){return this}();try{n=n||Function("return this")()||(0,eval)("this")}catch(e){"object"==typeof window&&(n=window)}e.exports=n},,,function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={"en-us":{sidemenu:[{title:"Nacos ",children:[{title:"What is Nacos",opened:!0,children:[{title:"What is Nacos",link:"/en-us/docs/what-is-nacos.html"},{title:"Concepts",link:"/en-us/docs/concepts.html"},{title:"Architecure",link:"/en-us/docs/architecture.html"}]},{title:"Quick Start",opened:!0,children:[{title:"Nacos",link:"/en-us/docs/quick-start.html"},{title:"Nacos Spring",link:"/en-us/docs/quick-start-spring.html"},{title:"Nacos Spring Boot",link:"/en-us/docs/quick-start-spring-boot.html"},{title:"Nacos Spring Cloud",link:"/en-us/docs/quick-start-spring-cloud.html"},{title:"Nacos Docker",link:"/en-us/docs/quick-start-docker.html"},{title:"Nacos with Dubbo",link:"/en-us/docs/use-nacos-with-dubbo.html"},{title:"Nacos with K8s",link:"/en-us/docs/use-nacos-with-kubernetes.html"},{title:"Nacos Sync",link:"/en-us/docs/nacos-sync.html"}]},{title:"User Guide",children:[{title:"Java SDK",link:"/en-us/docs/sdk.html"},{title:"Other Language",link:"/en-us/docs/other-language.html"},{title:"Open-API",link:"/en-us/docs/open-api.html"},{title:"Nacos Spring",link:"/en-us/docs/nacos-spring.html"},{title:"Nacos System Configurations",link:"/en-us/docs/system-configurations.html"},{title:"Authentication",link:"/en-us/docs/auth.html"},{title:"FAQ",link:"/en-us/docs/faq.html"}]},{title:"Admin Guide",children:[{title:"Deployment Guide",link:"/en-us/docs/deployment.html"},{title:"Cluster Mode Deployment",link:"/en-us/docs/cluster-mode-quick-start.html"},{title:"Management OpenAPI",link:"/en-us/docs/management-api.html"},{title:"Admin Console Guide",link:"/en-us/docs/console-guide.html"},{title:"Nacos Monitor Guide",link:"/en-us/docs/monitor-guide.html"},{title:"Nacos Config Benchmark",link:"/en-us/docs/nacos-config-benchmark.html"},{title:"Nacos Naming Benchmark",link:"/en-us/docs/nacos-naming-benchmark.html"},{title:"Data Migration to Nacos",link:"/en-us/docs/nacos-sync-use.html"}]},{title:"Contributor Guide",children:[{title:"Contribute to Nacos",link:"/en-us/docs/contributing.html"},{title:"Contributing Flow",link:"/en-us/docs/contributing-flow.html"},{title:"Pull Request Template",link:"/en-us/docs/pull-request.html"},{title:"How to report bugs",link:"/en-us/docs/how-to-reporting-bugs.html"}]},{title:"Community",children:[{title:"Community",link:"/en-us/docs/community.html"},{title:"Develop Team",link:"/en-us/docs/nacos-dev.html"}]}]}],barText:"Documentation"},"zh-cn":{sidemenu:[{title:"Nacos ",children:[{title:"Nacos是什么?",opened:!0,children:[{title:"Nacos简介",link:"/zh-cn/docs/what-is-nacos.html"},{title:"概念",link:"/zh-cn/docs/concepts.html"},{title:"架构",link:"/zh-cn/docs/architecture.html"}]},{title:"快速开始",opened:!0,children:[{title:"Nacos",link:"/zh-cn/docs/quick-start.html"},{title:"Nacos Spring",link:"/zh-cn/docs/quick-start-spring.html"},{title:"Nacos Spring Boot",link:"/zh-cn/docs/quick-start-spring-boot.html"},{title:"Nacos Spring Cloud",link:"/zh-cn/docs/quick-start-spring-cloud.html"},{title:"Nacos Docker",link:"/zh-cn/docs/quick-start-docker.html"},{title:"Nacos Dubbo",link:"/zh-cn/docs/use-nacos-with-dubbo.html"},{title:"Nacos k8s",link:"/zh-cn/docs/use-nacos-with-kubernetes.html"},{title:"Nacos Sync",link:"/zh-cn/docs/nacos-sync.html"}]},{title:"用户指南",children:[{title:"Java的SDK",link:"/zh-cn/docs/sdk.html"},{title:"其他语言的SDK",link:"/zh-cn/docs/other-language.html"},{title:"Open-API指南",link:"/zh-cn/docs/open-api.html"},{title:"Nacos Spring",link:"/zh-cn/docs/nacos-spring.html"},{title:"Nacos系统参数介绍",link:"/zh-cn/docs/system-configurations.html"},{title:"权限认证",link:"/zh-cn/docs/auth.html"},{title:"FAQ",link:"/zh-cn/docs/faq.html"}]},{title:"运维指南",children:[{title:"部署手册",link:"/zh-cn/docs/deployment.html"},{title:"集群部署说明",link:"/zh-cn/docs/cluster-mode-quick-start.html"},{title:"运维API",link:"/zh-cn/docs/management-api.html"},{title:"控制台手册",link:"/zh-cn/docs/console-guide.html"},{title:"监控手册",link:"/zh-cn/docs/monitor-guide.html"},{title:"Nacos配置模块压测报告",link:"/zh-cn/docs/nacos-config-benchmark.html"},{title:"Nacos服务发现模块压测报告",link:"/zh-cn/docs/nacos-naming-benchmark.html"},{title:"其他环境迁移到Nacos",link:"/zh-cn/docs/nacos-sync-use.html"}]},{title:"开源共建",children:[{title:"贡献源码",link:"/zh-cn/docs/contributing.html"},{title:"贡献流程",link:"/zh-cn/docs/contributing-flow.html"},{title:"pull request模板",link:"/zh-cn/docs/pull-request.html"},{title:"如何提交问题报告",link:"/zh-cn/docs/how-to-reporting-bugs.html"}]},{title:"社区",children:[{title:"社区",link:"/zh-cn/docs/community.html"},{title:"开发团队",link:"/zh-cn/docs/nacos-dev.html"}]}]}],barText:"Nacos 文档"}}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={"en-us":{sidemenu:[{title:"Nacos ",children:[{title:"What is Nacos",opened:!0,children:[{title:"What is Nacos",link:"/en-us/docs/v2/what-is-nacos.html"},{title:"Concepts",link:"/en-us/docs/v2/concepts.html"},{title:"Architecure",link:"/en-us/docs/v2/architecture.html"}]},{title:"Quick Start",opened:!0,children:[{title:"Nacos",link:"/en-us/docs/v2/quickstart/quick-start.html"},{title:"Nacos Docker",link:"/en-us/docs/v2/quickstart/quick-start-docker.html"},{title:"Nacos kubernetes",link:"/en-us/docs/v2/quickstart/quick-start-kubernetes.html"}]},{title:"Upgrading",children:[{title:"Compatibility&Usage",link:"/en-us/docs/v2/upgrading/2.0.0-compatibility.html"},{title:"Upgrading",link:"/en-us/docs/v2/upgrading/2.0.0-upgrading.html"}]},{title:"Plugin",children:[{title:"Authentication",link:"/en-us/docs/v2/plugin/auth-plugin.html"},{title:"Configuration Encryption",link:"/en-us/docs/v2/plugin/config-encryption-plugin.html"},{title:"Datasource",link:"/en-us/docs/v2/plugin/datasource-plugin.html"},{title:"Track Tracing",link:"/en-us/docs/v2/plugin/trace-plugin.html"},{title:"Custom Environment",link:"/en-us/docs/v2/plugin/custom-environment-plugin.html"}]},{title:"User Guide",children:[{title:"Java SDK",link:"/en-us/docs/v2/guide/user/sdk.html"},{title:"Other Language",link:"/en-us/docs/v2/guide/user/other-language.html"},{title:"Open-API",link:"/en-us/docs/v2/guide/user/open-api.html"},{title:"Authentication",link:"/en-us/docs/v2/guide/user/auth.html"},{title:"FAQ",link:"/en-us/docs/v2/guide/user/faq.html"}]},{title:"Admin Guide",children:[{title:"Deployment Guide",link:"/en-us/docs/v2/guide/admin/deployment.html"},{title:"Cluster Mode Deployment",link:"/en-us/docs/v2/guide/admin/cluster-mode-quick-start.html"},{title:"System Configurations",link:"/en-us/docs/v2/guide/admin/system-configurations.html"},{title:"Management OpenAPI",link:"/en-us/docs/v2/guide/admin/management-api.html"},{title:"Console Guide",link:"/en-us/docs/v2/guide/admin/console-guide.html"},{title:"Monitor Guide",link:"/en-us/docs/v2/guide/admin/monitor-guide.html"},{title:"Config Benchmark",link:"/en-us/docs/v2/guide/admin/nacos2-config-benchmark.html"},{title:"Naming Benchmark",link:"/en-us/docs/v2/guide/admin/nacos2-naming-benchmark.html"},{title:"Data Migration to Nacos",link:"/en-us/docs/v2/ecology/use-nacos-sync.html"}]},{title:"Ecology",children:[{title:"Dubbo",link:"/en-us/docs/v2/ecology/use-nacos-with-dubbo.html"},{title:"Spring",link:"/en-us/docs/v2/ecology/use-nacos-with-spring.html"},{title:"Spring Boot",link:"/en-us/docs/v2/ecology/use-nacos-with-spring-boot.html"},{title:"Spring Cloud",link:"/en-us/docs/v2/ecology/use-nacos-with-spring-cloud.html"},{title:"Nacos Sync",link:"/en-us/docs/v2/ecology/use-nacos-sync.html"},{title:"CoreDNS",link:"/en-us/docs/v2/ecology/use-nacos-with-coredns.html"}]},{title:"Contributor Guide",children:[{title:"Contribute to Nacos",link:"/en-us/docs/v2/contribution/contributing.html"},{title:"Contributing Flow",link:"/en-us/docs/v2/contribution/contributing-flow.html"},{title:"Pull Request Template",link:"/en-us/docs/v2/contribution/pull-request.html"},{title:"How to report bugs",link:"/en-us/docs/v2/contribution/how-to-reporting-bugs.html"}]},{title:"Community",children:[{title:"Community",link:"/en-us/docs/v2/community/community.html"},{title:"Develop Team",link:"/en-us/docs/v2/community/nacos-dev.html"}]}]}],barText:"Documentation"},"zh-cn":{sidemenu:[{title:"Nacos ",children:[{title:"Nacos是什么?",opened:!0,children:[{title:"Nacos简介",link:"/zh-cn/docs/v2/what-is-nacos.html"},{title:"概念",link:"/zh-cn/docs/v2/concepts.html"},{title:"架构",link:"/zh-cn/docs/v2/architecture.html"}]},{title:"快速开始",opened:!0,children:[{title:"Nacos",link:"/zh-cn/docs/v2/quickstart/quick-start.html"},{title:"Nacos Docker",link:"/zh-cn/docs/v2/quickstart/quick-start-docker.html"},{title:"Nacos kubernetes",link:"/zh-cn/docs/v2/quickstart/quick-start-kubernetes.html"}]},{title:"升级",children:[{title:"兼容性及使用",link:"/zh-cn/docs/v2/upgrading/2.0.0-compatibility.html"},{title:"升级文档",link:"/zh-cn/docs/v2/upgrading/2.0.0-upgrading.html"}]},{title:"插件",children:[{title:"鉴权",link:"/zh-cn/docs/v2/plugin/auth-plugin.html"},{title:"配置加密",link:"/zh-cn/docs/v2/plugin/config-encryption-plugin.html"},{title:"数据源",link:"/zh-cn/docs/v2/plugin/datasource-plugin.html"},{title:"轨迹追踪",link:"/zh-cn/docs/v2/plugin/trace-plugin.html"},{title:"环境变量",link:"/zh-cn/docs/v2/plugin/custom-environment-plugin.html"}]},{title:"用户指南",children:[{title:"Java的SDK",link:"/zh-cn/docs/v2/guide/user/sdk.html"},{title:"其他语言的SDK",link:"/zh-cn/docs/v2/guide/user/other-language.html"},{title:"Open-API指南",link:"/zh-cn/docs/v2/guide/user/open-api.html"},{title:"权限认证",link:"/zh-cn/docs/v2/guide/user/auth.html"},{title:"FAQ",link:"/zh-cn/docs/v2/guide/user/faq.html"}]},{title:"运维指南",children:[{title:"部署手册",link:"/zh-cn/docs/v2/guide/admin/deployment.html"},{title:"集群部署说明",link:"/zh-cn/docs/v2/guide/admin/cluster-mode-quick-start.html"},{title:"系统参数说明",link:"/zh-cn/docs/v2/guide/admin/system-configurations.html"},{title:"运维API",link:"/zh-cn/docs/v2/guide/admin/management-api.html"},{title:"控制台手册",link:"/zh-cn/docs/v2/guide/admin/console-guide.html"},{title:"监控手册",link:"/zh-cn/docs/v2/guide/admin/monitor-guide.html"},{title:"Nacos2.0配置模块压测报告",link:"/zh-cn/docs/v2/guide/admin/nacos2-config-benchmark.html"},{title:"Nacos2.0服务发现模块压测报告",link:"/zh-cn/docs/v2/guide/admin/nacos2-naming-benchmark.html"},{title:"其他环境迁移到Nacos",link:"/zh-cn/docs/v2/ecology/use-nacos-sync.html"}]},{title:"生态融合",children:[{title:"Dubbo",link:"/zh-cn/docs/v2/ecology/use-nacos-with-dubbo.html"},{title:"Spring",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring.html"},{title:"Spring Boot",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring-boot.html"},{title:"Spring Cloud",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring-cloud.html"},{title:"Nacos Sync",link:"/zh-cn/docs/v2/ecology/use-nacos-sync.html"},{title:"CoreDNS",link:"/zh-cn/docs/v2/ecology/use-nacos-with-coredns.html"}]},{title:"开源共建",children:[{title:"贡献源码",link:"/zh-cn/docs/v2/contribution/contributing.html"},{title:"贡献流程",link:"/zh-cn/docs/v2/contribution/contributing-flow.html"},{title:"pull request模板",link:"/zh-cn/docs/v2/contribution/pull-request.html"},{title:"如何提交问题报告",link:"/zh-cn/docs/v2/contribution/how-to-reporting-bugs.html"}]},{title:"社区",children:[{title:"社区",link:"/zh-cn/docs/v2/community/community.html"},{title:"开发团队",link:"/zh-cn/docs/v2/community/nacos-dev.html"}]}]}],barText:"Nacos 文档"}}},,,,function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}function o(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function i(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t}function a(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}Object.defineProperty(t,"__esModule",{value:!0});var l,u=function(){function e(e,t){for(var n=0;n1&&void 0!==arguments[1]?arguments[1]:t.key)+arguments[2]})}},{key:"key",get:function(){return this.childDescriptor.key}},{key:"parentNotation",get:function(){return this.parentKlass.constructor.name+"#"+this.parentPropertySignature}},{key:"childNotation",get:function(){return this.childKlass.constructor.name+"#"+this.childPropertySignature}},{key:"parentTopic",get:function(){return this._getTopic(this.parentDescriptor)}},{key:"childTopic",get:function(){return this._getTopic(this.childDescriptor)}},{key:"parentPropertySignature",get:function(){return this._extractTopicSignature(this.parentTopic)}},{key:"childPropertySignature",get:function(){return this._extractTopicSignature(this.childTopic)}}]),h(e,[{key:"assert",value:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"";!0!==e&&this.error("{child} does not properly override {parent}"+t)}},{key:"error",value:function(e){var t=this;throw e=e.replace("{parent}",function(e){return t.parentNotation}).replace("{child}",function(e){return t.childNotation}),new SyntaxError(e)}}]),e}(),g=[function(e){return e.toLowerCase()},function(e){return e.toUpperCase()},function(e){return e+"s"},function(e){return e.slice(0,-1)},function(e){return e.slice(1,e.length)}]},function(e,t,n){"use strict";function r(e,t,r,c){var s=l(c,3),f=s[0],d=void 0===f?null:f,p=s[1],h=void 0!==p&&p,m=s[2],y=void 0===m?u:m;if(!o.__enabled)return o.__warned||(y.warn("console.profile is not supported. All @profile decorators are disabled."),o.__warned=!0),r;var g=r.value;if(null===d&&(d=e.constructor.name+"."+t),"function"!=typeof g)throw new SyntaxError("@profile can only be used on functions, not: "+g);return a({},r,{value:function(){var e=Date.now(),t=n.i(i.c)(this);(!0===h&&!t.profileLastRan||!1===h||"number"==typeof h&&e-t.profileLastRan>h||"function"==typeof h&&h.apply(this,arguments))&&(y.profile(d),t.profileLastRan=e);try{return g.apply(this,arguments)}finally{y.profileEnd(d)}}})}function o(){for(var e=arguments.length,t=Array(e),o=0;o1){if(i=e({path:"/"},r.defaults,i),"number"==typeof i.expires){var l=new Date;l.setMilliseconds(l.getMilliseconds()+864e5*i.expires),i.expires=l}i.expires=i.expires?i.expires.toUTCString():"";try{a=JSON.stringify(o),/^[\{\[]/.test(a)&&(o=a)}catch(e){}o=n.write?n.write(o,t):encodeURIComponent(String(o)).replace(/%(23|24|26|2B|3A|3C|3E|3D|2F|3F|40|5B|5D|5E|60|7B|7D|7C)/g,decodeURIComponent),t=encodeURIComponent(String(t)),t=t.replace(/%(23|24|26|2B|5E|60|7C)/g,decodeURIComponent),t=t.replace(/[\(\)]/g,escape);var u="";for(var c in i)i[c]&&(u+="; "+c,!0!==i[c]&&(u+="="+i[c]));return document.cookie=t+"="+o+u}t||(a={});for(var s=document.cookie?document.cookie.split("; "):[],f=/(%[0-9A-Z]{2})+/g,d=0;d=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n}function i(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t1?t-1:0),r=1;r0&&(H.apply(void 0,t),L&&((0,N.default)("called ga('arguments');"),(0,N.default)("with arguments: "+JSON.stringify(t)))),window.ga}function f(e,t){return e?"object"!==(void 0===e?"undefined":w(e))?void(0,A.default)("Expected `fieldsObject` arg to be an Object"):(0===Object.keys(e).length&&(0,A.default)("empty `fieldsObject` given to .set()"),l(t,"set",e),void(L&&((0,N.default)("called ga('set', fieldsObject);"),(0,N.default)("with fieldsObject: "+JSON.stringify(e))))):void(0,A.default)("`fieldsObject` is required in .set()")}function d(e,t){l(t,"send",e),L&&((0,N.default)("called ga('send', fieldObject);"),(0,N.default)("with fieldObject: "+JSON.stringify(e)),(0,N.default)("with trackers: "+JSON.stringify(t)))}function p(e,t,n){if(!e)return void(0,A.default)("path is required in .pageview()");var r=(0,j.default)(e);if(""===r)return void(0,A.default)("path cannot be an empty string in .pageview()");var o={};if(n&&(o.title=n),"function"==typeof s&&(l(t,"send",b({hitType:"pageview",page:r},o)),L)){(0,N.default)("called ga('send', 'pageview', path);");var i="";n&&(i=" and title: "+n),(0,N.default)("with path: "+r+i)}}function h(e,t){if(!e)return void(0,A.default)("modalName is required in .modalview(modalName)");var n=(0,E.default)((0,j.default)(e));if(""===n)return void(0,A.default)("modalName cannot be an empty string or a single / in .modalview()");if("function"==typeof s){var r="/modal/"+n;l(t,"send","pageview",r),L&&((0,N.default)("called ga('send', 'pageview', path);"),(0,N.default)("with path: "+r))}}function m(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=e.category,n=e.variable,r=e.value,o=e.label,i=arguments[1];if("function"==typeof s){if(!t||!n||!r||"number"!=typeof r)return void(0,A.default)("args.category, args.variable AND args.value are required in timing() AND args.value has to be a number");var l={hitType:"timing",timingCategory:a(t),timingVar:a(n),timingValue:r};o&&(l.timingLabel=a(o)),d(l,i)}}function y(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments[1],n=e.category,r=e.action,i=e.label,l=e.value,u=e.nonInteraction,c=e.transport,f=o(e,["category","action","label","value","nonInteraction","transport"]);if("function"==typeof s){if(!n||!r)return void(0,A.default)("args.category AND args.action are required in event()");var p={hitType:"event",eventCategory:a(n),eventAction:a(r)};i&&(p.eventLabel=a(i)),void 0!==l&&("number"!=typeof l?(0,A.default)("Expected `args.value` arg to be a Number."):p.eventValue=l),void 0!==u&&("boolean"!=typeof u?(0,A.default)("`args.nonInteraction` must be a boolean."):p.nonInteraction=u),void 0!==c&&("string"!=typeof c?(0,A.default)("`args.transport` must be a string."):(-1===["beacon","xhr","image"].indexOf(c)&&(0,A.default)("`args.transport` must be either one of these values: `beacon`, `xhr` or `image`"),p.transport=c)),Object.keys(f).filter(function(e){return"dimension"===e.substr(0,"dimension".length)}).forEach(function(e){p[e]=f[e]}),Object.keys(f).filter(function(e){return"metric"===e.substr(0,"metric".length)}).forEach(function(e){p[e]=f[e]}),d(p,t)}}function g(e,t){var n=e.description,r=e.fatal;if("function"==typeof s){var o={hitType:"exception"};n&&(o.exDescription=a(n)),void 0!==r&&("boolean"!=typeof r?(0,A.default)("`args.fatal` must be a boolean."):o.exFatal=r),d(o,t)}}function v(e,t,n){if("function"!=typeof t)return void(0,A.default)("hitCallback function is required");if("function"==typeof s){if(!e||!e.label)return void(0,A.default)("args.label is required in outboundLink()");var r={hitType:"event",eventCategory:"Outbound",eventAction:"Click",eventLabel:a(e.label)},o=!1,i=function(){o=!0,t()},l=setTimeout(i,250),u=function(){clearTimeout(l),o||t()};r.hitCallback=u,d(r,n)}else setTimeout(t,0)}Object.defineProperty(t,"__esModule",{value:!0}),t.testModeAPI=t.OutboundLink=t.plugin=void 0;var b=Object.assign||function(e){for(var t=1;t0&&t+e.length!==n.length&&e.search(a)>-1&&":"!==n.charAt(t-2)&&("-"!==n.charAt(t+e.length)||"-"===n.charAt(t-1))&&n.charAt(t-1).search(/[^\s-]/)<0?e.toLowerCase():e.substr(1).search(/[A-Z]|\../)>-1?e:e.charAt(0).toUpperCase()+e.substr(1)})}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r;var o=n(1),i=function(e){return e&&e.__esModule?e:{default:e}}(o),a=/^(a|an|and|as|at|but|by|en|for|if|in|nor|of|on|or|per|the|to|vs?\.?|via)$/i},function(e,t,n){"use strict";function r(e){return"/"===e.substring(0,1)?e.substring(1):e}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default=function(e){!function(e,t,n,r,o,i,a){e.GoogleAnalyticsObject=o,e[o]=e[o]||function(){(e[o].q=e[o].q||[]).push(arguments)},e[o].l=1*new Date,i=t.createElement(n),a=t.getElementsByTagName(n)[0],i.async=1,i.src=r,a.parentNode.insertBefore(i,a)}(window,document,"script",e&&e.gaAddress?e.gaAddress:"https://www.google-analytics.com/analytics.js","ga")}},function(e,t,n){"use strict";function r(e){console.info("[react-ga]",e)}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=t.gaCalls=[];t.default={calls:r,ga:function(){for(var e=arguments.length,t=Array(e),n=0;n=Math.floor(a)&&s=Math.floor(l),p=n.getActiveLink();d&&(o===p&&n.setActiveLink(void 0),e.props.hashSpy&&w.default.getHash()===o&&w.default.changeHash(),e.props.spy&&e.state.active&&(e.setState({active:!1}),e.props.onSetInactive&&e.props.onSetInactive(o,i))),!f||p===o&&!1!==e.state.active||(n.setActiveLink(o),e.props.hashSpy&&w.default.changeHash(o),e.props.spy&&(e.setState({active:!0}),e.props.onSetActive&&e.props.onSetActive(o,i)))}}};return r.propTypes=k,r.defaultProps={offset:0},r}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=n(51),o=function(e){return e&&e.__esModule?e:{default:e}}(r),i=n(40),a=function(e){return(0,o.default)(e,66)},l={spyCallbacks:[],spySetState:[],scrollSpyContainers:[],mount:function(e){if(e){var t=a(function(t){l.scrollHandler(e)});l.scrollSpyContainers.push(e),(0,i.addPassiveEventListener)(e,"scroll",t)}},isMounted:function(e){return-1!==l.scrollSpyContainers.indexOf(e)},currentPositionY:function(e){if(e===document){var t=void 0!==window.pageXOffset,n="CSS1Compat"===(document.compatMode||"");return t?window.pageYOffset:n?document.documentElement.scrollTop:document.body.scrollTop}return e.scrollTop},scrollHandler:function(e){(l.scrollSpyContainers[l.scrollSpyContainers.indexOf(e)].spyCallbacks||[]).forEach(function(t){return t(l.currentPositionY(e))})},addStateHandler:function(e){l.spySetState.push(e)},addSpyHandler:function(e,t){var n=l.scrollSpyContainers[l.scrollSpyContainers.indexOf(t)];n.spyCallbacks||(n.spyCallbacks=[]),n.spyCallbacks.push(e),e(l.currentPositionY(t))},updateStates:function(){l.spySetState.forEach(function(e){return e()})},unmount:function(e,t){l.scrollSpyContainers.forEach(function(e){return e.spyCallbacks&&e.spyCallbacks.length&&e.spyCallbacks.splice(e.spyCallbacks.indexOf(t),1)}),l.spySetState&&l.spySetState.length&&l.spySetState.splice(l.spySetState.indexOf(e),1),document.removeEventListener("scroll",l.scrollHandler)},update:function(){return l.scrollSpyContainers.forEach(function(e){return l.scrollHandler(e)})}};t.default=l},function(e,t){},function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}Object.defineProperty(t,"__esModule",{value:!0});var o=Object.assign||function(e){for(var t=1;t=o.duration?1:t(o.progress/o.duration),o.currentPositionY=o.startPositionY+Math.ceil(o.deltaTop*o.percent),o.containerElement&&o.containerElement!==document&&o.containerElement!==document.body?o.containerElement.scrollTop=o.currentPositionY:window.scrollTo(0,o.currentPositionY),o.percent<1){var i=e.bind(null,t,n);return void m.call(window,i)}f.default.registered.end&&f.default.registered.end(o.to,o.target,o.currentPositionY)},w=function(e){e.data.containerElement=e?e.containerId?document.getElementById(e.containerId):e.container&&e.container.nodeType?e.container:document:null},k=function(e,t,n,r){if(t.data=t.data||y(),window.clearTimeout(t.data.delayTimeout),c.default.subscribe(function(){t.data.cancel=!0}),w(t),t.data.start=null,t.data.cancel=!1,t.data.startPositionY=g(t),t.data.targetPositionY=t.absolute?e:e+t.data.startPositionY,t.data.startPositionY===t.data.targetPositionY)return void(f.default.registered.end&&f.default.registered.end(t.data.to,t.data.target,t.data.currentPositionY));t.data.deltaTop=Math.round(t.data.targetPositionY-t.data.startPositionY),t.data.duration=p(t.duration)(t.data.deltaTop),t.data.duration=isNaN(parseFloat(t.data.duration))?1e3:parseFloat(t.data.duration),t.data.to=n,t.data.target=r;var o=d(t),i=b.bind(null,o,t);if(t&&t.delay>0)return void(t.data.delayTimeout=window.setTimeout(function(){m.call(window,i)},t.delay));m.call(window,i)},O=function(e){return e=o({},e),e.data=e.data||y(),e.absolute=!0,e},_=function(e){k(0,O(e))},E=function(e,t){k(e,O(t))},S=function(e){e=O(e),w(e),k(v(e),e)},j=function(e,t){t=O(t),w(t),k(g(t)+e,t)};t.default={animateTopScroll:k,getAnimationType:d,scrollToTop:_,scrollToBottom:S,scrollTo:E,scrollMore:j}},function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}function o(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function i(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t}function a(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}Object.defineProperty(t,"__esModule",{value:!0});var l=Object.assign||function(e){for(var t=1;t=0;r--){var o=e[r];"."===o?e.splice(r,1):".."===o?(e.splice(r,1),n++):n&&(e.splice(r,1),n--)}if(t)for(;n--;n)e.unshift("..");return e}function r(e,t){if(e.filter)return e.filter(t);for(var n=[],r=0;r=-1&&!o;i--){var a=i>=0?arguments[i]:e.cwd();if("string"!=typeof a)throw new TypeError("Arguments to path.resolve must be strings");a&&(t=a+"/"+t,o="/"===a.charAt(0))}return t=n(r(t.split("/"),function(e){return!!e}),!o).join("/"),(o?"/":"")+t||"."},t.normalize=function(e){var o=t.isAbsolute(e),i="/"===a(e,-1);return e=n(r(e.split("/"),function(e){return!!e}),!o).join("/"),e||o||(e="."),e&&i&&(e+="/"),(o?"/":"")+e},t.isAbsolute=function(e){return"/"===e.charAt(0)},t.join=function(){var e=Array.prototype.slice.call(arguments,0);return t.normalize(r(e,function(e,t){if("string"!=typeof e)throw new TypeError("Arguments to path.join must be strings");return e}).join("/"))},t.relative=function(e,n){function r(e){for(var t=0;t=0&&""===e[n];n--);return t>n?[]:e.slice(t,n-t+1)}e=t.resolve(e).substr(1),n=t.resolve(n).substr(1);for(var o=r(e.split("/")),i=r(n.split("/")),a=Math.min(o.length,i.length),l=a,u=0;u-1?t:e}function p(e,t){t=t||{};var n=t.body;if(e instanceof p){if(e.bodyUsed)throw new TypeError("Already read");this.url=e.url,this.credentials=e.credentials,t.headers||(this.headers=new o(e.headers)),this.method=e.method,this.mode=e.mode,n||null==e._bodyInit||(n=e._bodyInit,e.bodyUsed=!0)}else this.url=String(e);if(this.credentials=t.credentials||this.credentials||"omit",!t.headers&&this.headers||(this.headers=new o(t.headers)),this.method=d(t.method||this.method||"GET"),this.mode=t.mode||this.mode||null,this.referrer=null,("GET"===this.method||"HEAD"===this.method)&&n)throw new TypeError("Body not allowed for GET or HEAD requests");this._initBody(n)}function h(e){var t=new FormData;return e.trim().split("&").forEach(function(e){if(e){var n=e.split("="),r=n.shift().replace(/\+/g," "),o=n.join("=").replace(/\+/g," ");t.append(decodeURIComponent(r),decodeURIComponent(o))}}),t}function m(e){var t=new o;return e.replace(/\r?\n[\t ]+/g," ").split(/\r?\n/).forEach(function(e){var n=e.split(":"),r=n.shift().trim();if(r){var o=n.join(":").trim();t.append(r,o)}}),t}function y(e,t){t||(t={}),this.type="default",this.status=void 0===t.status?200:t.status,this.ok=this.status>=200&&this.status<300,this.statusText="statusText"in t?t.statusText:"OK",this.headers=new o(t.headers),this.url=t.url||"",this._initBody(e)}if(!e.fetch){var g={searchParams:"URLSearchParams"in e,iterable:"Symbol"in e&&"iterator"in Symbol,blob:"FileReader"in e&&"Blob"in e&&function(){try{return new Blob,!0}catch(e){return!1}}(),formData:"FormData"in e,arrayBuffer:"ArrayBuffer"in e};if(g.arrayBuffer)var v=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],b=function(e){return e&&DataView.prototype.isPrototypeOf(e)},w=ArrayBuffer.isView||function(e){return e&&v.indexOf(Object.prototype.toString.call(e))>-1};o.prototype.append=function(e,r){e=t(e),r=n(r);var o=this.map[e];this.map[e]=o?o+","+r:r},o.prototype.delete=function(e){delete this.map[t(e)]},o.prototype.get=function(e){return e=t(e),this.has(e)?this.map[e]:null},o.prototype.has=function(e){return this.map.hasOwnProperty(t(e))},o.prototype.set=function(e,r){this.map[t(e)]=n(r)},o.prototype.forEach=function(e,t){for(var n in this.map)this.map.hasOwnProperty(n)&&e.call(t,this.map[n],n,this)},o.prototype.keys=function(){var e=[];return this.forEach(function(t,n){e.push(n)}),r(e)},o.prototype.values=function(){var e=[];return this.forEach(function(t){e.push(t)}),r(e)},o.prototype.entries=function(){var e=[];return this.forEach(function(t,n){e.push([n,t])}),r(e)},g.iterable&&(o.prototype[Symbol.iterator]=o.prototype.entries);var k=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];p.prototype.clone=function(){return new p(this,{body:this._bodyInit})},f.call(p.prototype),f.call(y.prototype),y.prototype.clone=function(){return new y(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new o(this.headers),url:this.url})},y.error=function(){var e=new y(null,{status:0,statusText:""});return e.type="error",e};var O=[301,302,303,307,308];y.redirect=function(e,t){if(-1===O.indexOf(t))throw new RangeError("Invalid status code");return new y(null,{status:t,headers:{location:e}})},e.Headers=o,e.Request=p,e.Response=y,e.fetch=function(e,t){return new Promise(function(n,r){var o=new p(e,t),i=new XMLHttpRequest;i.onload=function(){var e={status:i.status,statusText:i.statusText,headers:m(i.getAllResponseHeaders()||"")};e.url="responseURL"in i?i.responseURL:e.headers.get("X-Request-URL");var t="response"in i?i.response:i.responseText;n(new y(t,e))},i.onerror=function(){r(new TypeError("Network request failed"))},i.ontimeout=function(){r(new TypeError("Network request failed"))},i.open(o.method,o.url,!0),"include"===o.credentials?i.withCredentials=!0:"omit"===o.credentials&&(i.withCredentials=!1),"responseType"in i&&g.blob&&(i.responseType="blob"),o.headers.forEach(function(e,t){i.setRequestHeader(t,e)}),i.send(void 0===o._bodyInit?null:o._bodyInit)})},e.fetch.polyfill=!0}}("undefined"!=typeof self?self:this)},function(e,t,n){(function(t){function n(e,t,n){function r(t){var n=m,r=y;return m=y=void 0,E=t,v=e.apply(r,n)}function i(e){return E=e,b=setTimeout(s,t),S?r(e):v}function a(e){var n=e-w,r=e-E,o=t-n;return j?O(o,g-r):o}function c(e){var n=e-w,r=e-E;return void 0===w||n>=t||n<0||j&&r>=g}function s(){var e=_();if(c(e))return f(e);b=setTimeout(s,a(e))}function f(e){return b=void 0,P&&m?r(e):(m=y=void 0,v)}function d(){void 0!==b&&clearTimeout(b),E=0,m=w=y=b=void 0}function p(){return void 0===b?v:f(_())}function h(){var e=_(),n=c(e);if(m=arguments,y=this,w=e,n){if(void 0===b)return i(w);if(j)return b=setTimeout(s,t),r(w)}return void 0===b&&(b=setTimeout(s,t)),v}var m,y,g,v,b,w,E=0,S=!1,j=!1,P=!0;if("function"!=typeof e)throw new TypeError(u);return t=l(t)||0,o(n)&&(S=!!n.leading,j="maxWait"in n,g=j?k(l(n.maxWait)||0,t):g,P="trailing"in n?!!n.trailing:P),h.cancel=d,h.flush=p,h}function r(e,t,r){var i=!0,a=!0;if("function"!=typeof e)throw new TypeError(u);return o(r)&&(i="leading"in r?!!r.leading:i,a="trailing"in r?!!r.trailing:a),n(e,t,{leading:i,maxWait:t,trailing:a})}function o(e){var t=typeof e;return!!e&&("object"==t||"function"==t)}function i(e){return!!e&&"object"==typeof e}function a(e){return"symbol"==typeof e||i(e)&&w.call(e)==s}function l(e){if("number"==typeof e)return e;if(a(e))return c;if(o(e)){var t="function"==typeof e.valueOf?e.valueOf():e;e=o(t)?t+"":t}if("string"!=typeof e)return 0===e?e:+e;e=e.replace(f,"");var n=p.test(e);return n||h.test(e)?m(e.slice(2),n?2:8):d.test(e)?c:+e}var u="Expected a function",c=NaN,s="[object Symbol]",f=/^\s+|\s+$/g,d=/^[-+]0x[0-9a-f]+$/i,p=/^0b[01]+$/i,h=/^0o[0-7]+$/i,m=parseInt,y="object"==typeof t&&t&&t.Object===Object&&t,g="object"==typeof self&&self&&self.Object===Object&&self,v=y||g||Function("return this")(),b=Object.prototype,w=b.toString,k=Math.max,O=Math.min,_=function(){return v.Date.now()};e.exports=r}).call(t,n(59))},function(e,t){function n(){throw new Error("setTimeout has not been defined")}function r(){throw new Error("clearTimeout has not been defined")}function o(e){if(s===setTimeout)return setTimeout(e,0);if((s===n||!s)&&setTimeout)return s=setTimeout,setTimeout(e,0);try{return s(e,0)}catch(t){try{return s.call(null,e,0)}catch(t){return s.call(this,e,0)}}}function i(e){if(f===clearTimeout)return clearTimeout(e);if((f===r||!f)&&clearTimeout)return f=clearTimeout,clearTimeout(e);try{return f(e)}catch(t){try{return f.call(null,e)}catch(t){return f.call(this,e)}}}function a(){m&&p&&(m=!1,p.length?h=p.concat(h):y=-1,h.length&&l())}function l(){if(!m){var e=o(a);m=!0;for(var t=h.length;t;){for(p=h,h=[];++y1)for(var n=1;n=Math.floor(a)&&f=Math.floor(l),m=n.getActiveLink();return h?(o===m&&n.setActiveLink(void 0),e.props.hashSpy&&d.getHash()===o&&d.changeHash(),e.props.spy&&e.state.active&&(e.setState({active:!1}),e.props.onSetInactive&&e.props.onSetInactive()),c.updateStates()):p&&m!==o?(n.setActiveLink(o),e.props.hashSpy&&d.changeHash(o),e.props.spy&&(e.setState({active:!0}),e.props.onSetActive&&e.props.onSetActive(o)),c.updateStates()):void 0}}};return f.propTypes=p,f.defaultProps={offset:0},f},Element:function(e){console.warn("Helpers.Element is deprecated since v1.7.0");var t=function(t){function n(e){r(this,n);var t=o(this,(n.__proto__||Object.getPrototypeOf(n)).call(this,e));return t.childBindings={domNode:null},t}return i(n,t),l(n,[{key:"componentDidMount",value:function(){if("undefined"==typeof window)return!1;this.registerElems(this.props.name)}},{key:"componentWillReceiveProps",value:function(e){this.props.name!==e.name&&this.registerElems(e.name)}},{key:"componentWillUnmount",value:function(){if("undefined"==typeof window)return!1;s.unregister(this.props.name)}},{key:"registerElems",value:function(e){s.register(e,this.childBindings.domNode)}},{key:"render",value:function(){return u.createElement(e,a({},this.props,{parentBindings:this.childBindings}))}}]),n}(u.Component);return t.propTypes={name:f.string,id:f.string},t}};e.exports=h},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=n(40),o=["mousedown","mousewheel","touchmove","keydown"];t.default={subscribe:function(e){return"undefined"!=typeof document&&o.forEach(function(t){return(0,r.addPassiveEventListener)(document,t,e)})}}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={defaultEasing:function(e){return e<.5?Math.pow(2*e,2)/2:1-Math.pow(2*(1-e),2)/2},linear:function(e){return e},easeInQuad:function(e){return e*e},easeOutQuad:function(e){return e*(2-e)},easeInOutQuad:function(e){return e<.5?2*e*e:(4-2*e)*e-1},easeInCubic:function(e){return e*e*e},easeOutCubic:function(e){return--e*e*e+1},easeInOutCubic:function(e){return e<.5?4*e*e*e:(e-1)*(2*e-2)*(2*e-2)+1},easeInQuart:function(e){return e*e*e*e},easeOutQuart:function(e){return 1- --e*e*e*e},easeInOutQuart:function(e){return e<.5?8*e*e*e*e:1-8*--e*e*e*e},easeInQuint:function(e){return e*e*e*e*e},easeOutQuint:function(e){return 1+--e*e*e*e*e},easeInOutQuint:function(e){return e<.5?16*e*e*e*e*e:1+16*--e*e*e*e*e}}},function(e,t){var n;n=function(){return this}();try{n=n||Function("return this")()||(0,eval)("this")}catch(e){"object"==typeof window&&(n=window)}e.exports=n},,,function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={"en-us":{sidemenu:[{title:"Nacos ",children:[{title:"What is Nacos",opened:!0,children:[{title:"What is Nacos",link:"/en-us/docs/what-is-nacos.html"},{title:"Concepts",link:"/en-us/docs/concepts.html"},{title:"Architecure",link:"/en-us/docs/architecture.html"}]},{title:"Quick Start",opened:!0,children:[{title:"Nacos",link:"/en-us/docs/quick-start.html"},{title:"Nacos Spring",link:"/en-us/docs/quick-start-spring.html"},{title:"Nacos Spring Boot",link:"/en-us/docs/quick-start-spring-boot.html"},{title:"Nacos Spring Cloud",link:"/en-us/docs/quick-start-spring-cloud.html"},{title:"Nacos Docker",link:"/en-us/docs/quick-start-docker.html"},{title:"Nacos with Dubbo",link:"/en-us/docs/use-nacos-with-dubbo.html"},{title:"Nacos with K8s",link:"/en-us/docs/use-nacos-with-kubernetes.html"},{title:"Nacos Sync",link:"/en-us/docs/nacos-sync.html"}]},{title:"User Guide",children:[{title:"Java SDK",link:"/en-us/docs/sdk.html"},{title:"Other Language",link:"/en-us/docs/other-language.html"},{title:"Open-API",link:"/en-us/docs/open-api.html"},{title:"Nacos Spring",link:"/en-us/docs/nacos-spring.html"},{title:"Nacos System Configurations",link:"/en-us/docs/system-configurations.html"},{title:"Authentication",link:"/en-us/docs/auth.html"},{title:"FAQ",link:"/en-us/docs/faq.html"}]},{title:"Admin Guide",children:[{title:"Deployment Guide",link:"/en-us/docs/deployment.html"},{title:"Cluster Mode Deployment",link:"/en-us/docs/cluster-mode-quick-start.html"},{title:"Management OpenAPI",link:"/en-us/docs/management-api.html"},{title:"Admin Console Guide",link:"/en-us/docs/console-guide.html"},{title:"Nacos Monitor Guide",link:"/en-us/docs/monitor-guide.html"},{title:"Nacos Config Benchmark",link:"/en-us/docs/nacos-config-benchmark.html"},{title:"Nacos Naming Benchmark",link:"/en-us/docs/nacos-naming-benchmark.html"},{title:"Data Migration to Nacos",link:"/en-us/docs/nacos-sync-use.html"}]},{title:"Contributor Guide",children:[{title:"Contribute to Nacos",link:"/en-us/docs/contributing.html"},{title:"Contributing Flow",link:"/en-us/docs/contributing-flow.html"},{title:"Pull Request Template",link:"/en-us/docs/pull-request.html"},{title:"How to report bugs",link:"/en-us/docs/how-to-reporting-bugs.html"}]},{title:"Community",children:[{title:"Community",link:"/en-us/docs/community.html"},{title:"Develop Team",link:"/en-us/docs/nacos-dev.html"}]}]}],barText:"Documentation"},"zh-cn":{sidemenu:[{title:"Nacos ",children:[{title:"Nacos是什么?",opened:!0,children:[{title:"Nacos简介",link:"/zh-cn/docs/what-is-nacos.html"},{title:"概念",link:"/zh-cn/docs/concepts.html"},{title:"架构",link:"/zh-cn/docs/architecture.html"}]},{title:"快速开始",opened:!0,children:[{title:"Nacos",link:"/zh-cn/docs/quick-start.html"},{title:"Nacos Spring",link:"/zh-cn/docs/quick-start-spring.html"},{title:"Nacos Spring Boot",link:"/zh-cn/docs/quick-start-spring-boot.html"},{title:"Nacos Spring Cloud",link:"/zh-cn/docs/quick-start-spring-cloud.html"},{title:"Nacos Docker",link:"/zh-cn/docs/quick-start-docker.html"},{title:"Nacos Dubbo",link:"/zh-cn/docs/use-nacos-with-dubbo.html"},{title:"Nacos k8s",link:"/zh-cn/docs/use-nacos-with-kubernetes.html"},{title:"Nacos Sync",link:"/zh-cn/docs/nacos-sync.html"}]},{title:"用户指南",children:[{title:"Java的SDK",link:"/zh-cn/docs/sdk.html"},{title:"其他语言的SDK",link:"/zh-cn/docs/other-language.html"},{title:"Open-API指南",link:"/zh-cn/docs/open-api.html"},{title:"Nacos Spring",link:"/zh-cn/docs/nacos-spring.html"},{title:"Nacos系统参数介绍",link:"/zh-cn/docs/system-configurations.html"},{title:"权限认证",link:"/zh-cn/docs/auth.html"},{title:"FAQ",link:"/zh-cn/docs/faq.html"}]},{title:"运维指南",children:[{title:"部署手册",link:"/zh-cn/docs/deployment.html"},{title:"集群部署说明",link:"/zh-cn/docs/cluster-mode-quick-start.html"},{title:"运维API",link:"/zh-cn/docs/management-api.html"},{title:"控制台手册",link:"/zh-cn/docs/console-guide.html"},{title:"监控手册",link:"/zh-cn/docs/monitor-guide.html"},{title:"Nacos配置模块压测报告",link:"/zh-cn/docs/nacos-config-benchmark.html"},{title:"Nacos服务发现模块压测报告",link:"/zh-cn/docs/nacos-naming-benchmark.html"},{title:"其他环境迁移到Nacos",link:"/zh-cn/docs/nacos-sync-use.html"}]},{title:"开源共建",children:[{title:"贡献源码",link:"/zh-cn/docs/contributing.html"},{title:"贡献流程",link:"/zh-cn/docs/contributing-flow.html"},{title:"pull request模板",link:"/zh-cn/docs/pull-request.html"},{title:"如何提交问题报告",link:"/zh-cn/docs/how-to-reporting-bugs.html"}]},{title:"社区",children:[{title:"社区",link:"/zh-cn/docs/community.html"},{title:"开发团队",link:"/zh-cn/docs/nacos-dev.html"}]}]}],barText:"Nacos 文档"}}},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default={"en-us":{sidemenu:[{title:"Nacos ",children:[{title:"What is Nacos",opened:!0,children:[{title:"What is Nacos",link:"/en-us/docs/v2/what-is-nacos.html"},{title:"Concepts",link:"/en-us/docs/v2/concepts.html"},{title:"Architecure",link:"/en-us/docs/v2/architecture.html"}]},{title:"Quick Start",opened:!0,children:[{title:"Nacos",link:"/en-us/docs/v2/quickstart/quick-start.html"},{title:"Nacos Docker",link:"/en-us/docs/v2/quickstart/quick-start-docker.html"},{title:"Nacos kubernetes",link:"/en-us/docs/v2/quickstart/quick-start-kubernetes.html"}]},{title:"Upgrading",children:[{title:"Compatibility&Usage",link:"/en-us/docs/v2/upgrading/2.0.0-compatibility.html"},{title:"Upgrading",link:"/en-us/docs/v2/upgrading/2.0.0-upgrading.html"}]},{title:"Plugin",children:[{title:"Authentication",link:"/en-us/docs/v2/plugin/auth-plugin.html"},{title:"Configuration Encryption",link:"/en-us/docs/v2/plugin/config-encryption-plugin.html"},{title:"Datasource",link:"/en-us/docs/v2/plugin/datasource-plugin.html"},{title:"Track Tracing",link:"/en-us/docs/v2/plugin/trace-plugin.html"},{title:"Custom Environment",link:"/en-us/docs/v2/plugin/custom-environment-plugin.html"},{title:"Control",link:"/en-us/docs/v2/plugin/control-plugin.html"},{title:"Configuration Change",link:"/en-us/docs/v2/plugin/config-change-plugin.html"}]},{title:"User Guide",children:[{title:"Java SDK",link:"/en-us/docs/v2/guide/user/sdk.html"},{title:"Other Language",link:"/en-us/docs/v2/guide/user/other-language.html"},{title:"Open-API",link:"/en-us/docs/v2/guide/user/open-api.html"},{title:"Authentication",link:"/en-us/docs/v2/guide/user/auth.html"},{title:"FAQ",link:"/en-us/docs/v2/guide/user/faq.html"}]},{title:"Admin Guide",children:[{title:"Deployment Guide",link:"/en-us/docs/v2/guide/admin/deployment.html"},{title:"Cluster Mode Deployment",link:"/en-us/docs/v2/guide/admin/cluster-mode-quick-start.html"},{title:"System Configurations",link:"/en-us/docs/v2/guide/admin/system-configurations.html"},{title:"Management OpenAPI",link:"/en-us/docs/v2/guide/admin/management-api.html"},{title:"Console Guide",link:"/en-us/docs/v2/guide/admin/console-guide.html"},{title:"Monitor Guide",link:"/en-us/docs/v2/guide/admin/monitor-guide.html"},{title:"Config Benchmark",link:"/en-us/docs/v2/guide/admin/nacos2-config-benchmark.html"},{title:"Naming Benchmark",link:"/en-us/docs/v2/guide/admin/nacos2-naming-benchmark.html"},{title:"Data Migration to Nacos",link:"/en-us/docs/v2/ecology/use-nacos-sync.html"}]},{title:"Ecology",children:[{title:"Dubbo",link:"/en-us/docs/v2/ecology/use-nacos-with-dubbo.html"},{title:"Spring",link:"/en-us/docs/v2/ecology/use-nacos-with-spring.html"},{title:"Spring Boot",link:"/en-us/docs/v2/ecology/use-nacos-with-spring-boot.html"},{title:"Spring Cloud",link:"/en-us/docs/v2/ecology/use-nacos-with-spring-cloud.html"},{title:"Nacos Sync",link:"/en-us/docs/v2/ecology/use-nacos-sync.html"},{title:"CoreDNS",link:"/en-us/docs/v2/ecology/use-nacos-with-coredns.html"}]},{title:"Contributor Guide",children:[{title:"Contribute to Nacos",link:"/en-us/docs/v2/contribution/contributing.html"},{title:"Contributing Flow",link:"/en-us/docs/v2/contribution/contributing-flow.html"},{title:"Pull Request Template",link:"/en-us/docs/v2/contribution/pull-request.html"},{title:"How to report bugs",link:"/en-us/docs/v2/contribution/how-to-reporting-bugs.html"}]},{title:"Community",children:[{title:"Community",link:"/en-us/docs/v2/community/community.html"},{title:"Develop Team",link:"/en-us/docs/v2/community/nacos-dev.html"}]}]}],barText:"Documentation"},"zh-cn":{sidemenu:[{title:"Nacos ",children:[{title:"Nacos是什么?",opened:!0,children:[{title:"Nacos简介",link:"/zh-cn/docs/v2/what-is-nacos.html"},{title:"概念",link:"/zh-cn/docs/v2/concepts.html"},{title:"架构",link:"/zh-cn/docs/v2/architecture.html"}]},{title:"快速开始",opened:!0,children:[{title:"Nacos",link:"/zh-cn/docs/v2/quickstart/quick-start.html"},{title:"Nacos Docker",link:"/zh-cn/docs/v2/quickstart/quick-start-docker.html"},{title:"Nacos kubernetes",link:"/zh-cn/docs/v2/quickstart/quick-start-kubernetes.html"}]},{title:"升级",children:[{title:"兼容性及使用",link:"/zh-cn/docs/v2/upgrading/2.0.0-compatibility.html"},{title:"升级文档",link:"/zh-cn/docs/v2/upgrading/2.0.0-upgrading.html"}]},{title:"插件",children:[{title:"鉴权",link:"/zh-cn/docs/v2/plugin/auth-plugin.html"},{title:"配置加密",link:"/zh-cn/docs/v2/plugin/config-encryption-plugin.html"},{title:"数据源",link:"/zh-cn/docs/v2/plugin/datasource-plugin.html"},{title:"轨迹追踪",link:"/zh-cn/docs/v2/plugin/trace-plugin.html"},{title:"环境变量",link:"/zh-cn/docs/v2/plugin/custom-environment-plugin.html"},{title:"反脆弱",link:"/zh-cn/docs/v2/plugin/control-plugin.html"},{title:"配置变更",link:"/zh-cn/docs/v2/plugin/config-change-plugin.html"}]},{title:"用户指南",children:[{title:"Java的SDK",link:"/zh-cn/docs/v2/guide/user/sdk.html"},{title:"其他语言的SDK",link:"/zh-cn/docs/v2/guide/user/other-language.html"},{title:"Open-API指南",link:"/zh-cn/docs/v2/guide/user/open-api.html"},{title:"权限认证",link:"/zh-cn/docs/v2/guide/user/auth.html"},{title:"FAQ",link:"/zh-cn/docs/v2/guide/user/faq.html"}]},{title:"运维指南",children:[{title:"部署手册",link:"/zh-cn/docs/v2/guide/admin/deployment.html"},{title:"集群部署说明",link:"/zh-cn/docs/v2/guide/admin/cluster-mode-quick-start.html"},{title:"系统参数说明",link:"/zh-cn/docs/v2/guide/admin/system-configurations.html"},{title:"运维API",link:"/zh-cn/docs/v2/guide/admin/management-api.html"},{title:"控制台手册",link:"/zh-cn/docs/v2/guide/admin/console-guide.html"},{title:"监控手册",link:"/zh-cn/docs/v2/guide/admin/monitor-guide.html"},{title:"Nacos2.0配置模块压测报告",link:"/zh-cn/docs/v2/guide/admin/nacos2-config-benchmark.html"},{title:"Nacos2.0服务发现模块压测报告",link:"/zh-cn/docs/v2/guide/admin/nacos2-naming-benchmark.html"},{title:"其他环境迁移到Nacos",link:"/zh-cn/docs/v2/ecology/use-nacos-sync.html"}]},{title:"生态融合",children:[{title:"Dubbo",link:"/zh-cn/docs/v2/ecology/use-nacos-with-dubbo.html"},{title:"Spring",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring.html"},{title:"Spring Boot",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring-boot.html"},{title:"Spring Cloud",link:"/zh-cn/docs/v2/ecology/use-nacos-with-spring-cloud.html"},{title:"Nacos Sync",link:"/zh-cn/docs/v2/ecology/use-nacos-sync.html"},{title:"CoreDNS",link:"/zh-cn/docs/v2/ecology/use-nacos-with-coredns.html"}]},{title:"开源共建",children:[{title:"贡献源码",link:"/zh-cn/docs/v2/contribution/contributing.html"},{title:"贡献流程",link:"/zh-cn/docs/v2/contribution/contributing-flow.html"},{title:"pull request模板",link:"/zh-cn/docs/v2/contribution/pull-request.html"},{title:"如何提交问题报告",link:"/zh-cn/docs/v2/contribution/how-to-reporting-bugs.html"}]},{title:"社区",children:[{title:"社区",link:"/zh-cn/docs/v2/community/community.html"},{title:"开发团队",link:"/zh-cn/docs/v2/community/nacos-dev.html"}]}]}],barText:"Nacos 文档"}}},,,,function(e,t,n){"use strict";function r(e){return e&&e.__esModule?e:{default:e}}function o(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function i(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t}function a(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}Object.defineProperty(t,"__esModule",{value:!0});var l,u=function(){function e(e,t){for(var n=0;n1&&/^\/[^\/]/.test(""+e)?""+window.rootPath+e:e}},function(e,t,n){var r,a;/*! +!function(e){function t(r){if(n[r])return n[r].exports;var a=n[r]={i:r,l:!1,exports:{}};return e[r].call(a.exports,a,a.exports,t),a.l=!0,a.exports}var n={};t.m=e,t.c=n,t.i=function(e){return e},t.d=function(e,n,r){t.o(e,n)||Object.defineProperty(e,n,{configurable:!1,enumerable:!0,get:r})},t.n=function(e){var n=e&&e.__esModule?function(){return e.default}:function(){return e};return t.d(n,"a",n),n},t.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},t.p="/build/",t(t.s=86)}([function(e,t,n){"use strict";function r(e,t,n,r){n&&Object.defineProperty(e,t,{enumerable:n.enumerable,configurable:n.configurable,writable:n.writable,value:n.initializer?n.initializer.call(r):void 0})}function a(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function o(e,t,n,r,a){var o={};return Object.keys(r).forEach(function(e){o[e]=r[e]}),o.enumerable=!!o.enumerable,o.configurable=!!o.configurable,("value"in o||o.initializer)&&(o.writable=!0),o=n.slice().reverse().reduce(function(n,r){return r(e,t,n)||n},o),a&&void 0!==o.initializer&&(o.value=o.initializer?o.initializer.call(a):void 0,o.initializer=void 0),void 0===o.initializer&&(Object.defineProperty(e,t,o),o=null),o}function i(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t1&&/^\/[^\/]/.test(""+e)?""+window.rootPath+e:e}},function(e,t,n){var r,a;/*! Copyright (c) 2017 Jed Watson. Licensed under the MIT License (MIT), see http://jedwatson.github.io/classnames */ -!function(){"use strict";function n(){for(var e=[],t=0;t1&&void 0!==arguments[1]?arguments[1]:t.key)+arguments[2]})}},{key:"key",get:function(){return this.childDescriptor.key}},{key:"parentNotation",get:function(){return this.parentKlass.constructor.name+"#"+this.parentPropertySignature}},{key:"childNotation",get:function(){return this.childKlass.constructor.name+"#"+this.childPropertySignature}},{key:"parentTopic",get:function(){return this._getTopic(this.parentDescriptor)}},{key:"childTopic",get:function(){return this._getTopic(this.childDescriptor)}},{key:"parentPropertySignature",get:function(){return this._extractTopicSignature(this.parentTopic)}},{key:"childPropertySignature",get:function(){return this._extractTopicSignature(this.childTopic)}}]),m(e,[{key:"assert",value:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"";!0!==e&&this.error("{child} does not properly override {parent}"+t)}},{key:"error",value:function(e){var t=this;throw e=e.replace("{parent}",function(e){return t.parentNotation}).replace("{child}",function(e){return t.childNotation}),new SyntaxError(e)}}]),e}(),y=[function(e){return e.toLowerCase()},function(e){return e.toUpperCase()},function(e){return e+"s"},function(e){return e.slice(0,-1)},function(e){return e.slice(1,e.length)}]},function(e,t,n){"use strict";function r(e,t,r,l){var s=u(l,3),f=s[0],d=void 0===f?null:f,p=s[1],m=void 0!==p&&p,g=s[2],h=void 0===g?c:g;if(!a.__enabled)return a.__warned||(h.warn("console.profile is not supported. All @profile decorators are disabled."),a.__warned=!0),r;var y=r.value;if(null===d&&(d=e.constructor.name+"."+t),"function"!=typeof y)throw new SyntaxError("@profile can only be used on functions, not: "+y);return i({},r,{value:function(){var e=Date.now(),t=n.i(o.c)(this);(!0===m&&!t.profileLastRan||!1===m||"number"==typeof m&&e-t.profileLastRan>m||"function"==typeof m&&m.apply(this,arguments))&&(h.profile(d),t.profileLastRan=e);try{return y.apply(this,arguments)}finally{h.profileEnd(d)}}})}function a(){for(var e=arguments.length,t=Array(e),a=0;a1){if(o=e({path:"/"},r.defaults,o),"number"==typeof o.expires){var u=new Date;u.setMilliseconds(u.getMilliseconds()+864e5*o.expires),o.expires=u}o.expires=o.expires?o.expires.toUTCString():"";try{i=JSON.stringify(a),/^[\{\[]/.test(i)&&(a=i)}catch(e){}a=n.write?n.write(a,t):encodeURIComponent(String(a)).replace(/%(23|24|26|2B|3A|3C|3E|3D|2F|3F|40|5B|5D|5E|60|7B|7D|7C)/g,decodeURIComponent),t=encodeURIComponent(String(t)),t=t.replace(/%(23|24|26|2B|5E|60|7C)/g,decodeURIComponent),t=t.replace(/[\(\)]/g,escape);var c="";for(var l in o)o[l]&&(c+="; "+l,!0!==o[l]&&(c+="="+o[l]));return document.cookie=t+"="+a+c}t||(i={});for(var s=document.cookie?document.cookie.split("; "):[],f=/(%[0-9A-Z]{2})+/g,d=0;d=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n}function o(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t1?t-1:0),r=1;r0&&(R.apply(void 0,t),q&&((0,C.default)("called ga('arguments');"),(0,C.default)("with arguments: "+JSON.stringify(t)))),window.ga}function f(e,t){return e?"object"!==(void 0===e?"undefined":w(e))?void(0,T.default)("Expected `fieldsObject` arg to be an Object"):(0===Object.keys(e).length&&(0,T.default)("empty `fieldsObject` given to .set()"),u(t,"set",e),void(q&&((0,C.default)("called ga('set', fieldsObject);"),(0,C.default)("with fieldsObject: "+JSON.stringify(e))))):void(0,T.default)("`fieldsObject` is required in .set()")}function d(e,t){u(t,"send",e),q&&((0,C.default)("called ga('send', fieldObject);"),(0,C.default)("with fieldObject: "+JSON.stringify(e)),(0,C.default)("with trackers: "+JSON.stringify(t)))}function p(e,t,n){if(!e)return void(0,T.default)("path is required in .pageview()");var r=(0,S.default)(e);if(""===r)return void(0,T.default)("path cannot be an empty string in .pageview()");var a={};if(n&&(a.title=n),"function"==typeof s&&(u(t,"send",b({hitType:"pageview",page:r},a)),q)){(0,C.default)("called ga('send', 'pageview', path);");var o="";n&&(o=" and title: "+n),(0,C.default)("with path: "+r+o)}}function m(e,t){if(!e)return void(0,T.default)("modalName is required in .modalview(modalName)");var n=(0,_.default)((0,S.default)(e));if(""===n)return void(0,T.default)("modalName cannot be an empty string or a single / in .modalview()");if("function"==typeof s){var r="/modal/"+n;u(t,"send","pageview",r),q&&((0,C.default)("called ga('send', 'pageview', path);"),(0,C.default)("with path: "+r))}}function g(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=e.category,n=e.variable,r=e.value,a=e.label,o=arguments[1];if("function"==typeof s){if(!t||!n||!r||"number"!=typeof r)return void(0,T.default)("args.category, args.variable AND args.value are required in timing() AND args.value has to be a number");var u={hitType:"timing",timingCategory:i(t),timingVar:i(n),timingValue:r};a&&(u.timingLabel=i(a)),d(u,o)}}function h(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments[1],n=e.category,r=e.action,o=e.label,u=e.value,c=e.nonInteraction,l=e.transport,f=a(e,["category","action","label","value","nonInteraction","transport"]);if("function"==typeof s){if(!n||!r)return void(0,T.default)("args.category AND args.action are required in event()");var p={hitType:"event",eventCategory:i(n),eventAction:i(r)};o&&(p.eventLabel=i(o)),void 0!==u&&("number"!=typeof u?(0,T.default)("Expected `args.value` arg to be a Number."):p.eventValue=u),void 0!==c&&("boolean"!=typeof c?(0,T.default)("`args.nonInteraction` must be a boolean."):p.nonInteraction=c),void 0!==l&&("string"!=typeof l?(0,T.default)("`args.transport` must be a string."):(-1===["beacon","xhr","image"].indexOf(l)&&(0,T.default)("`args.transport` must be either one of these values: `beacon`, `xhr` or `image`"),p.transport=l)),Object.keys(f).filter(function(e){return"dimension"===e.substr(0,"dimension".length)}).forEach(function(e){p[e]=f[e]}),Object.keys(f).filter(function(e){return"metric"===e.substr(0,"metric".length)}).forEach(function(e){p[e]=f[e]}),d(p,t)}}function y(e,t){var n=e.description,r=e.fatal;if("function"==typeof s){var a={hitType:"exception"};n&&(a.exDescription=i(n)),void 0!==r&&("boolean"!=typeof r?(0,T.default)("`args.fatal` must be a boolean."):a.exFatal=r),d(a,t)}}function v(e,t,n){if("function"!=typeof t)return void(0,T.default)("hitCallback function is required");if("function"==typeof s){if(!e||!e.label)return void(0,T.default)("args.label is required in outboundLink()");var r={hitType:"event",eventCategory:"Outbound",eventAction:"Click",eventLabel:i(e.label)},a=!1,o=function(){a=!0,t()},u=setTimeout(o,250),c=function(){clearTimeout(u),a||t()};r.hitCallback=c,d(r,n)}else setTimeout(t,0)}Object.defineProperty(t,"__esModule",{value:!0}),t.testModeAPI=t.OutboundLink=t.plugin=void 0;var b=Object.assign||function(e){for(var t=1;t0&&t+e.length!==n.length&&e.search(i)>-1&&":"!==n.charAt(t-2)&&("-"!==n.charAt(t+e.length)||"-"===n.charAt(t-1))&&n.charAt(t-1).search(/[^\s-]/)<0?e.toLowerCase():e.substr(1).search(/[A-Z]|\../)>-1?e:e.charAt(0).toUpperCase()+e.substr(1)})}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r;var a=n(1),o=function(e){return e&&e.__esModule?e:{default:e}}(a),i=/^(a|an|and|as|at|but|by|en|for|if|in|nor|of|on|or|per|the|to|vs?\.?|via)$/i},function(e,t,n){"use strict";function r(e){return"/"===e.substring(0,1)?e.substring(1):e}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default=function(e){!function(e,t,n,r,a,o,i){e.GoogleAnalyticsObject=a,e[a]=e[a]||function(){(e[a].q=e[a].q||[]).push(arguments)},e[a].l=1*new Date,o=t.createElement(n),i=t.getElementsByTagName(n)[0],o.async=1,o.src=r,i.parentNode.insertBefore(o,i)}(window,document,"script",e&&e.gaAddress?e.gaAddress:"https://www.google-analytics.com/analytics.js","ga")}},function(e,t,n){"use strict";function r(e){console.info("[react-ga]",e)}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=t.gaCalls=[];t.default={calls:r,ga:function(){for(var e=arguments.length,t=Array(e),n=0;n66?e.setState({headerType:"normal"}):e.setState({headerType:"primary"})}),fetch("https://api.github.com/repos/alibaba/nacos").then(function(e){return e.json()}).then(function(t){e.setState({starCount:t.stargazers_count,forkCount:t.forks_count})})}},{key:"render",value:function(){var e=this.state,t=e.starCount,n=e.forkCount,r=this.getLanguage(),a=N.default[r],o=this.state.headerType,i="primary"===o?(0,d.getLink)("/img/nacos_white.png"):(0,d.getLink)("/img/nacos_colorful.png");return l.default.createElement("div",{className:"home-page"},l.default.createElement("section",{className:"top-section",style:{background:"url("+(0,d.getLink)("/img/black_dot.png")+") repeat",backgroundSize:"14px 14px"}},l.default.createElement(h.default,{currentKey:"home",type:o,logo:i,language:r,onLanguageChange:this.onLanguageChange}),l.default.createElement("div",{className:"vertical-middle"},l.default.createElement("img",{className:"product-logo",src:(0,d.getLink)("/img/nacos.png")}),l.default.createElement("p",{className:"product-desc"},a.brand.briefIntroduction),l.default.createElement("div",{className:"button-area"},a.brand.buttons.map(function(e){return l.default.createElement(v.default,{type:e.type,key:e.type,link:e.link},e.text)})),l.default.createElement("div",{className:"github-buttons"},l.default.createElement("a",{href:"https://github.com/alibaba/nacos",target:"_blank",rel:"noopener noreferrer"},l.default.createElement("div",{className:"star"},l.default.createElement("img",{src:"https://img.alicdn.com/tfs/TB1FlB1JwHqK1RjSZFPXXcwapXa-32-32.png"}),l.default.createElement("span",{className:"count"},t))),l.default.createElement("a",{href:"https://github.com/alibaba/nacos/fork",target:"_blank",rel:"noopener noreferrer"},l.default.createElement("div",{className:"fork"},l.default.createElement("img",{src:"https://img.alicdn.com/tfs/TB1zbxSJwDqK1RjSZSyXXaxEVXa-32-32.png"}),l.default.createElement("span",{className:"count"},n)))),l.default.createElement("div",{className:"version-note"},l.default.createElement("a",{target:"_blank",rel:"noopener noreferrer",href:(0,d.getLink)(a.brand.versionNote.link)},a.brand.versionNote.text),l.default.createElement("a",{target:"_blank",rel:"noopener noreferrer",href:(0,d.getLink)(a.brand.versionNote2.link)},a.brand.versionNote2.text)),l.default.createElement("div",{className:"release-date"},a.brand.releaseDate)),l.default.createElement("div",{className:"animation animation1"}),l.default.createElement("div",{className:"animation animation2"}),l.default.createElement("div",{className:"animation animation3"}),l.default.createElement("div",{className:"animation animation4"}),l.default.createElement("div",{className:"animation animation5"})),l.default.createElement("section",{className:"function-section"},l.default.createElement("h3",null,a.functions.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("div",null,a.functions.list.map(function(e,t){return l.default.createElement(_.default,{func:e,key:t,imgFirst:t%2==0})}))),l.default.createElement("section",{className:"msemap-section"},l.default.createElement("h3",null,a.landscape.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("div",{id:"mse-arc-container"})),l.default.createElement("section",{className:"feature-section"},l.default.createElement("div",{className:"feature-section-body"},l.default.createElement("h3",null,a.features.title),l.default.createElement(O.default,{type:"light"}),l.default.createElement("ul",{className:"feature-list"},a.features.list.map(function(e,t){return l.default.createElement(S.default,{feature:e,key:t})})))),l.default.createElement("section",{className:"users-section"},l.default.createElement("h3",null,a.users.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("p",null,a.users.desc),l.default.createElement("div",{className:"users"},a.users.list.map(function(e,t){return l.default.createElement("div",{className:"user-item",key:t},l.default.createElement("img",{src:e}))}))),l.default.createElement(w.default,{logo:(0,d.getLink)("/img/nacos_gray.png"),language:r}))}}]),t}(m.default);document.getElementById("root")&&f.default.render(l.default.createElement(P,null),document.getElementById("root")),t.default=P},function(e,t){}]); \ No newline at end of file +!function(){"use strict";function n(){for(var e=[],t=0;t1&&void 0!==arguments[1]?arguments[1]:t.key)+arguments[2]})}},{key:"key",get:function(){return this.childDescriptor.key}},{key:"parentNotation",get:function(){return this.parentKlass.constructor.name+"#"+this.parentPropertySignature}},{key:"childNotation",get:function(){return this.childKlass.constructor.name+"#"+this.childPropertySignature}},{key:"parentTopic",get:function(){return this._getTopic(this.parentDescriptor)}},{key:"childTopic",get:function(){return this._getTopic(this.childDescriptor)}},{key:"parentPropertySignature",get:function(){return this._extractTopicSignature(this.parentTopic)}},{key:"childPropertySignature",get:function(){return this._extractTopicSignature(this.childTopic)}}]),m(e,[{key:"assert",value:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"";!0!==e&&this.error("{child} does not properly override {parent}"+t)}},{key:"error",value:function(e){var t=this;throw e=e.replace("{parent}",function(e){return t.parentNotation}).replace("{child}",function(e){return t.childNotation}),new SyntaxError(e)}}]),e}(),y=[function(e){return e.toLowerCase()},function(e){return e.toUpperCase()},function(e){return e+"s"},function(e){return e.slice(0,-1)},function(e){return e.slice(1,e.length)}]},function(e,t,n){"use strict";function r(e,t,r,l){var s=u(l,3),f=s[0],d=void 0===f?null:f,p=s[1],m=void 0!==p&&p,g=s[2],h=void 0===g?c:g;if(!a.__enabled)return a.__warned||(h.warn("console.profile is not supported. All @profile decorators are disabled."),a.__warned=!0),r;var y=r.value;if(null===d&&(d=e.constructor.name+"."+t),"function"!=typeof y)throw new SyntaxError("@profile can only be used on functions, not: "+y);return i({},r,{value:function(){var e=Date.now(),t=n.i(o.c)(this);(!0===m&&!t.profileLastRan||!1===m||"number"==typeof m&&e-t.profileLastRan>m||"function"==typeof m&&m.apply(this,arguments))&&(h.profile(d),t.profileLastRan=e);try{return y.apply(this,arguments)}finally{h.profileEnd(d)}}})}function a(){for(var e=arguments.length,t=Array(e),a=0;a1){if(o=e({path:"/"},r.defaults,o),"number"==typeof o.expires){var u=new Date;u.setMilliseconds(u.getMilliseconds()+864e5*o.expires),o.expires=u}o.expires=o.expires?o.expires.toUTCString():"";try{i=JSON.stringify(a),/^[\{\[]/.test(i)&&(a=i)}catch(e){}a=n.write?n.write(a,t):encodeURIComponent(String(a)).replace(/%(23|24|26|2B|3A|3C|3E|3D|2F|3F|40|5B|5D|5E|60|7B|7D|7C)/g,decodeURIComponent),t=encodeURIComponent(String(t)),t=t.replace(/%(23|24|26|2B|5E|60|7C)/g,decodeURIComponent),t=t.replace(/[\(\)]/g,escape);var c="";for(var l in o)o[l]&&(c+="; "+l,!0!==o[l]&&(c+="="+o[l]));return document.cookie=t+"="+a+c}t||(i={});for(var s=document.cookie?document.cookie.split("; "):[],f=/(%[0-9A-Z]{2})+/g,d=0;d=0||Object.prototype.hasOwnProperty.call(e,r)&&(n[r]=e[r]);return n}function o(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t1?t-1:0),r=1;r0&&(F.apply(void 0,t),q&&((0,C.default)("called ga('arguments');"),(0,C.default)("with arguments: "+JSON.stringify(t)))),window.ga}function f(e,t){return e?"object"!==(void 0===e?"undefined":w(e))?void(0,P.default)("Expected `fieldsObject` arg to be an Object"):(0===Object.keys(e).length&&(0,P.default)("empty `fieldsObject` given to .set()"),u(t,"set",e),void(q&&((0,C.default)("called ga('set', fieldsObject);"),(0,C.default)("with fieldsObject: "+JSON.stringify(e))))):void(0,P.default)("`fieldsObject` is required in .set()")}function d(e,t){u(t,"send",e),q&&((0,C.default)("called ga('send', fieldObject);"),(0,C.default)("with fieldObject: "+JSON.stringify(e)),(0,C.default)("with trackers: "+JSON.stringify(t)))}function p(e,t,n){if(!e)return void(0,P.default)("path is required in .pageview()");var r=(0,S.default)(e);if(""===r)return void(0,P.default)("path cannot be an empty string in .pageview()");var a={};if(n&&(a.title=n),"function"==typeof s&&(u(t,"send",b({hitType:"pageview",page:r},a)),q)){(0,C.default)("called ga('send', 'pageview', path);");var o="";n&&(o=" and title: "+n),(0,C.default)("with path: "+r+o)}}function m(e,t){if(!e)return void(0,P.default)("modalName is required in .modalview(modalName)");var n=(0,_.default)((0,S.default)(e));if(""===n)return void(0,P.default)("modalName cannot be an empty string or a single / in .modalview()");if("function"==typeof s){var r="/modal/"+n;u(t,"send","pageview",r),q&&((0,C.default)("called ga('send', 'pageview', path);"),(0,C.default)("with path: "+r))}}function g(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=e.category,n=e.variable,r=e.value,a=e.label,o=arguments[1];if("function"==typeof s){if(!t||!n||!r||"number"!=typeof r)return void(0,P.default)("args.category, args.variable AND args.value are required in timing() AND args.value has to be a number");var u={hitType:"timing",timingCategory:i(t),timingVar:i(n),timingValue:r};a&&(u.timingLabel=i(a)),d(u,o)}}function h(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments[1],n=e.category,r=e.action,o=e.label,u=e.value,c=e.nonInteraction,l=e.transport,f=a(e,["category","action","label","value","nonInteraction","transport"]);if("function"==typeof s){if(!n||!r)return void(0,P.default)("args.category AND args.action are required in event()");var p={hitType:"event",eventCategory:i(n),eventAction:i(r)};o&&(p.eventLabel=i(o)),void 0!==u&&("number"!=typeof u?(0,P.default)("Expected `args.value` arg to be a Number."):p.eventValue=u),void 0!==c&&("boolean"!=typeof c?(0,P.default)("`args.nonInteraction` must be a boolean."):p.nonInteraction=c),void 0!==l&&("string"!=typeof l?(0,P.default)("`args.transport` must be a string."):(-1===["beacon","xhr","image"].indexOf(l)&&(0,P.default)("`args.transport` must be either one of these values: `beacon`, `xhr` or `image`"),p.transport=l)),Object.keys(f).filter(function(e){return"dimension"===e.substr(0,"dimension".length)}).forEach(function(e){p[e]=f[e]}),Object.keys(f).filter(function(e){return"metric"===e.substr(0,"metric".length)}).forEach(function(e){p[e]=f[e]}),d(p,t)}}function y(e,t){var n=e.description,r=e.fatal;if("function"==typeof s){var a={hitType:"exception"};n&&(a.exDescription=i(n)),void 0!==r&&("boolean"!=typeof r?(0,P.default)("`args.fatal` must be a boolean."):a.exFatal=r),d(a,t)}}function v(e,t,n){if("function"!=typeof t)return void(0,P.default)("hitCallback function is required");if("function"==typeof s){if(!e||!e.label)return void(0,P.default)("args.label is required in outboundLink()");var r={hitType:"event",eventCategory:"Outbound",eventAction:"Click",eventLabel:i(e.label)},a=!1,o=function(){a=!0,t()},u=setTimeout(o,250),c=function(){clearTimeout(u),a||t()};r.hitCallback=c,d(r,n)}else setTimeout(t,0)}Object.defineProperty(t,"__esModule",{value:!0}),t.testModeAPI=t.OutboundLink=t.plugin=void 0;var b=Object.assign||function(e){for(var t=1;t0&&t+e.length!==n.length&&e.search(i)>-1&&":"!==n.charAt(t-2)&&("-"!==n.charAt(t+e.length)||"-"===n.charAt(t-1))&&n.charAt(t-1).search(/[^\s-]/)<0?e.toLowerCase():e.substr(1).search(/[A-Z]|\../)>-1?e:e.charAt(0).toUpperCase()+e.substr(1)})}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r;var a=n(1),o=function(e){return e&&e.__esModule?e:{default:e}}(a),i=/^(a|an|and|as|at|but|by|en|for|if|in|nor|of|on|or|per|the|to|vs?\.?|via)$/i},function(e,t,n){"use strict";function r(e){return"/"===e.substring(0,1)?e.substring(1):e}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0}),t.default=function(e){!function(e,t,n,r,a,o,i){e.GoogleAnalyticsObject=a,e[a]=e[a]||function(){(e[a].q=e[a].q||[]).push(arguments)},e[a].l=1*new Date,o=t.createElement(n),i=t.getElementsByTagName(n)[0],o.async=1,o.src=r,i.parentNode.insertBefore(o,i)}(window,document,"script",e&&e.gaAddress?e.gaAddress:"https://www.google-analytics.com/analytics.js","ga")}},function(e,t,n){"use strict";function r(e){console.info("[react-ga]",e)}Object.defineProperty(t,"__esModule",{value:!0}),t.default=r},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=t.gaCalls=[];t.default={calls:r,ga:function(){for(var e=arguments.length,t=Array(e),n=0;n66?e.setState({headerType:"normal"}):e.setState({headerType:"primary"})}),fetch("https://api.github.com/repos/alibaba/nacos").then(function(e){return e.json()}).then(function(t){e.setState({starCount:t.stargazers_count,forkCount:t.forks_count})})}},{key:"render",value:function(){var e=this.state,t=e.starCount,n=e.forkCount,r=this.getLanguage(),a=N.default[r],o=this.state.headerType,i="primary"===o?(0,d.getLink)("/img/nacos_white.png"):(0,d.getLink)("/img/nacos_colorful.png");return l.default.createElement("div",{className:"home-page"},l.default.createElement("section",{className:"top-section",style:{background:"url("+(0,d.getLink)("/img/black_dot.png")+") repeat",backgroundSize:"14px 14px"}},l.default.createElement(h.default,{currentKey:"home",type:o,logo:i,language:r,onLanguageChange:this.onLanguageChange}),l.default.createElement("div",{className:"vertical-middle"},l.default.createElement("img",{className:"product-logo",src:(0,d.getLink)("/img/nacos.png")}),l.default.createElement("p",{className:"product-desc"},a.brand.briefIntroduction),l.default.createElement("div",{className:"button-area"},a.brand.buttons.map(function(e){return l.default.createElement(v.default,{type:e.type,key:e.type,link:e.link},e.text)})),l.default.createElement("div",{className:"github-buttons"},l.default.createElement("a",{href:"https://github.com/alibaba/nacos",target:"_blank",rel:"noopener noreferrer"},l.default.createElement("div",{className:"star"},l.default.createElement("img",{src:"https://img.alicdn.com/tfs/TB1FlB1JwHqK1RjSZFPXXcwapXa-32-32.png"}),l.default.createElement("span",{className:"count"},t))),l.default.createElement("a",{href:"https://github.com/alibaba/nacos/fork",target:"_blank",rel:"noopener noreferrer"},l.default.createElement("div",{className:"fork"},l.default.createElement("img",{src:"https://img.alicdn.com/tfs/TB1zbxSJwDqK1RjSZSyXXaxEVXa-32-32.png"}),l.default.createElement("span",{className:"count"},n)))),l.default.createElement("div",{className:"version-note"},l.default.createElement("a",{target:"_blank",rel:"noopener noreferrer",href:(0,d.getLink)(a.brand.versionNote.link)},a.brand.versionNote.text),l.default.createElement("a",{target:"_blank",rel:"noopener noreferrer",href:(0,d.getLink)(a.brand.versionNote2.link)},a.brand.versionNote2.text)),l.default.createElement("div",{className:"release-date"},a.brand.releaseDate)),l.default.createElement("div",{className:"animation animation1"}),l.default.createElement("div",{className:"animation animation2"}),l.default.createElement("div",{className:"animation animation3"}),l.default.createElement("div",{className:"animation animation4"}),l.default.createElement("div",{className:"animation animation5"})),l.default.createElement("section",{className:"function-section"},l.default.createElement("h3",null,a.functions.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("div",null,a.functions.list.map(function(e,t){return l.default.createElement(_.default,{func:e,key:t,imgFirst:t%2==0})}))),l.default.createElement("section",{className:"msemap-section"},l.default.createElement("h3",null,a.landscape.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("div",{id:"mse-arc-container"})),l.default.createElement("section",{className:"feature-section"},l.default.createElement("div",{className:"feature-section-body"},l.default.createElement("h3",null,a.features.title),l.default.createElement(O.default,{type:"light"}),l.default.createElement("ul",{className:"feature-list"},a.features.list.map(function(e,t){return l.default.createElement(S.default,{feature:e,key:t})})))),l.default.createElement("section",{className:"users-section"},l.default.createElement("h3",null,a.users.title),l.default.createElement(O.default,{type:"dark"}),l.default.createElement("p",null,a.users.desc),l.default.createElement("div",{className:"users"},a.users.list.map(function(e,t){return l.default.createElement("div",{className:"user-item",key:t},l.default.createElement("img",{src:e}))}))),l.default.createElement(w.default,{logo:(0,d.getLink)("/img/nacos_gray.png"),language:r}))}}]),t}(m.default);document.getElementById("root")&&f.default.render(l.default.createElement(T,null),document.getElementById("root")),t.default=T},function(e,t){}]); \ No newline at end of file diff --git a/docs/en-us/v2/guide/admin/console-guide.md b/docs/en-us/v2/guide/admin/console-guide.md index 3d656e19c9c..13b72271a4e 100644 --- a/docs/en-us/v2/guide/admin/console-guide.md +++ b/docs/en-us/v2/guide/admin/console-guide.md @@ -135,19 +135,21 @@ INSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN'); ### Close the login function -As part of its own development console, do not want to be nacos security filter interceptor.Therefore nacos support custom close the login functionFind the configuration file `${nacoshome}/conf/application.properties`. The properties, replace the following content. +Before `version 2.2.2`, the Nacos default console would always redirect to the login page, whether the authentication function was enabled, which could mislead users into thinking that there was authentication function, but in fact it was disabled, posing a security risk. -``` -## spring security config -### turn off security -spring.security.enabled=false -management.security=false -security.basic.enabled=false -nacos.security.ignore.urls=/** +After consultation and discussion with the community and security engineers, it was decided that when the authentication switch is turned off, the console login function will be automatically disabled when using the Nacos default console. -#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/** +Therefore, starting from `version 2.2.2`, when the authentication switch `nacos.core.auth.enabled` is `false`, the Nacos default console will no longer redirect to the login page, but will add a page prompt indicating that the current cluster has not enabled authentication function. -``` +At the same time, a new interface `com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled` has been added for custom authentication plugins to control the login page. By default, it returns `false`. + +### Close default console ui + +Some companies or users may want to disable the default console of Nacos and use their own unified platform for Nacos configuration and service management. They may also want to separate the authentication for console operations and client requests, meaning that authentication is required for console operations but not for client requests. + +Starting from `version 2.3.0`, maintainers can use the `nacos.console.ui.enabled` property in `${nacoshome}/conf/application.properties` to enable or disable the Nacos default console. It is `true` by default. + +When the default console is disabled, it will read the content of the `${nacoshome}/conf/console-guide.conf` file and generate a guide page in the default console. This allows maintainers to customize the guide to redirect users who would have used the default console to their own unified platform for operations. ### Session time diff --git a/docs/en-us/v2/plugin/config-change-plugin.md b/docs/en-us/v2/plugin/config-change-plugin.md new file mode 100644 index 00000000000..c3671fd68b6 --- /dev/null +++ b/docs/en-us/v2/plugin/config-change-plugin.md @@ -0,0 +1,140 @@ +--- +title: Config Change Plugin +keywords: Config Change,Config audit,Config format check,webhook +description: Nacos support Config Change Plugin. +--- + +> Translated by AI. + +# Config Change Plugin + +Community has long been hoping for Nacos Configuration Center to provide notifications to specific systems when configurations change. These notifications are used for recording, warning, and auditing purposes. Before version 2.3.0, the only way to achieve this was by simulating Nacos client subscription to the configurations. This approach involved subscribing to changes in core configurations and executing functionalities such as sending records and warnings upon receiving change notifications. + +However, this implementation had a few significant issues. Firstly, individual configurations needed to be added one by one, making it difficult to capture all configuration changes. Secondly, functionalities could only be executed after configuration changes, and there was no capability for performing pre-change operations such as format validation or whitelist validation. + +To address these limitations, starting from Nacos version 2.3.0, Nacos introduced support for injecting configuration change plugins through the [SPI](https://docs.oracle.com/javase/tutorial/sound/SPI-intro.html). This allows users to define custom plugins to execute specific logic before and after configuration changes. Examples of such logic include format validation, whitelist validation, and webhook integrations. + +These enhancements provide more flexibility and control for users to implement their own custom logic before and after configuration changes in Nacos Configuration Center. + +## Concepts in Config Change Plugin + +Nacos's configuration change plugin design takes inspiration from the aspect-oriented programming (AOP) paradigm. It treats configuration change operations, such as adding, updating, and deleting, as the `pointcuts` and weaves the plugins `before` and `after` these pointcuts. + +### ConfigChangePointCutTypes + +Nacos has categorized the configuration change operations based on their behaviors and sources. These configuration change operations are defined as several `ConfigChangePointCutTypes` in `com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes`. The specific details are as follows: + +|PointCut Name|Description|Start version| +|-----|-----|-----| +|PUBLISH_BY_HTTP|Configuration is published through the HTTP interface, including creating and modifying configurations.|2.3.0| +|PUBLISH_BY_RPC|Configuration is published through the gRPC interface, including creating and modifying configurations.|2.3.0| +|REMOVE_BY_HTTP|Configuration is removed through the HTTP interface.|2.3.0| +|REMOVE_BY_RPC|Configuration is removed through the gRPC interface.|2.3.0| +|IMPORT_BY_HTTP|Configuration is imported through the HTTP interface.|2.3.0| +|REMOVE_BATCH_HTTP|Configurations are batch removed through the HTTP interface.|2.3.0| + +### ConfigChangeExecuteTypes + +In Nacos, the configuration change plugins need to be executed before or after the `ConfigChangePointCutTypes` by selecting the `ConfigChangeExecuteTypes`. These execution types are defined in `com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes`. The specific details are as follows: + +|Execute Type|Description|Start version| +|-----|-----|-----| +|EXECUTE_BEFORE_TYPE|Plugin execute **Before** `ConfigChangePointCutTypes`|2.3.0| +|EXECUTE_AFTER_TYPE|Plugin execute **After** `ConfigChangePointCutTypes`|2.3.0| + +## Plugin Development + +To develop a config change plugin for the Nacos server, you first need to depend on the relevant API of the config change plugin. + +```xml + + com.alibaba.nacos + nacos-config-plugin + ${project.version} + +``` + +`${project.version}` is the Nacos version, at least `2.3.0`. + +Then implement `com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService` interface, which include following method: + +|Method name|Parameters|Return|Description| +|-----|-----|-----|-----| +|getServiceType|void|String|Returns the name of the plugin, which is used to differentiate between different types of plugin implementations.| +|getOrder|void|int|Returns the execution order of the plugin. The configuration change plugin is designed with a chain plugin pattern, where multiple plugins are executed in order. The smaller the return value of getOrder, the earlier the plugin is executed.| +|executeType|void|ConfigChangeExecuteTypes|Returns `ConfigChangeExecuteTypes` implemented by the plugin.| +|pointcutMethodNames|void|ConfigChangePointCutTypes[]|Returns `ConfigChangePointCutTypes` where the plugin is implemented.| +|execute|ConfigChangeRequest,ConfigChangeResponse|void|Executes the actual logic of the plugin.| + +`ConfigChangeRequest` and `ConfigChangeResponse` refer to the contents passed in during the execution of logic and the resulting execution outcome, respectively. + +`ConfigChangeRequest` mainly include follow contents: + +|Field name|Type|Description| +|-----|-----|-----| +|requestType|ConfigChangePointCutTypes|The pointcut types of this configuration change| +|requestArgs|HashMap|The actual parameters of this configuration change, mainly including `namespace`, `group`, `dataId`, `content`, etc., with different parameters for different pointcut types| + +`ConfigChangeResponse` mainly include follow contents: + +|Field name|Type|Description| +|-----|-----|-----| +|responseType|ConfigChangePointCutTypes|The pointcut types of this configuration change| +|isSuccess|boolean|Indicates whether the execution is successful. When the return value is `false`, the configuration change will be intercepted and the failure result will be returned directly| +|retVal|Object|Return content, a reserved field that is currently not used| +|msg|String|Execution result information, obtained when `isSuccess` is `false`, used to return information to the client| +|args|Object[]|The execution parameters of the configuration change, effective for the `EXECUTE_BEFORE_TYPE` plugin type. It can be used to modify the content of the actual executed configuration change, such as changing certain values in content to other values| + +### Load Plugin + +After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under `${nacos-server.path}/plugins` directly. + +After Adding plugins into classpath, also need to modify some configuration in `${nacos-server.path}/conf/application.properties`. + +```properties +### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method. +nacos.core.config.plugin.${configChangePluginName}.enabled=true +``` + +After restarting the Nacos cluster and completing the startup, you can see the following logs in `${nacos-server.path}/logs/plugin-control.log`: + +```text +[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully. +``` + +### Custom Plugin properties + +Some plugins may want to set certain parameters through a configuration file. Custom plugins can modify the following configurations in `${nacos-server.path}/conf/application.properties` to achieve this: + +```properties +### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method. +nacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue} +``` + +In `ConfigChangeRequest`, getting properties by following way: + +```Java +final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES); +final String ${propertyKey} = properties.getProperty("${propertyKey}"); +``` + +## Plugin Demo + +In the [nacos-group/nacos-plugin](https://github.com/nacos-group/nacos-plugin) repository, there is a demo implementation of a configuration change plugin. This demo plugin implements validation of the configuration content format, validation of the whitelist for importing configuration names, and a webhook callback after the change. To use this plugin, you need to package it as a JAR/ZIP file and place it in the classpath of the Nacos server. Additionally, add the following configuration in `${nacos-server.path}/conf/application.properties`: + +```properties +# webhook +#nacos.core.config.plugin.webhook.enabled=true +# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html +#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=*** +# The content push max capacity ,byte +#nacos.core.config.plugin.webhook.contentMaxCapacity=102400 + +# whitelist +#nacos.core.config.plugin.whitelist.enabled=true +# The import file suffixs +#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html + +# fileformatcheck,which validate the import file of type and content +#nacos.core.config.plugin.fileformatcheck.enabled=true +``` \ No newline at end of file diff --git a/docs/en-us/v2/plugin/control-plugin.md b/docs/en-us/v2/plugin/control-plugin.md new file mode 100644 index 00000000000..630be6e6d28 --- /dev/null +++ b/docs/en-us/v2/plugin/control-plugin.md @@ -0,0 +1,235 @@ +--- +title: control plugin +keywords: anti-fragile,control limit,connection limit,TPS +description: Nacos support control plugin. +--- + +> Translated by AI. + +# Control Plugin + +Starting from version 2.3.0, Nacos supports injecting control related plugins through [SPI](https://docs.oracle.com/javase/tutorial/sound/SPI-intro.html), and selecting a specific plugin implementation as the actual control capability in the `application.properties` configuration file. This document provides a detailed introduction on how to implement an control plugin and how to make it effective. + +## Concepts in Control Plugin + +Anti-fragility is a strategy that restricts access to a **certain resource** on the server when the **frequency and number** of accesses reach a certain level. It is used to protect the server from quickly rejecting requests under high pressure, preventing widespread unavailability caused by excessive resource access and exhaustion. The Nacos control plugin abstracts information primarily into `control point` and `control rules`. + +### Control Point + +The control point correspond to the mapping of resources occupied when making requests to the server. Currently, they mainly focus on `Connections` and `Transactions Per Second (TPS)`. + +- The "Connections" control point primarily monitors the number of long connections used by Nacos 2.X clients and the number of long polling connections used by Nacos 1.X clients. These two types of connections are monitored independently. +- The "Transactions Per Second (TPS)" control point mainly monitors the frequency of access to core interfaces in the Nacos server. Similar operation interfaces are considered as the same monitor point. For example, the registration service v1 interface and v2 interface are treated as the same monitor point. Please refer to the document for specific TPS monitor [point names](#1.1). + +### Control Rule + +The control rules are different limitation rules that are executed for each control point. They are specifically categorized as "Connection Control Rules" `ConnectionControlRule` and "Transactions Per Second Control Rules" `TpsControlRule`. + +`ConnectionControlRule` mainly include follow contents: + +|Field name|Type|Description| +|-----|-----|-----| +| countLimit | int | Total count limit for connections. Default is -1, indicating no limitation. | +| monitorIpList | Set | List of IPs to be monitored by trace. It is used to observe the operations performed on the corresponding IP connections in detail. Once added, the connection requests from the corresponding IPs will be logged in detail in the `remote-digest.log` file. | + +`TpsControlRule` mainly include follow contents: + +|Field name|Type|Description| +|-----|-----|-----| +| pointName | String | Name of the control point corresponding to the rule. | +| pointRule | RuleDetail | Specific details of the rule content. | + +And `RuleDetail` mainly include follow contents: + +|Field name|Type|Description| +|-----|-----|-----| +| ruleName | String | Name of the rule. It is different from the control point name. A control point can have multiple rule names. | +| maxCount | int | Total count limit for TPS. Default is -1, indicating no limitation. | +| period | TimeUnit | The period in which the rule is effective, such as counting at the second level or minute level. Default is `TimeUnit.SECONDS` for second level. | +| monitorType | String | Monitoring type, can be either `monitor` or `intercept`. It corresponds to monitoring mode (only counting and printing TPS, even if the rule is triggered, no interception) and interception mode. | + +## Plugin Development + +To develop a control plugin for the Nacos server, you first need to depend on the relevant API of the control plugin. + +```xml + + com.alibaba.nacos + nacos-control-plugin + ${project.version} + +``` + +`${project.version}` is the Nacos version, at least `2.3.0`. + +Then to extend `com.alibaba.nacos.plugin.control.connection.ConnectionControlManager` abstract class and `com.alibaba.nacos.plugin.control.tps.TpsControlManager`abstract class,implement their methods; Then implement `com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder` interface to build the two above classes; Finally add the SPI file to `META-INF/services`. + +`com.alibaba.nacos.plugin.control.connection.ConnectionControlManager` need implement follow methods: + +|Method name|Parameters|Return|Description| +|-----|-----|-----|-----| +|applyConnectionLimitRule|ConnectionControlRule|void|Apply new connection limit rule| +|check|ConnectionCheckRequest|ConnectionCheckResponse|To determine if the connection count rule is hit, if the success field in the ConnectionCheckResponse is `false`, new connections will be rejected.| + +`com.alibaba.nacos.plugin.control.tps.TpsControlManager` need implement follow methods: + +|Method name|Parameters|Return|Description| +|-----|-----|-----|-----| +| registerTpsPoint | String | void | Registers a TPS control point. The Nacos server will register the current TPS control point to the plugin during startup. The input parameter is the name of the TPS control point. Please refer to the document for [point names](#1.1). The plugin needs to maintain a TpsBarrier within this method to record TPS and rule content. For more details, refer to [Custom Time Windows for TPS](#1.2). | +| applyTpsRule | String, TpsControlRule | void | Applies a new TPS rule and associates it with the TPS control point name for update. | +| check | TpsCheckRequest | TpsCheckResponse | Checks if the TPS rule is hit. If the success field in the TpsCheckResponse is false, new requests will be rejected. | + +`com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder` need implement follow methods: + +|Method name|Parameters|Return|Description| +|-----|-----|-----|-----| +| getName | void | String | Returns the name of the plugin. It is used to match the specified type in the configuration file and use the matched plugin. | +| buildConnectionControlManager | void | ConnectionControlManager | Creates an implementation of `ConnectionControlManager` for the plugin. When it is null, the `no limit` implementation will be used. | +| buildTpsControlManager | void | TpsControlManager | Creates an implementation of `TpsControlManager` for the plugin. When it is null, the `no limit` implementation will be used. | + +### Load Plugin + +After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under `${nacos-server.path}/plugins` directly. + +After Adding plugins into classpath, also need to modify some configuration in `${nacos-server.path}/conf/application.properties`. + +```properties +### The name of the control plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder's getName method. +nacos.plugin.control.manager.type=${controlPluginName} +``` + +After restarting the Nacos cluster and completing the startup, you can see the following logs in `${nacos-server.path}/logs/plugin-control.log`: + +```text +Found control manager plugin of name=${controlPluginName} + +Build connection control manager, class=${your plugin ConnectionControlManager class} + +Build tps control manager, class=${your plugin TpsControlManager class} +``` + +## Use the default Nacos Control Plugin + +Starting from version 2.3.0, Nacos comes with a built-in simple control plugin implementation, which can limit the connection count and specified interface TPS of the Nacos server. + +### Enable the default Nacos Control Plugin + +Modify the following configurations in `${nacos-server.path}/conf/application.properties`: + +```properties +nacos.plugin.control.manager.type=nacos +``` + +### Setting control plugin for default Nacos control plugin + +You can modify and set control rules by creating and modifying the control rule file. By default, the rules for the control plugin are defined in JSON format. + +For example, if you want to set the connection limit to 100, you can perform the following steps: + +```shell +mkdir -p ${nacos.home}/data/connection/ +echo '{"countLimit": 100}' > ${nacos.home}/data/connection/limitRule +``` + +Then restart Nacos server node. + +What's more, if you want set the TPS of query config as 100, you can perform the following steps: + + ```shell + mkdir -p ${nacos.home}/data/tps/ + # ConfigQuery is the PointName of the query config API. + echo '{"pointName":"ConfigQuery","pointRule":{"maxCount":100,"monitorType":"intercept"}}' > ${nacos.home}/data/tps/ConfigQuery + ``` + +Then restart Nacos server node. + +More control rules and control point names please move to [point names](#1.1). + +### The Storage Dir of Control Rules + +The built-in simple control plugin implementation in Nacos stores and reads control rules through the local file system. By default, the rules are stored in `${nacos.home}/data/connection` and `${nacos.home}/data/tps` directories. + +If you want to change the directory for storing the rule files, you can modify the following configuration in `${nacos-server.path}/conf/application.properties`: + +```properties +nacos.plugin.control.rule.local.basedir=${expectedDir} +``` + +In this way, the control rules will be stored in `${expectedDir}/data/connection` and `${expectedDir}/data/tps`. + +

+ +## Supported Control PointNames + +|control pointNames|meaning|description|started version| +|-----|-----|-----|-----| +|connection|Total Connections|Maximum Supported Connection Limit for a Specific Node|2.3.0| +|ConfigPublish|Configuration Publish Interface TPS|Maximum Supported TPS Limit for Configuration Publishing on a Specific Node, including both HTTP and gRPC access sources|2.3.0| +|ConfigQuery|Configuration Query Interface TPS|Maximum Supported TPS Limit for Configuration Querying on a Specific Node, including both HTTP and gRPC access sources|2.3.0| +|ConfigRemove|Configuration Removal Interface TPS|Maximum Supported TPS Limit for Configuration Removal on a Specific Node, including both HTTP and gRPC access sources|2.3.0| +|ConfigListen|Configuration Listening Interface TPS|Maximum Supported TPS Limit for Configuration Listening on a Specific Node, only including gRPC access source|2.3.0| +|RemoteNamingInstanceRegisterDeregister|Remote Naming Instance Register and Deregister Interface TPS|TPS Limit for Registering or Deregistering Service Instances, only including gRPC access source|2.3.0| +|RemoteNamingInstanceBatchRegister|Remote Naming Instance Batch Register Interface TPS|TPS Limit for Batch Registering Service Instances, only including gRPC access source|2.3.0| +|RemoteNamingServiceListQuery|Remote Naming Service List Query Interface TPS|TPS Limit for Service List Query, only including gRPC access source|2.3.0| +|RemoteNamingServiceQuery|Remote Naming Service Query Interface TPS|TPS Limit for Service Query, only including gRPC access source|2.3.0| +|RemoteNamingServiceSubscribeUnSubscribe|Remote Naming Service Subscribe and Unsubscribe Interface TPS|TPS Limit for Service Subscribe and Unsubscribe, only including gRPC access source|2.3.0| +|NamingInstanceRegister|Naming Instance Register Interface TPS|TPS Limit for Registering Service Instances, only including HTTP access source|2.3.0| +|NamingInstanceDeregister|Naming Instance Deregister Interface TPS|TPS Limit for Deregistering Service Instances, only including HTTP access source|2.3.0| +|NamingInstanceUpdate|Naming Instance Metadata Update Interface TPS|TPS Limit for Updating Service Instance Metadata, only including HTTP access source|2.3.0| +|NamingInstanceMetadataUpdate|Naming Instance Batch Metadata Update Interface TPS|TPS Limit for Batch Updating Service Instance Metadata, only including HTTP access source|2.3.0| +|NamingServiceSubscribe|Naming Service Query and Subscribe Interface TPS|TPS Limit for Service Query and Subscribe, only including HTTP access source|2.3.0| +|NamingInstanceQuery|Single Service Instance Query Interface TPS|TPS Limit for Querying Single Service Instance, only including HTTP access source|2.3.0| +|HttpHealthCheck|Service Instance Heartbeat Renewal Interface TPS|TPS Limit for Service Instance Heartbeat Renewal, only including HTTP access source|2.3.0| +|NamingServiceRegister|Service Create Interface TPS|TPS Limit for Creating Services, different from `NamingInstanceRegister`, this monitoring point represents the TPS for the interface of creating an empty service, only including HTTP access source|2.3.0| +|NamingServiceDeregister|Service Delete Interface TPS|TPS Limit for Deleting Services, different from `NamingInstanceDeregister`, this monitoring point represents the TPS for the interface of deleting services, only including HTTP access source|2.3.0| +|NamingServiceQuery|Service Query Interface TPS|TPS Limit for Service Query, different from `NamingInstanceQuery`, this monitoring point represents the TPS for the interface of querying service information, only including HTTP access source|2.3.0| +|NamingServiceListQuery|Service List Query Interface TPS|TPS Limit for Service List Query, different from `NamingServiceSubscribe`, this monitoring point represents the TPS for the interface of service list query, only including HTTP access source|2.3.0| +|NamingServiceUpdate|Service Metadata Update Interface TPS|TPS Limit for Service Metadata Update, different from `NamingInstanceUpdate`, this monitoring point represents the TPS for the interface of updating service metadata, only including HTTP access source|2.3.0| + +## Advanced Development + +Nacos control plugin also supports advanced extensions to meet the higher requirements of developers and users in this aspect. + +### External Storage for Control Rules + +By default, the Nacos control plugin only supports storing and modifying control rules for individual nodes through the local file system. For users with large-scale or multiple clusters, adjusting each node individually can be time-consuming and cumbersome. Additionally, in many containerized environments, there may be issues with disk mounting and persistence for the local file system. + +To address these concerns, the Nacos control plugin allows the option to add an external storage for unified storage and distribution of control rules. This external storage can be implemented by the plugin itself, such as using a `database` or a `configuration center`. + +To enable external storage for control rules, you need to implement the `com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder` interface in your plugin development, and place the plugin jar file along with the interface implementation in the `${nacos-server.path}/plugins` directory. + +After placing the plugin files, you need to modify the following configuration in `${nacos-server.path}/conf/application.properties`: + +```properties +nacos.plugin.control.rule.external.storage=${controlPluginName} +``` + +Afterwards, restart the Nacos node for the changes to take effect. + +### Dynamic Loading of Control Rules + +In custom plugin implementations, there are two ways to dynamically load contrl rules: + +1. Call the `com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule` method or `com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule` method. + +2. Publish a `ConnectionLimitRuleChangeEvent` or `TpsControlRuleChangeEvent` event using `NotifyCenter.publishEvent()`. + +These methods allow you to reload and update the control rules dynamically in your custom plugin implementation. + +### Custom Format Parse for Control Rules + +By default, Nacos uses the `Json` format as the text format for control rules. However, plugin developers can use other formats such as `Yaml` or other custom formats for parsing. + +To use a custom format for rule parsing, you can override the `com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParser` and `com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser` methods. Implement a custom format rule parser by creating a `RuleParser` that can parse rules in your desired format. Nacos will then use this custom rule parser to parse the rule text. + +Additionally, you can enhance the default custom rules by parsing them into more advanced rules. This can be done in combination with the custom logic of your plugin, allowing for more advanced control. + +

+ +### Custom Time Windows for TPS + +It is well-known that different time window algorithms can lead to significant differences in TPS statistics. Nacos defaults to a simple per-second statistical method, where TPS is counted based on clock seconds. This is sufficient for most scenarios, but for users with higher accuracy requirements, more precise methods such as sliding windows may be needed for TPS statistics. + +In such cases, plugin developers can customize the time window and statistical methods for TPS by inheriting from `com.alibaba.nacos.plugin.control.tps.barrier.TpsBarrier` and `com.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier`. Additionally, the `com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator` method needs to be overridden. This allows for the generation of the corresponding custom implementation during plugin initialization and dynamic loading of control rules. + +By implementing these customizations, plugin developers can control the time window and statistical methods used for TPS in a way that meets the specific accuracy requirements of their users. diff --git a/docs/zh-cn/v2/guide/admin/console-guide.md b/docs/zh-cn/v2/guide/admin/console-guide.md index 2fb9b710320..3f8977426ab 100644 --- a/docs/zh-cn/v2/guide/admin/console-guide.md +++ b/docs/zh-cn/v2/guide/admin/console-guide.md @@ -133,21 +133,27 @@ INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHz INSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN'); ``` +

+ ### 关闭登录功能 -由于部分公司自己开发控制台,不希望被nacos的安全filter拦截。因此nacos支持定制关闭登录功能找到配置文件 `${nacoshome}/conf/application.properties`, 替换以下内容即可。 +Nacos默认控制台在`2.2.2`版本前,无论是否开启[鉴权](../user/auth.md)功能,默认控制台都会跳转到登录页,导致用户被误导认为控制台存在鉴权功能,实际没有开启鉴权,存在安全隐患。 -``` -## spring security config -### turn off security -spring.security.enabled=false -management.security=false -security.basic.enabled=false -nacos.security.ignore.urls=/** +经过社区和安全工程师协商讨论,需要在使用Nacos默认控制台时,鉴权开关关闭时将会自动关闭控制台登录功能。 -#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/** +因此从`2.2.2`版本开始,当鉴权开关`nacos.core.auth.enabled`关闭时,Nacos默认控制台将不再跳转登录页,同时添加页面提示,提示当前集群未开启鉴权功能。 -``` +同时针对自定义的[鉴权插件](../../plugin/auth-plugin.md)添加新接口`com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled(默认返回false)`来对自定义插件进行登录页控制。 + +

+ +### 关闭默认控制台 + +部分公司或用户希望关闭默认控制台,使用公司的统一平台进行Nacos的配置和服务管理;或将控制台鉴权和客户端访问的鉴权分离,即控制台操作进行鉴权但客户端请求不进行鉴权。 + +从`2.3.0`版本开始,可以通过`${nacoshome}/conf/application.properties`中的`nacos.console.ui.enabled`来开启或关闭Nacos默认控制台,默认为开启。 + +同时在关闭默认控制台时,默认控制台会读取`${nacoshome}/conf/console-guide.conf`文件中的内容,并在默认控制台中生成引导页,让维护者自定义将使用默认控制台的用户引导向自定义的统一平台上进行操作。 ### 会话时间 diff --git a/docs/zh-cn/v2/guide/user/auth.md b/docs/zh-cn/v2/guide/user/auth.md index 0702c2de001..a48582cbd2c 100644 --- a/docs/zh-cn/v2/guide/user/auth.md +++ b/docs/zh-cn/v2/guide/user/auth.md @@ -25,6 +25,12 @@ description: Authorization |~~nacos.core.auth.default.token.secret.key~~|SecretKey012345678901234567890123456789012345678901234567890123456789|1.2.0 ~ 2.0.4|同`nacos.core.auth.plugin.nacos.token.secret.key`| |~~nacos.core.auth.default.token.expire.seconds~~|18000|1.2.0 ~ 2.0.4|同`nacos.core.auth.plugin.nacos.token.expire.seconds`| +## 默认控制台登录页 + +2.2.2版本之前的Nacos默认控制台,无论服务端是否开启鉴权,都会存在一个登录页;这导致很多用户被**误导**认为Nacos默认是存在鉴权的。在社区安全工程师的建议下,Nacos自**2.2.2**版本开始,在未开启鉴权时,默认控制台将不需要登录即可访问,同时在控制台中给予提示,提醒用户当前集群未开启鉴权。 + +在用户开启鉴权后,控制台才需要进行登录访问。 同时针对不同的鉴权插件,提供新的接口方法,用于提示控制台是否开启登录页;同时在`2.2.3`版本后,Nacos可支持关闭开源控制台,并引导到用户自定义的Nacos控制台,详情可查看[Nacos鉴权插件-服务端插件](../../plugin/auth-plugin.md)及[控制台手册-关闭登录功能](/zh-cn/docs/v2/guide/admin/console-guide.html#1.1) + ## 服务端如何开启鉴权 ### 非Docker环境 diff --git a/docs/zh-cn/v2/guide/user/parameters-check.md b/docs/zh-cn/v2/guide/user/parameters-check.md new file mode 100644 index 00000000000..227180b7b4f --- /dev/null +++ b/docs/zh-cn/v2/guide/user/parameters-check.md @@ -0,0 +1,240 @@ +--- +title: 参数校验 +keywords: 参数校验,使用规则 +description: 参数校验 +date: 2023-10-24 +--- + +# 参数校验 + +2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。 + +在2.3.0版本中,Nacos明确了参数#### 校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据#### 校验规则对客户端向服务端发送的请求进行校验。 + +用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。 + +## 参数校验开关 + +### 服务端 + +服务端的参数校验功能**默认开启**,用户可以通过设置`${nacos.home}/conf`目录下的`application.properties`文件中的`nacos.core.param.check.enabled`值选择开启或者关闭服务端参数校验功能。 + +`nacos.core.param.check.enabled=true`时开启Nacos服务端参数校验,`false`关闭服务端参数校验 + +### 客户端 + +待实现 + +## 参数校验规则 + +开启参数校验后OpenAPI文档 和 SDK文档中的所有接口中的相关参数都会接受格式校验,现对相关参数以及#### 校验规则进行说明: + +|参数描述|最大字符长度|#### 校验规则| +|-----|-----|-----| +|命名空间名称|256|禁止`@#$%^&*`,对应正则表达式:`[^@#$%^&*]+$`| +|命名空间ID|64|只允许字母数字下划线以及"-"字符,对应正则表达式:`^[\w-]+`| +|配置名称|256|只允许字母数字以及`_-.:`,对应正则表达式:`^[a-zA-Z0-9-_:\.]*$`| +|服务名称|512|禁止中文和`@@`且禁止以`@`开头,禁止空白字符,对应正则表达式`^(?!@).((?!@@)[^\u4E00-\u9FA5])*$`| +|分组名称|128|只允许字母数字以及`_-.:`,对应正则表达式:`^[a-zA-Z0-9-_:\.]*$`| +|集群名称|64|只允许数字字母和`-_`,对应正则表达式`^[0-9a-zA-Z-_]+$`| +|IP地址|128|禁止中文字符和空白字符,对应正则表达式为`^[^\u4E00-\u9FA5]*$`| +|端口号|-|取值范围为`0~65535`| +|实例元数据|1024|字段名加字段值的总长度小于1024个字符| + +### 1. namespaceShowName + +#### 参数描述 + +命名空间名称 + +#### 校验规则 + +字符长度最大为256,禁止`@#$%^&*`,对应正则表达式:`[^@#$%^&*]+$` + +#### OpenAPI示例 + +- [命名空间](/zh-cn/docs/v2/guide/user/open-api.html#3.1) + +#### 校验失败报错信息 + +-超出长度:`Param 'namespaceShowName' is illegal, the param length should not exceed 256.` +-非法字符:`Param 'namespaceShowName' is illegal, illegal characters should not appear in the param.` + +### 2. namespaceId/tenant/namespace + +#### 参数描述 + +命名空间ID(租户空间) + +#### 校验规则 + +字符长度最大为64,只允许字母数字下划线以及"-"字符,对应正则表达式:`^[\w-]+` + +#### OpenAPI示例 + +- [获取配置](/zh-cn/docs/v2/guide/user/open-api.html#1.1) +- [注册实例](/zh-cn/docs/v2/guide/user/open-api.html#2.1) + +#### 校验失败报错信息 + +-超出长度:`Param 'namespaceId/tenant' is illegal, the param length should not exceed 64.` +-非法字符:`Param 'namespaceId/tenant' is illegal, illegal characters should not appear in the param.` + +### 3. dataId + +#### 参数描述 + +配置名称 + +#### 校验规则 + +字符长度最大为256,只允许字母数字以及`_-.:`,对应正则表达式:`^[a-zA-Z0-9-_:\.]*$` + +#### OpenAPI示例 + +[发布配置](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#1.2) + +#### Java SDK示例 + +监听配置:`public void addListener(String dataId, String group, Listener listener) ` + +#### 校验失败报错信息 + +-超出长度:`Param 'dataId' is illegal, the param length should not exceed 512.` +-非法字符:`Param 'dataId' is illegal, illegal characters should not appear in the param.` + +### 4. service/serviceName + +#### 参数描述 + +服务名称 + +#### 校验规则 + +字符长度最大为512,禁止中文和`@@`且禁止以`@`开头,对应正则表达式`^(?!@).((?!@@)[^\u4E00-\u9FA5])*$` + +#### OpenAPI示例 + +[注册实例](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.1) + +#### Java SDK示例 + +注册实例:`void registerInstance(String serviceName, String ip, int port) throws NacosException; ` + +#### 校验失败报错信息 + +-超出长度:`Param 'serviceName' is illegal, the param length should not exceed 512.` +-非法字符:`Param 'serviceName' is illegal, illegal characters should not appear in the param.` + +### 5. group/groupName + +#### 参数描述 + +分组名称 + +#### 校验规则 + +字符长度最大为128,只允许字母数字以及`_-.:`,对应正则表达式:`^[a-zA-Z0-9-_:\.]*$` + +#### OpenAPI示例 + +[查询实例列表](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.4) + +#### Java SDK示例 + +删除配置:`public boolean removeConfig(String dataId, String group) throws NacosException ` + +#### 校验失败报错信息 + +-超出长度:`Param 'group' is illegal, the param length should not exceed 512.` +-非法字符:`Param 'group' is illegal, illegal characters should not appear in the param.` + +### 6. cluster/clusterName + +#### 参数描述 + +集群名称 + +#### 校验规则 + +字符长度最大为64,只允许数字字母和`-_`,对应正则表达式`^[0-9a-zA-Z-_]+$` + +#### OpenAPI示例 + +[更新实例](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.3) + +#### Java SDK示例 + +获取全部实例:`List getAllInstances(String serviceName, List clusters) throws NacosException;` + +#### 校验失败报错信息 + +-超出长度:`Param 'cluster' is illegal, the param length should not exceed 64.` +-非法字符:`Param 'cluster' is illegal, illegal characters should not appear in the param.` + +### 7. ip + +#### 参数描述 + +IP地址 + +#### 校验规则 + +字符长度最大为128,禁止中文字符,对应正则表达式为`^[^\u4E00-\u9FA5]*$` + +#### OpenAPI示例 + +[查询指定服务的实例列表](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.5) + +#### Java SDK示例 + +注销实例:`void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;` + +#### 校验失败报错信息 + +-超出长度:`Param 'ip' is illegal, the param length should not exceed 128.` +-非法字符:`Param 'ip' is illegal, illegal characters should not appear in the param.` + +### 8. port + +#### 参数描述 + +端口号 + +#### 校验规则 + +取值范围为0~65535 + +#### OpenAPI示例 + +[更新实例](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.3) + +#### Java SDK示例 + +注销实例:`void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;` + +#### 校验失败报错信息 + +端口取值超出范围:`Param 'port' is illegal, the value should be between 0 and 65535` + +### 9. metadata + +#### 参数描述 + +实例元数据 + +#### 校验规则 + +字段名加字段值的总长度小于1024个字符 + +#### OpenAPI示例 + +[更新实例](https://nacos.io/zh-cn/docs/v2/guide/user/open-api.html#2.3) + +#### Java SDK示例 + +注册实例:`void registerInstance(String serviceName, Instance instance) throws NacosException;` + +#### 校验失败报错信息 + +实例总长度超出范围:`Param 'Metadata' is illegal, the param length should not exceed %d.` \ No newline at end of file diff --git a/docs/zh-cn/v2/plugin/auth-plugin.md b/docs/zh-cn/v2/plugin/auth-plugin.md index d6635d2189e..4d817398d0f 100644 --- a/docs/zh-cn/v2/plugin/auth-plugin.md +++ b/docs/zh-cn/v2/plugin/auth-plugin.md @@ -68,6 +68,7 @@ Nacos从2.1.0版本开始,支持通过[SPI](https://docs.oracle.com/javase/tut |enableAuth|ActionTypes,SignType|boolean|在调用`validateIdentity`和`validateAuthority`前调用,插件可自行判断是否对此类型的操作或此类型的模块进行鉴权。| |validateIdentity|IdentityContext, Resource|boolean|对身份信息进行验证,在`validateAuthority`前调用| |validateAuthority|IdentityContext, Permission|boolean|对权限进行验证,在`validateIdentity`返回为`true`时调用| +|isLoginEnabled|void|boolean|是否该插件开启开源控制台登录页,返回`true`时,访问开源控制台将需要通过登录页登录| ### 加载服务端插件 diff --git a/docs/zh-cn/v2/plugin/config-change-plugin.md b/docs/zh-cn/v2/plugin/config-change-plugin.md new file mode 100644 index 00000000000..bb747ae288d --- /dev/null +++ b/docs/zh-cn/v2/plugin/config-change-plugin.md @@ -0,0 +1,136 @@ +--- +title: 配置变更 +keywords: 配置变更,配置审计,配置格式校验,webhook +description: Nacos 支持配置变更插件,可支持配置审计、配置格式校验、及变更后webhook回调 +--- + +# 配置变更插件 + +社区中一直以来都希望Nacos配置中心能在配置发生变更时,通知一些特定系统,用于发送记录、警告等审计功能。在2.3.0版本前,只能通过模拟Nacos客户端订阅配置的方式,对核心配置的变更操作进行订阅,在收到变更通知后,进行发送记录、警告等功能的执行。 + +这种实现方式有几个比较大的问题,第一是监听的配置需要逐个添加,难以对所有配置变更进行获取;第二是只能在配置变更后执行功能逻辑,无法做到前置的操作,如格式校验,白名单校验等。 + +因此Nacos在2.3.0版本后,支持通过[SPI](https://docs.oracle.com/javase/tutorial/sound/SPI-intro.html)注入配置变更插件,允许用户通过自定义插件的方式,对配置变更前,和变更完成后分别执行一些自定义逻辑,如格式校验,白名单校验,webhook等。 + +## 配置变更插件中的概念 + +Nacos的配置变更插件,参考了面向切面编程AOP的设计思想,将配置的变更操作(如新增,更新,删除)作为`切点(PointCut)`,并在这些切点`前(Before)`和`后(After)`织入插件。 + +### 配置变更切点(ConfigChangePointCutTypes) + +Nacos根据配置变更的行为和来源,将配置变更操作在`com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes`中定位为了数个`配置变更切点(ConfigChangePointCutTypes)`,具体内容如下: + +|切点名称|描述|起始版本| +|-----|-----|-----| +|PUBLISH_BY_HTTP|配置通过HTTP接口进行发布,包含了创建配置及修改配置|2.3.0| +|PUBLISH_BY_RPC|配置通过GRPC接口进行发布,包含了创建配置及修改配置|2.3.0| +|REMOVE_BY_HTTP|配置通过HTTP接口进行删除|2.3.0| +|REMOVE_BY_RPC|配置通过GRPC接口进行删除|2.3.0| +|IMPORT_BY_HTTP|配置通过HTTP接口进行导入|2.3.0| +|REMOVE_BATCH_HTTP|配置通过HTTP接口进行批量删除|2.3.0| + +### 配置变更织入类型(ConfigChangeExecuteTypes) + +Nacos的配置变更插件需要在`配置变更切点`之前或之后进行执行,即需要选择`配置变更织入类型(ConfigChangeExecuteTypes)`,定义在`com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes`中,具体内容如下: + +|织入类型|描述|起始版本| +|-----|-----|-----| +|EXECUTE_BEFORE_TYPE|插件实现在`配置变更切点`之**前**执行|2.3.0| +|EXECUTE_AFTER_TYPE|插件实现在`配置变更切点`之**后**执行|2.3.0| + +## 插件开发 + +开发Nacos服务端配置变更插件,首先需要依赖配置变更插件的的相关API + +```xml + + com.alibaba.nacos + nacos-config-plugin + ${project.version} + +``` + +`${project.version}` 为您开发插件所对应的Nacos版本,`2.3.0`及以上。 + +随后实现`com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService`接口,该接口需要实现的方法如下: + +|方法名|入参内容|返回内容|描述| +|-----|-----|-----|-----| +|getServiceType|void|String|插件的名称,用于区分不同类型的插件实现| +|getOrder|void|int|插件的执行顺序,配置变更插件采用链式插件设计,多个插件实现时会按照顺序执行,getOrder越小,顺序越靠前| +|executeType|void|ConfigChangeExecuteTypes|插件实现的`配置变更织入类型`| +|pointcutMethodNames|void|ConfigChangePointCutTypes[]|插件实现织入的`配置变更切点`| +|execute|ConfigChangeRequest,ConfigChangeResponse|void|实际插件执行的逻辑| + +其中`ConfigChangeRequest`和`ConfigChangeResponse`分别为执行逻辑时传入的内容及执行结果, + +`ConfigChangeRequest`的具体内容如下: + +|字段名|字段类型|描述| +|-----|-----|-----| +|requestType|ConfigChangePointCutTypes|本次配置变更的切点类型| +|requestArgs|HashMap|本次配置变更的实际参数,主要包含有`namespace`,`group`,`dataId`,`content`等内容,不同的切点类型参数存在不同| + +`ConfigChangeResponse `的具体内容如下: + +|字段名|字段类型|描述| +|-----|-----|-----| +|responseType|ConfigChangePointCutTypes|本次配置变更的切点类型| +|isSuccess|boolean|执行是否成功,当返回值为`false`时,将会拦截本次配置变更,并直接返回失败的结果| +|retVal|Object|返回内容,预留字段,暂未启用| +|msg|String|执行结果信息,在`isSuccess`为`false`时获取,用于返回给客户端的信息| +|args|Object[]|配置变更操作的执行参数,在`EXECUTE_BEFORE_TYPE`的插件类型时生效,可用于修改实际执行的配置变更时的内容,如将content中的某些内容修改为其他值| + +### 加载插件 + +插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到`${nacos-server.path}/plugins`下 + +放置后,需要修改`${nacos-server.path}/conf/application.properties`中的以下配置 + +```properties +### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应 +nacos.core.config.plugin.${configChangePluginName}.enabled=true +``` + +随后重启nacos集群,启动完成后,可在`${nacos-server.path}/logs/nacos.log`日志中看到如下日志。 + +```text +[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully. +``` + +### 插件自定义参数传递 + +部分插件可能希望通过配置文件设置一些参数,自定义插件可以通过修改`${nacos-server.path}/conf/application.properties`中的以下配置完成: + +```properties +### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应 +nacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue} +``` + +之后能在`ConfigChangeRequest`中,通过下述方法获取该参数: + +```Java +final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES); +final String ${propertyKey} = properties.getProperty("${propertyKey}"); +``` + +## 插件DEMO实现 + +在[nacos-group/nacos-plugin](https://github.com/nacos-group/nacos-plugin)中,有一个demo的配置变更插件的实现,该demo插件实现了对配置内容格式的校验、配置导入名称白名单的校验、以及变更后回调webhook。打包成jar/zip,放置到nacos服务端的classpath中,在`${nacos-server.path}/conf/application.properties`中的加入以下配置: + +```properties +# webhook +#nacos.core.config.plugin.webhook.enabled=true +# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html +#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=*** +# The content push max capacity ,byte +#nacos.core.config.plugin.webhook.contentMaxCapacity=102400 + +# whitelist +#nacos.core.config.plugin.whitelist.enabled=true +# The import file suffixs +#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html + +# fileformatcheck,which validate the import file of type and content +#nacos.core.config.plugin.fileformatcheck.enabled=true +``` \ No newline at end of file diff --git a/docs/zh-cn/v2/plugin/control-plugin.md b/docs/zh-cn/v2/plugin/control-plugin.md new file mode 100644 index 00000000000..ecfe00c3369 --- /dev/null +++ b/docs/zh-cn/v2/plugin/control-plugin.md @@ -0,0 +1,222 @@ +--- +title: 反脆弱 +keywords: 反脆弱,限流,连接数限制,TPS +description: Nacos 支持反脆弱插件,避免高压下的集群容量问题。 +--- + +# 反脆弱插件 + +Nacos 从2.3.0版本开始,支持通过[SPI](https://docs.oracle.com/javase/tutorial/sound/SPI-intro.html)的方式注入反脆弱相关插件,并在`application.properties`配置文件中选择某一种插件实现作为实际反脆弱能力。本文档会详细介绍如何实现一个反脆弱插件和如何使其生效。 + +## 反脆弱插件中的概念 + +反脆弱是对访问服务端的**某种资源**的**频率和次数**达到一定程度时进行的限制访问的策略,用于保护服务端在高压情况下能快速拒绝请求,防止过多的资源访问导致服务端资源耗尽引起的大面积不可用;Nacos反脆弱插件,将信息主要抽象为`监控点`和`反脆弱规则`。 + +### 监控点(ControlPoint) + +监控点对应的请求服务端时所占用的资源的映射,目前主要针对的是`连接(Connection)`以及`每秒请求数(TPS)`。 + +- 连接(Connection)监控点主要监控Nacos 服务端中使用Nacos2.X客户端的长连接数量以及使用Nacos1.X客户端的配置长轮询数量,两者独立监控。 +- 每秒请求数(TPS)监控点主要是监控Nacos 服务端中各核心接口被访问的频率,同类型的操作接口会被视为相同的监控点,如注册服务的v1接口和v2接口,具体的每秒请求数(TPS)监控点可查看本文档下文[监控点名称](#1.1)。 + +### 反脆弱规则(ControlRule) + +反脆弱规则是针对每个监控点而执行的不同的限制规则,具体又分为`连接数规则(ConnectionControlRule)`和`每秒请求数规则(TpsControlRule)` + +`连接数规则(ConnectionControlRule)`主要包含如下内容: + +|字段名|类型|描述| +|-----|-----|-----| +|countLimit|int|连接数总数限制,默认为-1,不限制| +|monitorIpList|Set|trace监控的Ip列表,用于详细观察对应ip的连接做了哪些操作,添加后,对应ip的连接请求会被详细打印在remote-digest.log日志中| + +`每秒请求数规则(TpsControlRule)`主要包含如下内容: + +|字段名|类型|描述| +|-----|-----|-----| +|pointName|String|规则所对应的监控点名称| +|pointRule|RuleDetail|规则内容的具体细节| + +其中`RuleDetail`又包含如下内容: + +|字段名|类型|描述| +|-----|-----|-----| +|ruleName|String|规则的名称,区别于监控点名称,同一个监控点可以有多个规则名| +|maxCount|int|TPS总数限制,默认为-1,不限制| +|period|TimeUnit|规则生效的周期,即统计到秒级/分钟级等,默认`TimeUnit.SECONDS`秒级| +|monitorType|String|监控类型,取值为`monitor`或`intercept`,对应为监控模式(只统计和打印tps,即使触发规则也不拦截)和拦截模式| + +## 插件开发 + +开发Nacos服务端反脆弱插件,首先需要依赖反脆弱插件的相关API + +```xml + + com.alibaba.nacos + nacos-control-plugin + ${project.version} + +``` + +`${project.version}` 为您开发插件所对应的Nacos版本,`2.3.0`及以上。 + +随后继承`com.alibaba.nacos.plugin.control.connection.ConnectionControlManager`抽象类和`com.alibaba.nacos.plugin.control.tps.TpsControlManager`抽象类,实现缺失的方法;然后实现`com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder` 接口,创建上述实现的两个抽象类;最后将您的实现添加到SPI的services当中。 + +`com.alibaba.nacos.plugin.control.connection.ConnectionControlManager`需要实现的方法如下: + +|方法名|入参内容|返回内容|描述| +|-----|-----|-----|-----| +|applyConnectionLimitRule|ConnectionControlRule|void|应用新的连接数规则| +|check|ConnectionCheckRequest|ConnectionCheckResponse|判断是否命中连接数规则,如果ConnectionCheckResponse中的sucess为false,将会拒绝新连接的建立| + +`com.alibaba.nacos.plugin.control.tps.TpsControlManager`需要实现的方法如下: + +|方法名|入参内容|返回内容|描述| +|-----|-----|-----|-----| +|registerTpsPoint|String|void|注册TPS监控点,Nacos服务会在启动时向插件注册当前的TPS监控买点,入参为TPS监控点的名字,具体的监控点名称可查看本文档下文[监控点名称](#1.1);插件需要在方法内,维护一个用于记录TPS和规则内容的`TpsBarrier`,详情查看[自定义TPS时间窗口](#1.2)。| +|applyTpsRule|String,TpsControlRule|void|应用新的TPS规则,根据TPS监控点名称关联及更新。| +|check|TpsCheckRequest|TpsCheckResponse|判断是否命中TPS规则,如果TpsCheckResponse中的sucess为false,将会拒绝新的请求。| + +`com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder` 需要实现的方法如下: + +|方法名|入参内容|返回内容|描述| +|-----|-----|-----|-----| +|getName|void|String|插件的名称,和配置文件中指定的类型进行匹配,使用命中的的插件。| +|buildConnectionControlManager| void |ConnectionControlManager|创建插件对应的`ConnectionControlManager `实现,为null时会使用`no limit`实现。| +|buildTpsControlManager| void |TpsControlManager|创建插件对应的`TpsControlManager`实现,为null时会使用`no limit`实现。| + +### 加载插件 + +插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到`${nacos-server.path}/plugins`下 + +放置后,需要修改`${nacos-server.path}/conf/application.properties`中的以下配置 + +```properties +### 所启用的Nacos的反脆弱插件的名称,与com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 的getName 返回值对应 +nacos.plugin.control.manager.type=${controlPluginName} +``` + +随后重启nacos集群,启动完成后,可以从`${nacos-server.path}/logs/plugin-control.log`中看到如下日志: + +```text +Found control manager plugin of name=${controlPluginName} + +Build connection control manager, class=${your plugin ConnectionControlManager class} + +Build tps control manager, class=${your plugin TpsControlManager class} +``` + +## 使用Nacos自带的反脆弱插件 + +Nacos2.3.0版本起,自带一个简易的反脆弱插件实现,可以做到对Nacos服务端的连接数及指定接口TPS进行限制。 + +### 启用Nacos自带的反脆弱插件 + +需要修改`${nacos-server.path}/conf/application.properties`中的以下配置 + +```properties +nacos.plugin.control.manager.type=nacos +``` + +### 设置反脆弱规则 + +可通过创建和修改反脆弱规则文件的方式,修改和设置反脆弱规则,默认反脆弱插件的规则是通过json格式定义的;例如想要设置连接限制为100,可执行如下操作: + +```shell +mkdir -p ${nacos.home}/data/connection/ +echo '{"countLimit": 100}' > ${nacos.home}/data/connection/limitRule +``` + 随后重启Nacos节点即可。 + + 又例如想要设置配置查询接口的TPS为100,可执行如下操作: + + ```shell + mkdir -p ${nacos.home}/data/tps/ + # ConfigQuery 为配置查询接口的监控点名称(pointName) + echo '{"pointName":"ConfigQuery","pointRule":{"maxCount":100,"monitorType":"intercept"}}' > ${nacos.home}/data/tps/ConfigQuery + ``` + 随后重启Nacos节点即可。 + +其他更多反脆弱规则,以及具体的反脆弱监控点名称,请查看下文[监控点名称](#1.1)。 + +### 反脆弱规则存储位置 + +Nacos自带的简易反脆弱插件实现,Nacos服务端会通过本地文件系统,存储和读取反脆弱规则,默认所在目录的为`${nacos.home}/data/connection`及`${nacos.home}/data/tps`中,如果想将规则文件更换目录存储,可以在`${nacos-server.path}/conf/application.properties`中修改以下配置: + +```properties +nacos.plugin.control.rule.local.basedir=${expectedDir} +``` + +这样规则将会被存储在`${expectedDir}/data/connection`及`${expectedDir}/data/tps`中。 + +

+ +## 当前支持的监控点名称 + +|监控点名称|对应内容|描述|起始版本| +|-----|-----|-----|-----| +|connection|节点总连接数|指定节点最大可支持连接数限制|2.3.0| +|ConfigPublish|配置发布接口TPS|指定节点最大可支持配置发布的TPS限制,同时包含了通过http访问和grpc访问的来源|2.3.0| +|ConfigQuery|配置查询接口TPS|指定节点最大可支持配置查询的TPS限制,同时包含了通过http访问和grpc访问的来源|2.3.0| +|ConfigRemove|配置移除接口TPS|指定节点最大可支持配置移除的TPS限制,同时包含了通过http访问和grpc访问的来源|2.3.0| +|ConfigListen|配置监听接口TPS|指定节点最大可支持配置监听的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|RemoteNamingInstanceRegisterDeregister|服务实例注册及注销接口TPS|服务实例注册或注销的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|RemoteNamingInstanceBatchRegister|服务实例批量注册接口TPS|服务实例批量注册的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|RemoteNamingServiceListQuery|服务列表查询接口TPS|服务列表查询的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|RemoteNamingServiceQuery|服务查询接口TPS|服务查询的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|RemoteNamingServiceSubscribeUnSubscribe|服务订阅和取消订阅接口TPS|服务订阅和取消订阅的TPS限制,仅包含通过grpc访问的来源|2.3.0| +|NamingInstanceRegister|服务实例注册接口TPS|服务实例注册的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingInstanceDeregister|服务实例注销接口TPS|服务实例注销的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingInstanceUpdate|服务实例元数据更新接口TPS|服务实例更新的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingInstanceMetadataUpdate|服务实例元数据批量更新接口TPS|服务实例元数据批量更新的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingServiceSubscribe|服务实例查询及订阅接口TPS|服务订阅及查询的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingInstanceQuery|单个服务实例查询接口TPS|单个服务实例查询的TPS限制,仅包含通过http访问的来源|2.3.0| +|HttpHealthCheck|服务实例心跳续约接口TPS|服务实例心跳续约的TPS限制,仅包含通过http访问的来源|2.3.0| +|NamingServiceRegister|服务创建接口TPS|服务创建的TPS限制,与`NamingInstanceRegister`不同,此监控点表示的是创建空服务接口所对应的TPS,仅包含通过http访问的来源|2.3.0| +|NamingServiceDeregister|服务删除接口TPS|服务删除的TPS限制,与`NamingInstanceDeregister `不同,此监控点表示的是删除服务接口所对应的TPS,仅包含通过http访问的来源|2.3.0| +|NamingServiceQuery|服务查询接口TPS|服务查询的TPS限制,与`NamingInstanceQuery `不同,此监控点表示的是查询服务信息接口所对应的TPS,仅包含通过http访问的来源|2.3.0| +|NamingServiceListQuery|服务列表查询接口TPS|服务列表查询的TPS限制,与`NamingServiceSubscribe `不同,此监控点表示的是服务列表查询接口所对应的TPS,仅包含通过http访问的来源|2.3.0| +|NamingServiceUpdate|服务元数据更新接口TPS|服务元数据更新的TPS限制,与`NamingInstanceUpdate `不同,此监控点表示的是服务元数据更新接口所对应的TPS,仅包含通过http访问的来源|2.3.0| + +## 反脆弱插件进阶开发 + +Nacos反脆弱插件还支持一些进阶式的拓展,以满足对此方面有更高要求的开发者和用户。 + +### 反脆弱规则外部存储 + +Nacos反脆弱插件的默认情况下,仅支持通过本地文件系统来存储和修改单节点的反脆弱规则,对于一些集群规模较大或集群较多的用户,逐个节点进行调整会消耗大量时间和操作;同时本地文件系统在许多容器化环境中,存在磁盘挂载和持久化的问题。因此Nacos反脆弱插件允许增加一个可选的外部存储进行反脆弱规则的统一存储和下发,外部存储可有插件自行实现对接,例如`数据库`,`配置中心`等。 + +要实现反脆弱规则的外部存储,需要在开发插件时,实现`com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder`接口,并随插件jar文件一起放置在`${nacos-server.path}/plugins`下。 + +放置后,需要修改`${nacos-server.path}/conf/application.properties`中的以下配置 + +```properties +nacos.plugin.control.rule.external.storage=${controlPluginName} +``` + +随后重启Nacos节点即可。 + +### 动态加载反脆弱规则 + +在自定义插件实现中,可以通过两种方式进行反脆弱规则的动态加载: + +1. 调用`com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule`方法或`com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule`方法。 +2. 通过`NotifyCenter.publishEvent()`发布`ConnectionLimitRuleChangeEvent` 或`TpsControlRuleChangeEvent`事件。 + +### 反脆弱规则的自定义格式解析 + +Nacos 默认使用`Json`格式作为反脆弱规则的文本格式,插件开发者也可以使用其他的格式,如`Yaml`或其他自定义格式进行解析。 + +重写`com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParser`及`com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser`,实现自定义格式规则解析器`RuleParser`,Nacos将使用自定义的规则解析器`RuleParser`进行规则文本的解析。 + +同时,也可以解析成默认自定义规则的增强规则,配合自定义插件的定制逻辑实现更高程度的反脆弱控制。 + +

+ +### 自定义TPS时间窗口 + +众所周知,在统计TPS时,存在时间窗口算法的区别,不同的时间窗口对TPS的统计结果会有较大的区别。 + +Nacos默认使用简单的同秒统计方式,即按照时钟的秒来进行统计。对于大多数场景来说是足够使用的,但对于一些精确度要求高的用户而言,可能需要使用滑动窗口等更精确的方式进行统计。 + +此时需要插件开发者,继承`com.alibaba.nacos.plugin.control.tps.barrier.TpsBarrier`和`com.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier`,自定义实现TPS的时间窗口和统计方式。并重写`com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator`,在初始化插件和动态加载反脆弱规则时,生成对应的自定义实现。 \ No newline at end of file diff --git a/en-us/blog/index.html b/en-us/blog/index.html index da69e99dce3..957669b7450 100644 --- a/en-us/blog/index.html +++ b/en-us/blog/index.html @@ -15,7 +15,7 @@ -
Blog

All posts

+
Blog

All posts

diff --git a/en-us/docs/v2/guide/admin/console-guide.html b/en-us/docs/v2/guide/admin/console-guide.html index e65bd6faab1..e28b56d23f0 100644 --- a/en-us/docs/v2/guide/admin/console-guide.html +++ b/en-us/docs/v2/guide/admin/console-guide.html @@ -102,17 +102,14 @@

Change the default username/password method

INSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');

Close the login function

-

As part of its own development console, do not want to be nacos security filter interceptor.Therefore nacos support custom close the login functionFind the configuration file ${nacoshome}/conf/application.properties. The properties, replace the following content.

-
## spring security config
-### turn off security
-spring.security.enabled=false
-management.security=false
-security.basic.enabled=false
-nacos.security.ignore.urls=/**
-
-#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/**
-
-
+

Before version 2.2.2, the Nacos default console would always redirect to the login page, whether the authentication function was enabled, which could mislead users into thinking that there was authentication function, but in fact it was disabled, posing a security risk.

+

After consultation and discussion with the community and security engineers, it was decided that when the authentication switch is turned off, the console login function will be automatically disabled when using the Nacos default console.

+

Therefore, starting from version 2.2.2, when the authentication switch nacos.core.auth.enabled is false, the Nacos default console will no longer redirect to the login page, but will add a page prompt indicating that the current cluster has not enabled authentication function.

+

At the same time, a new interface com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled has been added for custom authentication plugins to control the login page. By default, it returns false.

+

Close default console ui

+

Some companies or users may want to disable the default console of Nacos and use their own unified platform for Nacos configuration and service management. They may also want to separate the authentication for console operations and client requests, meaning that authentication is required for console operations but not for client requests.

+

Starting from version 2.3.0, maintainers can use the nacos.console.ui.enabled property in ${nacoshome}/conf/application.properties to enable or disable the Nacos default console. It is true by default.

+

When the default console is disabled, it will read the content of the ${nacoshome}/conf/console-guide.conf file and generate a guide page in the default console. This allows maintainers to customize the guide to redirect users who would have used the default console to their own unified platform for operations.

Session time

The default session to keep time for 30 minutes.After 30 minutes need to login authentication.Temporarily does not support to modify the default time.

Community participation in the front end of the building

diff --git a/en-us/docs/v2/guide/admin/console-guide.json b/en-us/docs/v2/guide/admin/console-guide.json index 8cd937b6023..206a213af0e 100644 --- a/en-us/docs/v2/guide/admin/console-guide.json +++ b/en-us/docs/v2/guide/admin/console-guide.json @@ -1,6 +1,6 @@ { "filename": "console-guide.md", - "__html": "

Console Guide

\n

Nacos console aims to enhance the console for service list, health management, service management, a distributed configuration management control ability, in order to help users reduce the cost of micro management service application architecture, will provide basic functions include the following:

\n
    \n
  • Service management\n
      \n
    • Service list and health status display
    • \n
    • Service metadata storage and editing
    • \n
    • Service flow weight adjustment
    • \n
    • Service elegant line up and down
    • \n
    \n
  • \n
  • Configuration management\n
      \n
    • More configuration format editing
    • \n
    • Edit DIFF
    • \n
    • Sample code
    • \n
    • Push status query
    • \n
    • Configure version and rolled back
    • \n
    \n
  • \n
  • Namespace
  • \n
  • Login management
  • \n
\n

Features

\n

Service management

\n

Developer or operations staff often require after service registry, through friendly interface to view the service registration situation, the current system, including the registration of all of the details of the services and each service.And in a case, with access control service of some of the configuration editor.Nacos in this version of open service found that part of the console, main is to provide users a basic operations page, to view, edit, the current registration services.

\n

Service list management

\n

Service list to help users with a unified view management of all its service and health status.The overall layout is the upper left corner services and search box to search button, the page is the central service list.Service main display service name list, the cluster number, number of instances, health instance number and details button five columns.

\n

\"image.png

\n

In the service list page click details, you can see details of the service.Can look at the service, the basic information of the cluster and examples.

\n

Service flow weighted support and protection

\n

Nacos flow provides the user with the ability of weight control, open the threshold of service flow protection at the same time, in order to help users better protection service cluster service providers are not accidentally break.The diagram below so, click the edit button instance, modify instance weights.If you want to increase the flow of instance, to turn up the weight, if you don't want to flow method receives the instance, the weight can be set to 0.

\n

\"image.png

\n

Service metadata management

\n

Nacos provide multiple dimensions of service metadata exposed, help users to store the information of the custom.This information is based on data storage structure, K - V on the console, as to the k1 = v1, k2 = v2 show such format.Similarly, edit the metadata can be performed by the same format.Such as service metadata editing, first click on the service details in the top right corner of the page "edit service" button, and then in the metadata input: input box version = 1.0, env = prod.

\n

\"image.png

\n

Click on the confirmation, you can in the service details page, see the service metadata has been updated.

\n

\"image.png

\n

Service elegant line up and down

\n

Nacos also offers the service instance line operation, up and down in the service details page, you can click on the instance of "on-line" or "off" button, the offline instance, cases of health will not be included in the list.

\n

\"image.png

\n

Configuration management

\n

Nacos support Group configuration based on the Namespace and Group management, so that users more flexible according to their own needs in accordance with the environment or application, module, such as grouping management services as well as the configuration of Spring, in the configuration management major provides configuration version history, rollback, subscriber query such as the core management abilities.

\n

\"image.png

\n

More configuration format editor

\n

Nacos support YAML, Properties, TEXT, JSON, XML, HTML and other common configuration format online editing, syntax highlighting, format check, help users efficiently edit at the same time greatly reduced the risks of format error.

\n

Nacos support configuration tag ability, help users better and more flexible to the configuration of the classification and management based on the tag.Description of configuration and its change is support users at the same time, people or cross team collaboration management configuration.

\n

\"image.png

\n

Edit DIFF

\n

Nacos supports editing a DIFF ability, help the user to check the changes, and reduce the risks of correction.

\n

\"image.png

\n

Sample code

\n

Nacos provide sample code ability, can let a novice quickly using client-side programming consumption this configuration, novice slash barriers.

\n

\"image.png

\n

\"image.png

\n

Listener query

\n

Nacos provide configuration subscriber is the listener query ability, at the same time provide Client MD5 checksum value of the current configuration, in order to help users better check configuration changes pushed to the Client side.

\n

\"image.png

\n

Configure version and rolled back

\n

Nacos by providing a key roll back configuration version management and its ability, help users can configure to quick recovery, reduce the micro service system in configuration management will meet the availability of the risk.

\n

\"image.png

\n

\"image.png

\n

Namespace management

\n

Nacos based in Namespace helps users logic isolation based multiple namespaces, this can help users better management testing, service and configure the pretest, production environment, so that the same configuration environment (such as database data sources) can define different values.

\n

\"image.png

\n

\"image.png

\n

Login management

\n

Nacos 0.8 version supports simple login function, the default username/password for: nacos/nacos.

\n

\"login\"

\n

Change the default username/password method

\n
    \n
  1. Generate encrypted password in com.alibaba.nacos.console.utils.PasswordEncoderUtil.main function, change nacos to you want to change the password, running with encryption algorithm.Note that salt is random, so the generated password every time may be different, please don't worry about it.
  2. \n
\n
public class PasswordEncoderUtil {\n\n    public static void main(String[] args) {\n        System.out.println(new BCryptPasswordEncoder().encode("nacos"));\n    }\n}\n
\n
    \n
  1. Create a user name or password, use specify a user name password.
  2. \n
\n
INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHzz32dJN7jexM34MOeYirDdFAZm2kuWj7VEOJhhZkDrxfvUu', TRUE);\nINSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');\n
\n

Close the login function

\n

As part of its own development console, do not want to be nacos security filter interceptor.Therefore nacos support custom close the login functionFind the configuration file ${nacoshome}/conf/application.properties. The properties, replace the following content.

\n
## spring security config\n### turn off security\nspring.security.enabled=false\nmanagement.security=false\nsecurity.basic.enabled=false\nnacos.security.ignore.urls=/**\n\n#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/**\n\n
\n

Session time

\n

The default session to keep time for 30 minutes.After 30 minutes need to login authentication.Temporarily does not support to modify the default time.

\n

Community participation in the front end of the building

\n

In Nacos front style, the layout of the discussion, the community vote, finally choose the style of the classic black and white and blue skin, and through our UED Yao Cheng design, layout, make interaction is very natural.

\n

In the development of the console, we recruited through community many front students to participate in the development of the front-end code, in this especially thank Chen Li, Qing Wang, Yanmin Wang Nacos front-end development process in the strong support!

\n

Adhere to the community development, welcome to join and contribute to the community

\n
\n

DISS is cheap, show me your hand!

\n
\n

To join Nacos WeChat community discussion Nacos the evolution of the product, you can sweep through xuechaos WeChat QRcode, let "xuechaos" help you pull in "Nacos community communication group".

\n

\"Screen

\n

More Nacos related open source project information:

\n\n", + "__html": "

Console Guide

\n

Nacos console aims to enhance the console for service list, health management, service management, a distributed configuration management control ability, in order to help users reduce the cost of micro management service application architecture, will provide basic functions include the following:

\n
    \n
  • Service management\n
      \n
    • Service list and health status display
    • \n
    • Service metadata storage and editing
    • \n
    • Service flow weight adjustment
    • \n
    • Service elegant line up and down
    • \n
    \n
  • \n
  • Configuration management\n
      \n
    • More configuration format editing
    • \n
    • Edit DIFF
    • \n
    • Sample code
    • \n
    • Push status query
    • \n
    • Configure version and rolled back
    • \n
    \n
  • \n
  • Namespace
  • \n
  • Login management
  • \n
\n

Features

\n

Service management

\n

Developer or operations staff often require after service registry, through friendly interface to view the service registration situation, the current system, including the registration of all of the details of the services and each service.And in a case, with access control service of some of the configuration editor.Nacos in this version of open service found that part of the console, main is to provide users a basic operations page, to view, edit, the current registration services.

\n

Service list management

\n

Service list to help users with a unified view management of all its service and health status.The overall layout is the upper left corner services and search box to search button, the page is the central service list.Service main display service name list, the cluster number, number of instances, health instance number and details button five columns.

\n

\"image.png

\n

In the service list page click details, you can see details of the service.Can look at the service, the basic information of the cluster and examples.

\n

Service flow weighted support and protection

\n

Nacos flow provides the user with the ability of weight control, open the threshold of service flow protection at the same time, in order to help users better protection service cluster service providers are not accidentally break.The diagram below so, click the edit button instance, modify instance weights.If you want to increase the flow of instance, to turn up the weight, if you don't want to flow method receives the instance, the weight can be set to 0.

\n

\"image.png

\n

Service metadata management

\n

Nacos provide multiple dimensions of service metadata exposed, help users to store the information of the custom.This information is based on data storage structure, K - V on the console, as to the k1 = v1, k2 = v2 show such format.Similarly, edit the metadata can be performed by the same format.Such as service metadata editing, first click on the service details in the top right corner of the page "edit service" button, and then in the metadata input: input box version = 1.0, env = prod.

\n

\"image.png

\n

Click on the confirmation, you can in the service details page, see the service metadata has been updated.

\n

\"image.png

\n

Service elegant line up and down

\n

Nacos also offers the service instance line operation, up and down in the service details page, you can click on the instance of "on-line" or "off" button, the offline instance, cases of health will not be included in the list.

\n

\"image.png

\n

Configuration management

\n

Nacos support Group configuration based on the Namespace and Group management, so that users more flexible according to their own needs in accordance with the environment or application, module, such as grouping management services as well as the configuration of Spring, in the configuration management major provides configuration version history, rollback, subscriber query such as the core management abilities.

\n

\"image.png

\n

More configuration format editor

\n

Nacos support YAML, Properties, TEXT, JSON, XML, HTML and other common configuration format online editing, syntax highlighting, format check, help users efficiently edit at the same time greatly reduced the risks of format error.

\n

Nacos support configuration tag ability, help users better and more flexible to the configuration of the classification and management based on the tag.Description of configuration and its change is support users at the same time, people or cross team collaboration management configuration.

\n

\"image.png

\n

Edit DIFF

\n

Nacos supports editing a DIFF ability, help the user to check the changes, and reduce the risks of correction.

\n

\"image.png

\n

Sample code

\n

Nacos provide sample code ability, can let a novice quickly using client-side programming consumption this configuration, novice slash barriers.

\n

\"image.png

\n

\"image.png

\n

Listener query

\n

Nacos provide configuration subscriber is the listener query ability, at the same time provide Client MD5 checksum value of the current configuration, in order to help users better check configuration changes pushed to the Client side.

\n

\"image.png

\n

Configure version and rolled back

\n

Nacos by providing a key roll back configuration version management and its ability, help users can configure to quick recovery, reduce the micro service system in configuration management will meet the availability of the risk.

\n

\"image.png

\n

\"image.png

\n

Namespace management

\n

Nacos based in Namespace helps users logic isolation based multiple namespaces, this can help users better management testing, service and configure the pretest, production environment, so that the same configuration environment (such as database data sources) can define different values.

\n

\"image.png

\n

\"image.png

\n

Login management

\n

Nacos 0.8 version supports simple login function, the default username/password for: nacos/nacos.

\n

\"login\"

\n

Change the default username/password method

\n
    \n
  1. Generate encrypted password in com.alibaba.nacos.console.utils.PasswordEncoderUtil.main function, change nacos to you want to change the password, running with encryption algorithm.Note that salt is random, so the generated password every time may be different, please don't worry about it.
  2. \n
\n
public class PasswordEncoderUtil {\n\n    public static void main(String[] args) {\n        System.out.println(new BCryptPasswordEncoder().encode("nacos"));\n    }\n}\n
\n
    \n
  1. Create a user name or password, use specify a user name password.
  2. \n
\n
INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHzz32dJN7jexM34MOeYirDdFAZm2kuWj7VEOJhhZkDrxfvUu', TRUE);\nINSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');\n
\n

Close the login function

\n

Before version 2.2.2, the Nacos default console would always redirect to the login page, whether the authentication function was enabled, which could mislead users into thinking that there was authentication function, but in fact it was disabled, posing a security risk.

\n

After consultation and discussion with the community and security engineers, it was decided that when the authentication switch is turned off, the console login function will be automatically disabled when using the Nacos default console.

\n

Therefore, starting from version 2.2.2, when the authentication switch nacos.core.auth.enabled is false, the Nacos default console will no longer redirect to the login page, but will add a page prompt indicating that the current cluster has not enabled authentication function.

\n

At the same time, a new interface com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled has been added for custom authentication plugins to control the login page. By default, it returns false.

\n

Close default console ui

\n

Some companies or users may want to disable the default console of Nacos and use their own unified platform for Nacos configuration and service management. They may also want to separate the authentication for console operations and client requests, meaning that authentication is required for console operations but not for client requests.

\n

Starting from version 2.3.0, maintainers can use the nacos.console.ui.enabled property in ${nacoshome}/conf/application.properties to enable or disable the Nacos default console. It is true by default.

\n

When the default console is disabled, it will read the content of the ${nacoshome}/conf/console-guide.conf file and generate a guide page in the default console. This allows maintainers to customize the guide to redirect users who would have used the default console to their own unified platform for operations.

\n

Session time

\n

The default session to keep time for 30 minutes.After 30 minutes need to login authentication.Temporarily does not support to modify the default time.

\n

Community participation in the front end of the building

\n

In Nacos front style, the layout of the discussion, the community vote, finally choose the style of the classic black and white and blue skin, and through our UED Yao Cheng design, layout, make interaction is very natural.

\n

In the development of the console, we recruited through community many front students to participate in the development of the front-end code, in this especially thank Chen Li, Qing Wang, Yanmin Wang Nacos front-end development process in the strong support!

\n

Adhere to the community development, welcome to join and contribute to the community

\n
\n

DISS is cheap, show me your hand!

\n
\n

To join Nacos WeChat community discussion Nacos the evolution of the product, you can sweep through xuechaos WeChat QRcode, let "xuechaos" help you pull in "Nacos community communication group".

\n

\"Screen

\n

More Nacos related open source project information:

\n\n", "link": "/en-us/docs/v2/guide/admin/console-guide.html", "meta": { "title": "Console Guide", diff --git a/en-us/docs/v2/plugin/config-change-plugin.html b/en-us/docs/v2/plugin/config-change-plugin.html new file mode 100644 index 00000000000..1fc8ad914c8 --- /dev/null +++ b/en-us/docs/v2/plugin/config-change-plugin.html @@ -0,0 +1,270 @@ + + + + + + + + + + + + Config Change Plugin + + + + + +
Documentation
+

Translated by AI.

+
+

Config Change Plugin

+

Community has long been hoping for Nacos Configuration Center to provide notifications to specific systems when configurations change. These notifications are used for recording, warning, and auditing purposes. Before version 2.3.0, the only way to achieve this was by simulating Nacos client subscription to the configurations. This approach involved subscribing to changes in core configurations and executing functionalities such as sending records and warnings upon receiving change notifications.

+

However, this implementation had a few significant issues. Firstly, individual configurations needed to be added one by one, making it difficult to capture all configuration changes. Secondly, functionalities could only be executed after configuration changes, and there was no capability for performing pre-change operations such as format validation or whitelist validation.

+

To address these limitations, starting from Nacos version 2.3.0, Nacos introduced support for injecting configuration change plugins through the SPI. This allows users to define custom plugins to execute specific logic before and after configuration changes. Examples of such logic include format validation, whitelist validation, and webhook integrations.

+

These enhancements provide more flexibility and control for users to implement their own custom logic before and after configuration changes in Nacos Configuration Center.

+

Concepts in Config Change Plugin

+

Nacos's configuration change plugin design takes inspiration from the aspect-oriented programming (AOP) paradigm. It treats configuration change operations, such as adding, updating, and deleting, as the pointcuts and weaves the plugins before and after these pointcuts.

+

ConfigChangePointCutTypes

+

Nacos has categorized the configuration change operations based on their behaviors and sources. These configuration change operations are defined as several ConfigChangePointCutTypes in com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes. The specific details are as follows:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PointCut NameDescriptionStart version
PUBLISH_BY_HTTPConfiguration is published through the HTTP interface, including creating and modifying configurations.2.3.0
PUBLISH_BY_RPCConfiguration is published through the gRPC interface, including creating and modifying configurations.2.3.0
REMOVE_BY_HTTPConfiguration is removed through the HTTP interface.2.3.0
REMOVE_BY_RPCConfiguration is removed through the gRPC interface.2.3.0
IMPORT_BY_HTTPConfiguration is imported through the HTTP interface.2.3.0
REMOVE_BATCH_HTTPConfigurations are batch removed through the HTTP interface.2.3.0
+

ConfigChangeExecuteTypes

+

In Nacos, the configuration change plugins need to be executed before or after the ConfigChangePointCutTypes by selecting the ConfigChangeExecuteTypes. These execution types are defined in com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes. The specific details are as follows:

+ + + + + + + + + + + + + + + + + + + + +
Execute TypeDescriptionStart version
EXECUTE_BEFORE_TYPEPlugin execute Before ConfigChangePointCutTypes2.3.0
EXECUTE_AFTER_TYPEPlugin execute After ConfigChangePointCutTypes2.3.0
+

Plugin Development

+

To develop a config change plugin for the Nacos server, you first need to depend on the relevant API of the config change plugin.

+
        <dependency>
+            <groupId>com.alibaba.nacos</groupId>
+            <artifactId>nacos-config-plugin</artifactId>
+            <version>${project.version}</version>
+        </dependency>
+
+

${project.version} is the Nacos version, at least 2.3.0.

+

Then implement com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService interface, which include following method:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Method nameParametersReturnDescription
getServiceTypevoidStringReturns the name of the plugin, which is used to differentiate between different types of plugin implementations.
getOrdervoidintReturns the execution order of the plugin. The configuration change plugin is designed with a chain plugin pattern, where multiple plugins are executed in order. The smaller the return value of getOrder, the earlier the plugin is executed.
executeTypevoidConfigChangeExecuteTypesReturns ConfigChangeExecuteTypes implemented by the plugin.
pointcutMethodNamesvoidConfigChangePointCutTypes[]Returns ConfigChangePointCutTypes where the plugin is implemented.
executeConfigChangeRequest,ConfigChangeResponsevoidExecutes the actual logic of the plugin.
+

ConfigChangeRequest and ConfigChangeResponse refer to the contents passed in during the execution of logic and the resulting execution outcome, respectively.

+

ConfigChangeRequest mainly include follow contents:

+ + + + + + + + + + + + + + + + + + + + +
Field nameTypeDescription
requestTypeConfigChangePointCutTypesThe pointcut types of this configuration change
requestArgsHashMap<String, Object>The actual parameters of this configuration change, mainly including namespace, group, dataId, content, etc., with different parameters for different pointcut types
+

ConfigChangeResponse mainly include follow contents:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Field nameTypeDescription
responseTypeConfigChangePointCutTypesThe pointcut types of this configuration change
isSuccessbooleanIndicates whether the execution is successful. When the return value is false, the configuration change will be intercepted and the failure result will be returned directly
retValObjectReturn content, a reserved field that is currently not used
msgStringExecution result information, obtained when isSuccess is false, used to return information to the client
argsObject[]The execution parameters of the configuration change, effective for the EXECUTE_BEFORE_TYPE plugin type. It can be used to modify the content of the actual executed configuration change, such as changing certain values in content to other values
+

Load Plugin

+

After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under ${nacos-server.path}/plugins directly.

+

After Adding plugins into classpath, also need to modify some configuration in ${nacos-server.path}/conf/application.properties.

+
### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method.
+nacos.core.config.plugin.${configChangePluginName}.enabled=true
+
+

After restarting the Nacos cluster and completing the startup, you can see the following logs in ${nacos-server.path}/logs/plugin-control.log:

+
[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully.
+
+

Custom Plugin properties

+

Some plugins may want to set certain parameters through a configuration file. Custom plugins can modify the following configurations in ${nacos-server.path}/conf/application.properties to achieve this:

+
### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method.
+nacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue}
+
+

In ConfigChangeRequest, getting properties by following way:

+
final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES);
+final String ${propertyKey} = properties.getProperty("${propertyKey}");
+
+

Plugin Demo

+

In the nacos-group/nacos-plugin repository, there is a demo implementation of a configuration change plugin. This demo plugin implements validation of the configuration content format, validation of the whitelist for importing configuration names, and a webhook callback after the change. To use this plugin, you need to package it as a JAR/ZIP file and place it in the classpath of the Nacos server. Additionally, add the following configuration in ${nacos-server.path}/conf/application.properties:

+
# webhook
+#nacos.core.config.plugin.webhook.enabled=true
+# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html
+#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=***
+# The content push max capacity ,byte
+#nacos.core.config.plugin.webhook.contentMaxCapacity=102400
+
+# whitelist
+#nacos.core.config.plugin.whitelist.enabled=true
+# The import file suffixs
+#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html
+
+# fileformatcheck,which validate the import file of type and content
+#nacos.core.config.plugin.fileformatcheck.enabled=true
+
+
+ + + + + + + + + + + + + diff --git a/en-us/docs/v2/plugin/config-change-plugin.json b/en-us/docs/v2/plugin/config-change-plugin.json new file mode 100644 index 00000000000..d856b6c1f31 --- /dev/null +++ b/en-us/docs/v2/plugin/config-change-plugin.json @@ -0,0 +1,10 @@ +{ + "filename": "config-change-plugin.md", + "__html": "
\n

Translated by AI.

\n
\n

Config Change Plugin

\n

Community has long been hoping for Nacos Configuration Center to provide notifications to specific systems when configurations change. These notifications are used for recording, warning, and auditing purposes. Before version 2.3.0, the only way to achieve this was by simulating Nacos client subscription to the configurations. This approach involved subscribing to changes in core configurations and executing functionalities such as sending records and warnings upon receiving change notifications.

\n

However, this implementation had a few significant issues. Firstly, individual configurations needed to be added one by one, making it difficult to capture all configuration changes. Secondly, functionalities could only be executed after configuration changes, and there was no capability for performing pre-change operations such as format validation or whitelist validation.

\n

To address these limitations, starting from Nacos version 2.3.0, Nacos introduced support for injecting configuration change plugins through the SPI. This allows users to define custom plugins to execute specific logic before and after configuration changes. Examples of such logic include format validation, whitelist validation, and webhook integrations.

\n

These enhancements provide more flexibility and control for users to implement their own custom logic before and after configuration changes in Nacos Configuration Center.

\n

Concepts in Config Change Plugin

\n

Nacos's configuration change plugin design takes inspiration from the aspect-oriented programming (AOP) paradigm. It treats configuration change operations, such as adding, updating, and deleting, as the pointcuts and weaves the plugins before and after these pointcuts.

\n

ConfigChangePointCutTypes

\n

Nacos has categorized the configuration change operations based on their behaviors and sources. These configuration change operations are defined as several ConfigChangePointCutTypes in com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes. The specific details are as follows:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
PointCut NameDescriptionStart version
PUBLISH_BY_HTTPConfiguration is published through the HTTP interface, including creating and modifying configurations.2.3.0
PUBLISH_BY_RPCConfiguration is published through the gRPC interface, including creating and modifying configurations.2.3.0
REMOVE_BY_HTTPConfiguration is removed through the HTTP interface.2.3.0
REMOVE_BY_RPCConfiguration is removed through the gRPC interface.2.3.0
IMPORT_BY_HTTPConfiguration is imported through the HTTP interface.2.3.0
REMOVE_BATCH_HTTPConfigurations are batch removed through the HTTP interface.2.3.0
\n

ConfigChangeExecuteTypes

\n

In Nacos, the configuration change plugins need to be executed before or after the ConfigChangePointCutTypes by selecting the ConfigChangeExecuteTypes. These execution types are defined in com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes. The specific details are as follows:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Execute TypeDescriptionStart version
EXECUTE_BEFORE_TYPEPlugin execute Before ConfigChangePointCutTypes2.3.0
EXECUTE_AFTER_TYPEPlugin execute After ConfigChangePointCutTypes2.3.0
\n

Plugin Development

\n

To develop a config change plugin for the Nacos server, you first need to depend on the relevant API of the config change plugin.

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-config-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} is the Nacos version, at least 2.3.0.

\n

Then implement com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService interface, which include following method:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Method nameParametersReturnDescription
getServiceTypevoidStringReturns the name of the plugin, which is used to differentiate between different types of plugin implementations.
getOrdervoidintReturns the execution order of the plugin. The configuration change plugin is designed with a chain plugin pattern, where multiple plugins are executed in order. The smaller the return value of getOrder, the earlier the plugin is executed.
executeTypevoidConfigChangeExecuteTypesReturns ConfigChangeExecuteTypes implemented by the plugin.
pointcutMethodNamesvoidConfigChangePointCutTypes[]Returns ConfigChangePointCutTypes where the plugin is implemented.
executeConfigChangeRequest,ConfigChangeResponsevoidExecutes the actual logic of the plugin.
\n

ConfigChangeRequest and ConfigChangeResponse refer to the contents passed in during the execution of logic and the resulting execution outcome, respectively.

\n

ConfigChangeRequest mainly include follow contents:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Field nameTypeDescription
requestTypeConfigChangePointCutTypesThe pointcut types of this configuration change
requestArgsHashMap<String, Object>The actual parameters of this configuration change, mainly including namespace, group, dataId, content, etc., with different parameters for different pointcut types
\n

ConfigChangeResponse mainly include follow contents:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Field nameTypeDescription
responseTypeConfigChangePointCutTypesThe pointcut types of this configuration change
isSuccessbooleanIndicates whether the execution is successful. When the return value is false, the configuration change will be intercepted and the failure result will be returned directly
retValObjectReturn content, a reserved field that is currently not used
msgStringExecution result information, obtained when isSuccess is false, used to return information to the client
argsObject[]The execution parameters of the configuration change, effective for the EXECUTE_BEFORE_TYPE plugin type. It can be used to modify the content of the actual executed configuration change, such as changing certain values in content to other values
\n

Load Plugin

\n

After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under ${nacos-server.path}/plugins directly.

\n

After Adding plugins into classpath, also need to modify some configuration in ${nacos-server.path}/conf/application.properties.

\n
### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method.\nnacos.core.config.plugin.${configChangePluginName}.enabled=true\n
\n

After restarting the Nacos cluster and completing the startup, you can see the following logs in ${nacos-server.path}/logs/plugin-control.log:

\n
[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully.\n
\n

Custom Plugin properties

\n

Some plugins may want to set certain parameters through a configuration file. Custom plugins can modify the following configurations in ${nacos-server.path}/conf/application.properties to achieve this:

\n
### The name of the config change plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService's getServiceType method.\nnacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue}\n
\n

In ConfigChangeRequest, getting properties by following way:

\n
final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES);\nfinal String ${propertyKey} = properties.getProperty(\"${propertyKey}\");\n
\n

Plugin Demo

\n

In the nacos-group/nacos-plugin repository, there is a demo implementation of a configuration change plugin. This demo plugin implements validation of the configuration content format, validation of the whitelist for importing configuration names, and a webhook callback after the change. To use this plugin, you need to package it as a JAR/ZIP file and place it in the classpath of the Nacos server. Additionally, add the following configuration in ${nacos-server.path}/conf/application.properties:

\n
# webhook\n#nacos.core.config.plugin.webhook.enabled=true\n# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html\n#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=***\n# The content push max capacity ,byte\n#nacos.core.config.plugin.webhook.contentMaxCapacity=102400\n\n# whitelist\n#nacos.core.config.plugin.whitelist.enabled=true\n# The import file suffixs\n#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html\n\n# fileformatcheck,which validate the import file of type and content\n#nacos.core.config.plugin.fileformatcheck.enabled=true\n
\n", + "link": "/en-us/docs/v2/plugin/config-change-plugin.html", + "meta": { + "title": "Config Change Plugin", + "keywords": "Config Change,Config audit,Config format check,webhook", + "description": "Nacos support Config Change Plugin." + } +} \ No newline at end of file diff --git a/en-us/docs/v2/plugin/control-plugin.html b/en-us/docs/v2/plugin/control-plugin.html new file mode 100644 index 00000000000..2ae6fa9281f --- /dev/null +++ b/en-us/docs/v2/plugin/control-plugin.html @@ -0,0 +1,450 @@ + + + + + + + + + + + + control plugin + + + + + +
Documentation
+

Translated by AI.

+
+

Control Plugin

+

Starting from version 2.3.0, Nacos supports injecting control related plugins through SPI, and selecting a specific plugin implementation as the actual control capability in the application.properties configuration file. This document provides a detailed introduction on how to implement an control plugin and how to make it effective.

+

Concepts in Control Plugin

+

Anti-fragility is a strategy that restricts access to a certain resource on the server when the frequency and number of accesses reach a certain level. It is used to protect the server from quickly rejecting requests under high pressure, preventing widespread unavailability caused by excessive resource access and exhaustion. The Nacos control plugin abstracts information primarily into control point and control rules.

+

Control Point

+

The control point correspond to the mapping of resources occupied when making requests to the server. Currently, they mainly focus on Connections and Transactions Per Second (TPS).

+
    +
  • The "Connections" control point primarily monitors the number of long connections used by Nacos 2.X clients and the number of long polling connections used by Nacos 1.X clients. These two types of connections are monitored independently.
  • +
  • The "Transactions Per Second (TPS)" control point mainly monitors the frequency of access to core interfaces in the Nacos server. Similar operation interfaces are considered as the same monitor point. For example, the registration service v1 interface and v2 interface are treated as the same monitor point. Please refer to the document for specific TPS monitor point names.
  • +
+

Control Rule

+

The control rules are different limitation rules that are executed for each control point. They are specifically categorized as "Connection Control Rules" ConnectionControlRule and "Transactions Per Second Control Rules" TpsControlRule.

+

ConnectionControlRule mainly include follow contents:

+ + + + + + + + + + + + + + + + + + + + +
Field nameTypeDescription
countLimitintTotal count limit for connections. Default is -1, indicating no limitation.
monitorIpListSetList of IPs to be monitored by trace. It is used to observe the operations performed on the corresponding IP connections in detail. Once added, the connection requests from the corresponding IPs will be logged in detail in the remote-digest.log file.
+

TpsControlRule mainly include follow contents:

+ + + + + + + + + + + + + + + + + + + + +
Field nameTypeDescription
pointNameStringName of the control point corresponding to the rule.
pointRuleRuleDetailSpecific details of the rule content.
+

And RuleDetail mainly include follow contents:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Field nameTypeDescription
ruleNameStringName of the rule. It is different from the control point name. A control point can have multiple rule names.
maxCountintTotal count limit for TPS. Default is -1, indicating no limitation.
periodTimeUnitThe period in which the rule is effective, such as counting at the second level or minute level. Default is TimeUnit.SECONDS for second level.
monitorTypeStringMonitoring type, can be either monitor or intercept. It corresponds to monitoring mode (only counting and printing TPS, even if the rule is triggered, no interception) and interception mode.
+

Plugin Development

+

To develop a control plugin for the Nacos server, you first need to depend on the relevant API of the control plugin.

+
        <dependency>
+            <groupId>com.alibaba.nacos</groupId>
+            <artifactId>nacos-control-plugin</artifactId>
+            <version>${project.version}</version>
+        </dependency>
+
+

${project.version} is the Nacos version, at least 2.3.0.

+

Then to extend com.alibaba.nacos.plugin.control.connection.ConnectionControlManager abstract class and com.alibaba.nacos.plugin.control.tps.TpsControlManagerabstract class,implement their methods; Then implement com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder interface to build the two above classes; Finally add the SPI file to META-INF/services.

+

com.alibaba.nacos.plugin.control.connection.ConnectionControlManager need implement follow methods:

+ + + + + + + + + + + + + + + + + + + + + + + +
Method nameParametersReturnDescription
applyConnectionLimitRuleConnectionControlRulevoidApply new connection limit rule
checkConnectionCheckRequestConnectionCheckResponseTo determine if the connection count rule is hit, if the success field in the ConnectionCheckResponse is false, new connections will be rejected.
+

com.alibaba.nacos.plugin.control.tps.TpsControlManager need implement follow methods:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Method nameParametersReturnDescription
registerTpsPointStringvoidRegisters a TPS control point. The Nacos server will register the current TPS control point to the plugin during startup. The input parameter is the name of the TPS control point. Please refer to the document for point names. The plugin needs to maintain a TpsBarrier within this method to record TPS and rule content. For more details, refer to Custom Time Windows for TPS.
applyTpsRuleString, TpsControlRulevoidApplies a new TPS rule and associates it with the TPS control point name for update.
checkTpsCheckRequestTpsCheckResponseChecks if the TPS rule is hit. If the success field in the TpsCheckResponse is false, new requests will be rejected.
+

com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder need implement follow methods:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Method nameParametersReturnDescription
getNamevoidStringReturns the name of the plugin. It is used to match the specified type in the configuration file and use the matched plugin.
buildConnectionControlManagervoidConnectionControlManagerCreates an implementation of ConnectionControlManager for the plugin. When it is null, the no limit implementation will be used.
buildTpsControlManagervoidTpsControlManagerCreates an implementation of TpsControlManager for the plugin. When it is null, the no limit implementation will be used.
+

Load Plugin

+

After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under ${nacos-server.path}/plugins directly.

+

After Adding plugins into classpath, also need to modify some configuration in ${nacos-server.path}/conf/application.properties.

+
### The name of the control plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder's getName method.
+nacos.plugin.control.manager.type=${controlPluginName}
+
+

After restarting the Nacos cluster and completing the startup, you can see the following logs in ${nacos-server.path}/logs/plugin-control.log:

+
Found control manager plugin of name=${controlPluginName}
+
+Build connection control manager, class=${your plugin ConnectionControlManager class}
+
+Build tps control manager, class=${your plugin TpsControlManager class}
+
+

Use the default Nacos Control Plugin

+

Starting from version 2.3.0, Nacos comes with a built-in simple control plugin implementation, which can limit the connection count and specified interface TPS of the Nacos server.

+

Enable the default Nacos Control Plugin

+

Modify the following configurations in ${nacos-server.path}/conf/application.properties:

+
nacos.plugin.control.manager.type=nacos
+
+

Setting control plugin for default Nacos control plugin

+

You can modify and set control rules by creating and modifying the control rule file. By default, the rules for the control plugin are defined in JSON format.

+

For example, if you want to set the connection limit to 100, you can perform the following steps:

+
mkdir -p ${nacos.home}/data/connection/
+echo '{"countLimit": 100}' > ${nacos.home}/data/connection/limitRule
+
+

Then restart Nacos server node.

+

What's more, if you want set the TPS of query config as 100, you can perform the following steps:

+
mkdir -p ${nacos.home}/data/tps/
+# ConfigQuery is the PointName of the query config API.
+echo '{"pointName":"ConfigQuery","pointRule":{"maxCount":100,"monitorType":"intercept"}}' > ${nacos.home}/data/tps/ConfigQuery 
+
+

Then restart Nacos server node.

+

More control rules and control point names please move to point names.

+

The Storage Dir of Control Rules

+

The built-in simple control plugin implementation in Nacos stores and reads control rules through the local file system. By default, the rules are stored in ${nacos.home}/data/connection and ${nacos.home}/data/tps directories.

+

If you want to change the directory for storing the rule files, you can modify the following configuration in ${nacos-server.path}/conf/application.properties:

+
nacos.plugin.control.rule.local.basedir=${expectedDir}
+
+

In this way, the control rules will be stored in ${expectedDir}/data/connection and ${expectedDir}/data/tps.

+

+

Supported Control PointNames

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
control pointNamesmeaningdescriptionstarted version
connectionTotal ConnectionsMaximum Supported Connection Limit for a Specific Node2.3.0
ConfigPublishConfiguration Publish Interface TPSMaximum Supported TPS Limit for Configuration Publishing on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigQueryConfiguration Query Interface TPSMaximum Supported TPS Limit for Configuration Querying on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigRemoveConfiguration Removal Interface TPSMaximum Supported TPS Limit for Configuration Removal on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigListenConfiguration Listening Interface TPSMaximum Supported TPS Limit for Configuration Listening on a Specific Node, only including gRPC access source2.3.0
RemoteNamingInstanceRegisterDeregisterRemote Naming Instance Register and Deregister Interface TPSTPS Limit for Registering or Deregistering Service Instances, only including gRPC access source2.3.0
RemoteNamingInstanceBatchRegisterRemote Naming Instance Batch Register Interface TPSTPS Limit for Batch Registering Service Instances, only including gRPC access source2.3.0
RemoteNamingServiceListQueryRemote Naming Service List Query Interface TPSTPS Limit for Service List Query, only including gRPC access source2.3.0
RemoteNamingServiceQueryRemote Naming Service Query Interface TPSTPS Limit for Service Query, only including gRPC access source2.3.0
RemoteNamingServiceSubscribeUnSubscribeRemote Naming Service Subscribe and Unsubscribe Interface TPSTPS Limit for Service Subscribe and Unsubscribe, only including gRPC access source2.3.0
NamingInstanceRegisterNaming Instance Register Interface TPSTPS Limit for Registering Service Instances, only including HTTP access source2.3.0
NamingInstanceDeregisterNaming Instance Deregister Interface TPSTPS Limit for Deregistering Service Instances, only including HTTP access source2.3.0
NamingInstanceUpdateNaming Instance Metadata Update Interface TPSTPS Limit for Updating Service Instance Metadata, only including HTTP access source2.3.0
NamingInstanceMetadataUpdateNaming Instance Batch Metadata Update Interface TPSTPS Limit for Batch Updating Service Instance Metadata, only including HTTP access source2.3.0
NamingServiceSubscribeNaming Service Query and Subscribe Interface TPSTPS Limit for Service Query and Subscribe, only including HTTP access source2.3.0
NamingInstanceQuerySingle Service Instance Query Interface TPSTPS Limit for Querying Single Service Instance, only including HTTP access source2.3.0
HttpHealthCheckService Instance Heartbeat Renewal Interface TPSTPS Limit for Service Instance Heartbeat Renewal, only including HTTP access source2.3.0
NamingServiceRegisterService Create Interface TPSTPS Limit for Creating Services, different from NamingInstanceRegister, this monitoring point represents the TPS for the interface of creating an empty service, only including HTTP access source2.3.0
NamingServiceDeregisterService Delete Interface TPSTPS Limit for Deleting Services, different from NamingInstanceDeregister, this monitoring point represents the TPS for the interface of deleting services, only including HTTP access source2.3.0
NamingServiceQueryService Query Interface TPSTPS Limit for Service Query, different from NamingInstanceQuery, this monitoring point represents the TPS for the interface of querying service information, only including HTTP access source2.3.0
NamingServiceListQueryService List Query Interface TPSTPS Limit for Service List Query, different from NamingServiceSubscribe, this monitoring point represents the TPS for the interface of service list query, only including HTTP access source2.3.0
NamingServiceUpdateService Metadata Update Interface TPSTPS Limit for Service Metadata Update, different from NamingInstanceUpdate, this monitoring point represents the TPS for the interface of updating service metadata, only including HTTP access source2.3.0
+

Advanced Development

+

Nacos control plugin also supports advanced extensions to meet the higher requirements of developers and users in this aspect.

+

External Storage for Control Rules

+

By default, the Nacos control plugin only supports storing and modifying control rules for individual nodes through the local file system. For users with large-scale or multiple clusters, adjusting each node individually can be time-consuming and cumbersome. Additionally, in many containerized environments, there may be issues with disk mounting and persistence for the local file system.

+

To address these concerns, the Nacos control plugin allows the option to add an external storage for unified storage and distribution of control rules. This external storage can be implemented by the plugin itself, such as using a database or a configuration center.

+

To enable external storage for control rules, you need to implement the com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder interface in your plugin development, and place the plugin jar file along with the interface implementation in the ${nacos-server.path}/plugins directory.

+

After placing the plugin files, you need to modify the following configuration in ${nacos-server.path}/conf/application.properties:

+
nacos.plugin.control.rule.external.storage=${controlPluginName}
+
+

Afterwards, restart the Nacos node for the changes to take effect.

+

Dynamic Loading of Control Rules

+

In custom plugin implementations, there are two ways to dynamically load contrl rules:

+
    +
  1. +

    Call the com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule method or com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule method.

    +
  2. +
  3. +

    Publish a ConnectionLimitRuleChangeEvent or TpsControlRuleChangeEvent event using NotifyCenter.publishEvent().

    +
  4. +
+

These methods allow you to reload and update the control rules dynamically in your custom plugin implementation.

+

Custom Format Parse for Control Rules

+

By default, Nacos uses the Json format as the text format for control rules. However, plugin developers can use other formats such as Yaml or other custom formats for parsing.

+

To use a custom format for rule parsing, you can override the com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParser and com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser methods. Implement a custom format rule parser by creating a RuleParser that can parse rules in your desired format. Nacos will then use this custom rule parser to parse the rule text.

+

Additionally, you can enhance the default custom rules by parsing them into more advanced rules. This can be done in combination with the custom logic of your plugin, allowing for more advanced control.

+

+

Custom Time Windows for TPS

+

It is well-known that different time window algorithms can lead to significant differences in TPS statistics. Nacos defaults to a simple per-second statistical method, where TPS is counted based on clock seconds. This is sufficient for most scenarios, but for users with higher accuracy requirements, more precise methods such as sliding windows may be needed for TPS statistics.

+

In such cases, plugin developers can customize the time window and statistical methods for TPS by inheriting from com.alibaba.nacos.plugin.control.tps.barrier.TpsBarrier and com.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier. Additionally, the com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator method needs to be overridden. This allows for the generation of the corresponding custom implementation during plugin initialization and dynamic loading of control rules.

+

By implementing these customizations, plugin developers can control the time window and statistical methods used for TPS in a way that meets the specific accuracy requirements of their users.

+
+ + + + + + + + + + + + + diff --git a/en-us/docs/v2/plugin/control-plugin.json b/en-us/docs/v2/plugin/control-plugin.json new file mode 100644 index 00000000000..574f0fdfa1c --- /dev/null +++ b/en-us/docs/v2/plugin/control-plugin.json @@ -0,0 +1,10 @@ +{ + "filename": "control-plugin.md", + "__html": "
\n

Translated by AI.

\n
\n

Control Plugin

\n

Starting from version 2.3.0, Nacos supports injecting control related plugins through SPI, and selecting a specific plugin implementation as the actual control capability in the application.properties configuration file. This document provides a detailed introduction on how to implement an control plugin and how to make it effective.

\n

Concepts in Control Plugin

\n

Anti-fragility is a strategy that restricts access to a certain resource on the server when the frequency and number of accesses reach a certain level. It is used to protect the server from quickly rejecting requests under high pressure, preventing widespread unavailability caused by excessive resource access and exhaustion. The Nacos control plugin abstracts information primarily into control point and control rules.

\n

Control Point

\n

The control point correspond to the mapping of resources occupied when making requests to the server. Currently, they mainly focus on Connections and Transactions Per Second (TPS).

\n
    \n
  • The "Connections" control point primarily monitors the number of long connections used by Nacos 2.X clients and the number of long polling connections used by Nacos 1.X clients. These two types of connections are monitored independently.
  • \n
  • The "Transactions Per Second (TPS)" control point mainly monitors the frequency of access to core interfaces in the Nacos server. Similar operation interfaces are considered as the same monitor point. For example, the registration service v1 interface and v2 interface are treated as the same monitor point. Please refer to the document for specific TPS monitor point names.
  • \n
\n

Control Rule

\n

The control rules are different limitation rules that are executed for each control point. They are specifically categorized as "Connection Control Rules" ConnectionControlRule and "Transactions Per Second Control Rules" TpsControlRule.

\n

ConnectionControlRule mainly include follow contents:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Field nameTypeDescription
countLimitintTotal count limit for connections. Default is -1, indicating no limitation.
monitorIpListSetList of IPs to be monitored by trace. It is used to observe the operations performed on the corresponding IP connections in detail. Once added, the connection requests from the corresponding IPs will be logged in detail in the remote-digest.log file.
\n

TpsControlRule mainly include follow contents:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Field nameTypeDescription
pointNameStringName of the control point corresponding to the rule.
pointRuleRuleDetailSpecific details of the rule content.
\n

And RuleDetail mainly include follow contents:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Field nameTypeDescription
ruleNameStringName of the rule. It is different from the control point name. A control point can have multiple rule names.
maxCountintTotal count limit for TPS. Default is -1, indicating no limitation.
periodTimeUnitThe period in which the rule is effective, such as counting at the second level or minute level. Default is TimeUnit.SECONDS for second level.
monitorTypeStringMonitoring type, can be either monitor or intercept. It corresponds to monitoring mode (only counting and printing TPS, even if the rule is triggered, no interception) and interception mode.
\n

Plugin Development

\n

To develop a control plugin for the Nacos server, you first need to depend on the relevant API of the control plugin.

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-control-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} is the Nacos version, at least 2.3.0.

\n

Then to extend com.alibaba.nacos.plugin.control.connection.ConnectionControlManager abstract class and com.alibaba.nacos.plugin.control.tps.TpsControlManagerabstract class,implement their methods; Then implement com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder interface to build the two above classes; Finally add the SPI file to META-INF/services.

\n

com.alibaba.nacos.plugin.control.connection.ConnectionControlManager need implement follow methods:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Method nameParametersReturnDescription
applyConnectionLimitRuleConnectionControlRulevoidApply new connection limit rule
checkConnectionCheckRequestConnectionCheckResponseTo determine if the connection count rule is hit, if the success field in the ConnectionCheckResponse is false, new connections will be rejected.
\n

com.alibaba.nacos.plugin.control.tps.TpsControlManager need implement follow methods:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Method nameParametersReturnDescription
registerTpsPointStringvoidRegisters a TPS control point. The Nacos server will register the current TPS control point to the plugin during startup. The input parameter is the name of the TPS control point. Please refer to the document for point names. The plugin needs to maintain a TpsBarrier within this method to record TPS and rule content. For more details, refer to Custom Time Windows for TPS.
applyTpsRuleString, TpsControlRulevoidApplies a new TPS rule and associates it with the TPS control point name for update.
checkTpsCheckRequestTpsCheckResponseChecks if the TPS rule is hit. If the success field in the TpsCheckResponse is false, new requests will be rejected.
\n

com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder need implement follow methods:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
Method nameParametersReturnDescription
getNamevoidStringReturns the name of the plugin. It is used to match the specified type in the configuration file and use the matched plugin.
buildConnectionControlManagervoidConnectionControlManagerCreates an implementation of ConnectionControlManager for the plugin. When it is null, the no limit implementation will be used.
buildTpsControlManagervoidTpsControlManagerCreates an implementation of TpsControlManager for the plugin. When it is null, the no limit implementation will be used.
\n

Load Plugin

\n

After the plugin finished, it needs to be packaged into jar/zip and places in the classpath of the nacos server. If you don't know how to add plugins into the classpath, please place plugins under ${nacos-server.path}/plugins directly.

\n

After Adding plugins into classpath, also need to modify some configuration in ${nacos-server.path}/conf/application.properties.

\n
### The name of the control plugin enabled in Nacos, corresponding to the return value of com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder's getName method.\nnacos.plugin.control.manager.type=${controlPluginName}\n
\n

After restarting the Nacos cluster and completing the startup, you can see the following logs in ${nacos-server.path}/logs/plugin-control.log:

\n
Found control manager plugin of name=${controlPluginName}\n\nBuild connection control manager, class=${your plugin ConnectionControlManager class}\n\nBuild tps control manager, class=${your plugin TpsControlManager class}\n
\n

Use the default Nacos Control Plugin

\n

Starting from version 2.3.0, Nacos comes with a built-in simple control plugin implementation, which can limit the connection count and specified interface TPS of the Nacos server.

\n

Enable the default Nacos Control Plugin

\n

Modify the following configurations in ${nacos-server.path}/conf/application.properties:

\n
nacos.plugin.control.manager.type=nacos\n
\n

Setting control plugin for default Nacos control plugin

\n

You can modify and set control rules by creating and modifying the control rule file. By default, the rules for the control plugin are defined in JSON format.

\n

For example, if you want to set the connection limit to 100, you can perform the following steps:

\n
mkdir -p ${nacos.home}/data/connection/\necho '{\"countLimit\": 100}' > ${nacos.home}/data/connection/limitRule\n
\n

Then restart Nacos server node.

\n

What's more, if you want set the TPS of query config as 100, you can perform the following steps:

\n
mkdir -p ${nacos.home}/data/tps/\n# ConfigQuery is the PointName of the query config API.\necho '{\"pointName\":\"ConfigQuery\",\"pointRule\":{\"maxCount\":100,\"monitorType\":\"intercept\"}}' > ${nacos.home}/data/tps/ConfigQuery \n
\n

Then restart Nacos server node.

\n

More control rules and control point names please move to point names.

\n

The Storage Dir of Control Rules

\n

The built-in simple control plugin implementation in Nacos stores and reads control rules through the local file system. By default, the rules are stored in ${nacos.home}/data/connection and ${nacos.home}/data/tps directories.

\n

If you want to change the directory for storing the rule files, you can modify the following configuration in ${nacos-server.path}/conf/application.properties:

\n
nacos.plugin.control.rule.local.basedir=${expectedDir}\n
\n

In this way, the control rules will be stored in ${expectedDir}/data/connection and ${expectedDir}/data/tps.

\n

\n

Supported Control PointNames

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
control pointNamesmeaningdescriptionstarted version
connectionTotal ConnectionsMaximum Supported Connection Limit for a Specific Node2.3.0
ConfigPublishConfiguration Publish Interface TPSMaximum Supported TPS Limit for Configuration Publishing on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigQueryConfiguration Query Interface TPSMaximum Supported TPS Limit for Configuration Querying on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigRemoveConfiguration Removal Interface TPSMaximum Supported TPS Limit for Configuration Removal on a Specific Node, including both HTTP and gRPC access sources2.3.0
ConfigListenConfiguration Listening Interface TPSMaximum Supported TPS Limit for Configuration Listening on a Specific Node, only including gRPC access source2.3.0
RemoteNamingInstanceRegisterDeregisterRemote Naming Instance Register and Deregister Interface TPSTPS Limit for Registering or Deregistering Service Instances, only including gRPC access source2.3.0
RemoteNamingInstanceBatchRegisterRemote Naming Instance Batch Register Interface TPSTPS Limit for Batch Registering Service Instances, only including gRPC access source2.3.0
RemoteNamingServiceListQueryRemote Naming Service List Query Interface TPSTPS Limit for Service List Query, only including gRPC access source2.3.0
RemoteNamingServiceQueryRemote Naming Service Query Interface TPSTPS Limit for Service Query, only including gRPC access source2.3.0
RemoteNamingServiceSubscribeUnSubscribeRemote Naming Service Subscribe and Unsubscribe Interface TPSTPS Limit for Service Subscribe and Unsubscribe, only including gRPC access source2.3.0
NamingInstanceRegisterNaming Instance Register Interface TPSTPS Limit for Registering Service Instances, only including HTTP access source2.3.0
NamingInstanceDeregisterNaming Instance Deregister Interface TPSTPS Limit for Deregistering Service Instances, only including HTTP access source2.3.0
NamingInstanceUpdateNaming Instance Metadata Update Interface TPSTPS Limit for Updating Service Instance Metadata, only including HTTP access source2.3.0
NamingInstanceMetadataUpdateNaming Instance Batch Metadata Update Interface TPSTPS Limit for Batch Updating Service Instance Metadata, only including HTTP access source2.3.0
NamingServiceSubscribeNaming Service Query and Subscribe Interface TPSTPS Limit for Service Query and Subscribe, only including HTTP access source2.3.0
NamingInstanceQuerySingle Service Instance Query Interface TPSTPS Limit for Querying Single Service Instance, only including HTTP access source2.3.0
HttpHealthCheckService Instance Heartbeat Renewal Interface TPSTPS Limit for Service Instance Heartbeat Renewal, only including HTTP access source2.3.0
NamingServiceRegisterService Create Interface TPSTPS Limit for Creating Services, different from NamingInstanceRegister, this monitoring point represents the TPS for the interface of creating an empty service, only including HTTP access source2.3.0
NamingServiceDeregisterService Delete Interface TPSTPS Limit for Deleting Services, different from NamingInstanceDeregister, this monitoring point represents the TPS for the interface of deleting services, only including HTTP access source2.3.0
NamingServiceQueryService Query Interface TPSTPS Limit for Service Query, different from NamingInstanceQuery, this monitoring point represents the TPS for the interface of querying service information, only including HTTP access source2.3.0
NamingServiceListQueryService List Query Interface TPSTPS Limit for Service List Query, different from NamingServiceSubscribe, this monitoring point represents the TPS for the interface of service list query, only including HTTP access source2.3.0
NamingServiceUpdateService Metadata Update Interface TPSTPS Limit for Service Metadata Update, different from NamingInstanceUpdate, this monitoring point represents the TPS for the interface of updating service metadata, only including HTTP access source2.3.0
\n

Advanced Development

\n

Nacos control plugin also supports advanced extensions to meet the higher requirements of developers and users in this aspect.

\n

External Storage for Control Rules

\n

By default, the Nacos control plugin only supports storing and modifying control rules for individual nodes through the local file system. For users with large-scale or multiple clusters, adjusting each node individually can be time-consuming and cumbersome. Additionally, in many containerized environments, there may be issues with disk mounting and persistence for the local file system.

\n

To address these concerns, the Nacos control plugin allows the option to add an external storage for unified storage and distribution of control rules. This external storage can be implemented by the plugin itself, such as using a database or a configuration center.

\n

To enable external storage for control rules, you need to implement the com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder interface in your plugin development, and place the plugin jar file along with the interface implementation in the ${nacos-server.path}/plugins directory.

\n

After placing the plugin files, you need to modify the following configuration in ${nacos-server.path}/conf/application.properties:

\n
nacos.plugin.control.rule.external.storage=${controlPluginName}\n
\n

Afterwards, restart the Nacos node for the changes to take effect.

\n

Dynamic Loading of Control Rules

\n

In custom plugin implementations, there are two ways to dynamically load contrl rules:

\n
    \n
  1. \n

    Call the com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule method or com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule method.

    \n
  2. \n
  3. \n

    Publish a ConnectionLimitRuleChangeEvent or TpsControlRuleChangeEvent event using NotifyCenter.publishEvent().

    \n
  4. \n
\n

These methods allow you to reload and update the control rules dynamically in your custom plugin implementation.

\n

Custom Format Parse for Control Rules

\n

By default, Nacos uses the Json format as the text format for control rules. However, plugin developers can use other formats such as Yaml or other custom formats for parsing.

\n

To use a custom format for rule parsing, you can override the com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParser and com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser methods. Implement a custom format rule parser by creating a RuleParser that can parse rules in your desired format. Nacos will then use this custom rule parser to parse the rule text.

\n

Additionally, you can enhance the default custom rules by parsing them into more advanced rules. This can be done in combination with the custom logic of your plugin, allowing for more advanced control.

\n

\n

Custom Time Windows for TPS

\n

It is well-known that different time window algorithms can lead to significant differences in TPS statistics. Nacos defaults to a simple per-second statistical method, where TPS is counted based on clock seconds. This is sufficient for most scenarios, but for users with higher accuracy requirements, more precise methods such as sliding windows may be needed for TPS statistics.

\n

In such cases, plugin developers can customize the time window and statistical methods for TPS by inheriting from com.alibaba.nacos.plugin.control.tps.barrier.TpsBarrier and com.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier. Additionally, the com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator method needs to be overridden. This allows for the generation of the corresponding custom implementation during plugin initialization and dynamic loading of control rules.

\n

By implementing these customizations, plugin developers can control the time window and statistical methods used for TPS in a way that meets the specific accuracy requirements of their users.

\n", + "link": "/en-us/docs/v2/plugin/control-plugin.html", + "meta": { + "title": "control plugin", + "keywords": "anti-fragile,control limit,connection limit,TPS", + "description": "Nacos support control plugin." + } +} \ No newline at end of file diff --git a/en-us/index.html b/en-us/index.html index f79762591d8..2557a33b0d9 100644 --- a/en-us/index.html +++ b/en-us/index.html @@ -15,7 +15,7 @@ -

an easy-to-use dynamic service discovery, configuration and service management platform for building cloud native applications

Released on May 25, 2023

Features

Dynamic Configuration Service

Dynamic Configuration Service allows you to manage configurations in all environments in a centralized, externalized, and dynamic approach. Dynamic configuration saves you from redeploying your applications and services when configuration is updated. You can implement stateless services and achieve on-demand scaling effortlessly.

Service Discovery and Management

Dynamic Service Discovery is key to service-centric (for example microservice or cloud-native) architectures. Nacos supports both DNS-based and RPC-based (Dubbo, gRPC) service discovery, and provides real-time service health checks to prevent routing requests from being sent to unhealthy hosts or service instances. With Nacos, you can also implement circuit breakers for your services with ease.

Dynamic DNS Service

By supporting weighted routing, Dynamic DNS Service helps you implement mid-tier load balancing, more flexible routing, traffic control and DNS resolution services in the production environment within your data center. Dynamic DNS Service also makes it easier for you to implement DNS-based service discovery, which minimizes the risk of coupling to vendor-specific service discovery APIs.

microservice ecosystem landscape

Why Nacos

  • Easy to use

    • One-stop solution for dynamic service discovery, configuration management and dynamic DNS service
    • 20+ out-of-the-box features for service-centric architectures
    • Light-weight production-ready console
  • More adaptive to cloud architectures

    • Seamlessly support kubernetes and spring cloud
    • Easier to deploy and run on popular public cloud (for example AliCloud and AWS)
    • Support multi-tenants and multi-environments
  • Production grade

    • Originated from time-tested internal products from Alibaba Group
    • Supports large-scale scenarios with millions of services
    • Open-source product with enterprise-level SLA
  • Rich internet application scenarios affinity

    • Supports rate throttling, big promotion plans, and multi-region active-active architectures
    • Supports a variety of relevant internet-based use cases directly or with slight extension
    • Traffic scheduling & service governance

Who is using Nacos

Providing your info on Wanted: who's using nacos to help improving nacos better.

+

an easy-to-use dynamic service discovery, configuration and service management platform for building cloud native applications

Released on Oct 19, 2023

Features

Dynamic Configuration Service

Dynamic Configuration Service allows you to manage configurations in all environments in a centralized, externalized, and dynamic approach. Dynamic configuration saves you from redeploying your applications and services when configuration is updated. You can implement stateless services and achieve on-demand scaling effortlessly.

Service Discovery and Management

Dynamic Service Discovery is key to service-centric (for example microservice or cloud-native) architectures. Nacos supports both DNS-based and RPC-based (Dubbo, gRPC) service discovery, and provides real-time service health checks to prevent routing requests from being sent to unhealthy hosts or service instances. With Nacos, you can also implement circuit breakers for your services with ease.

Dynamic DNS Service

By supporting weighted routing, Dynamic DNS Service helps you implement mid-tier load balancing, more flexible routing, traffic control and DNS resolution services in the production environment within your data center. Dynamic DNS Service also makes it easier for you to implement DNS-based service discovery, which minimizes the risk of coupling to vendor-specific service discovery APIs.

microservice ecosystem landscape

Why Nacos

  • Easy to use

    • One-stop solution for dynamic service discovery, configuration management and dynamic DNS service
    • 20+ out-of-the-box features for service-centric architectures
    • Light-weight production-ready console
  • More adaptive to cloud architectures

    • Seamlessly support kubernetes and spring cloud
    • Easier to deploy and run on popular public cloud (for example AliCloud and AWS)
    • Support multi-tenants and multi-environments
  • Production grade

    • Originated from time-tested internal products from Alibaba Group
    • Supports large-scale scenarios with millions of services
    • Open-source product with enterprise-level SLA
  • Rich internet application scenarios affinity

    • Supports rate throttling, big promotion plans, and multi-region active-active architectures
    • Supports a variety of relevant internet-based use cases directly or with slight extension
    • Traffic scheduling & service governance

Who is using Nacos

Providing your info on Wanted: who's using nacos to help improving nacos better.

diff --git a/md_json/blog.json b/md_json/blog.json index a6d466cfe5f..de5d14222ab 100644 --- a/md_json/blog.json +++ b/md_json/blog.json @@ -420,6 +420,16 @@ "description": "Nacos 2.2.2发布,优化启动体验和鉴权提示" } }, + { + "filename": "2.3.0-beta-release.md", + "link": "/zh-cn/blog/2.3.0-beta-release.html", + "meta": { + "title": "Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用", + "keywords": "2.3.0", + "description": "Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用", + "date": "2023-10-24" + } + }, { "filename": "212-and-220beta-release.md", "link": "/zh-cn/blog/212-and-220beta-release.html", diff --git a/md_json/docs.json b/md_json/docs.json index b6d15cb3672..c30a8a18985 100644 --- a/md_json/docs.json +++ b/md_json/docs.json @@ -622,6 +622,15 @@ "description": "This article describes how to develop and use Nacos' authentication plugin." } }, + { + "filename": "config-change-plugin.md", + "link": "/en-us/docs/v2/plugin/config-change-plugin.html", + "meta": { + "title": "Config Change Plugin", + "keywords": "Config Change,Config audit,Config format check,webhook", + "description": "Nacos support Config Change Plugin." + } + }, { "filename": "config-encryption-plugin.md", "link": "/en-us/docs/v2/plugin/config-encryption-plugin.html", @@ -631,6 +640,15 @@ "description": "CONFIGURE ENCRYPTION" } }, + { + "filename": "control-plugin.md", + "link": "/en-us/docs/v2/plugin/control-plugin.html", + "meta": { + "title": "control plugin", + "keywords": "anti-fragile,control limit,connection limit,TPS", + "description": "Nacos support control plugin." + } + }, { "filename": "custom-environment-plugin.md", "link": "/en-us/docs/v2/plugin/custom-environment-plugin.html", @@ -1318,6 +1336,16 @@ "description": "其他语言的SDK" } }, + { + "filename": "parameters-check.md", + "link": "/zh-cn/docs/v2/guide/user/parameters-check.html", + "meta": { + "title": "参数校验", + "keywords": "参数校验,使用规则", + "description": "参数校验", + "date": "2023-10-24" + } + }, { "filename": "sdk.md", "link": "/zh-cn/docs/v2/guide/user/sdk.html", @@ -1345,6 +1373,15 @@ "description": "本文描述如何开发及使用Nacos的鉴权插件" } }, + { + "filename": "config-change-plugin.md", + "link": "/zh-cn/docs/v2/plugin/config-change-plugin.html", + "meta": { + "title": "配置变更", + "keywords": "配置变更,配置审计,配置格式校验,webhook", + "description": "Nacos 支持配置变更插件,可支持配置审计、配置格式校验、及变更后webhook回调" + } + }, { "filename": "config-encryption-plugin.md", "link": "/zh-cn/docs/v2/plugin/config-encryption-plugin.html", @@ -1354,6 +1391,15 @@ "description": "配置加密" } }, + { + "filename": "control-plugin.md", + "link": "/zh-cn/docs/v2/plugin/control-plugin.html", + "meta": { + "title": "反脆弱", + "keywords": "反脆弱,限流,连接数限制,TPS", + "description": "Nacos 支持反脆弱插件,避免高压下的集群容量问题。" + } + }, { "filename": "custom-environment-plugin.md", "link": "/zh-cn/docs/v2/plugin/custom-environment-plugin.html", diff --git a/site_config/blog.js b/site_config/blog.js index 896aa4bf304..ce439cd4495 100644 --- a/site_config/blog.js +++ b/site_config/blog.js @@ -3,6 +3,13 @@ export default { barText: 'Blog', postsTitle: 'All posts', list: [ + { + title: 'Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用', + author: '@杨翊', + dateStr: 'Oct 24, 2023', + desc: 'Nacos 2.3.0-BETA发布,欢迎试用。', + link: '/zh-cn/blog/2.3.0-beta-release.html', + }, { title: 'Nacos开源之夏2023,贡献社区赢取12000奖金', author: '@Nacos开源之夏', @@ -349,6 +356,13 @@ export default { barText: '博客', postsTitle: '所有文章', list: [ + { + title: 'Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用', + author: '@杨翊', + dateStr: 'Oct 24, 2023', + desc: 'Nacos 2.3.0-BETA发布,欢迎试用。', + link: '/zh-cn/blog/2.3.0-beta-release.html', + }, { title: 'Nacos开源之夏2023,贡献社区赢取12000奖金', author: '@Nacos开源之夏', diff --git a/site_config/docs2.js b/site_config/docs2.js index dd18f4e77d8..976aa1f3192 100644 --- a/site_config/docs2.js +++ b/site_config/docs2.js @@ -76,6 +76,14 @@ export default { title: 'Custom Environment', link: '/en-us/docs/v2/plugin/custom-environment-plugin.html', }, + { + title: 'Control', + link: '/en-us/docs/v2/plugin/control-plugin.html', + }, + { + title: 'Configuration Change', + link: '/en-us/docs/v2/plugin/config-change-plugin.html', + }, ] }, { @@ -93,10 +101,10 @@ export default { title: 'Open-API', link: '/en-us/docs/v2/guide/user/open-api.html', }, - // { - // title: 'Nacos Spring', - // link: '/zh-cn/docs/nacos-spring.html', - // }, + { + title: 'Parameter Check', + link: '/zh-cn/docs/v2/guide/user/parameters-check.html', + }, { title: 'Authentication', link: '/en-us/docs/v2/guide/user/auth.html', @@ -293,6 +301,14 @@ export default { title: '环境变量', link: '/zh-cn/docs/v2/plugin/custom-environment-plugin.html', }, + { + title: '反脆弱', + link: '/zh-cn/docs/v2/plugin/control-plugin.html', + }, + { + title: '配置变更', + link: '/zh-cn/docs/v2/plugin/config-change-plugin.html', + }, ] }, { @@ -310,10 +326,10 @@ export default { title: 'Open-API指南', link: '/zh-cn/docs/v2/guide/user/open-api.html', }, - // { - // title: 'Nacos Spring', - // link: '/zh-cn/docs/nacos-spring.html', - // }, + { + title: '参数校验', + link: '/zh-cn/docs/v2/guide/user/parameters-check.html', + }, { title: '权限认证', link: '/zh-cn/docs/v2/guide/user/auth.html', diff --git a/site_config/home.jsx b/site_config/home.jsx index 7f598ca6280..cdf0c5403af 100644 --- a/site_config/home.jsx +++ b/site_config/home.jsx @@ -17,14 +17,14 @@ export default { }, ], versionNote: { - text: 'Release Note of V2.2.3', - link: 'https://github.com/alibaba/nacos/releases/tag/2.2.3', + text: 'Release Note of V2.3.0-BETA', + link: 'https://github.com/alibaba/nacos/releases/tag/2.3.0-BETA', }, versionNote2: { text: 'V1.4.6', link: 'https://github.com/alibaba/nacos/releases/tag/1.4.6', }, - releaseDate: 'Released on May 25, 2023', + releaseDate: 'Released on Oct 19, 2023', }, functions: { @@ -136,14 +136,14 @@ export default { }, ], versionNote: { - text: 'V2.2.3 版本说明', - link: 'https://github.com/alibaba/nacos/releases/tag/2.2.3', + text: 'V2.3.0-BETA 版本说明', + link: 'https://github.com/alibaba/nacos/releases/tag/2.3.0-BETA', }, versionNote2: { text: 'V1.4.6', link: 'https://github.com/alibaba/nacos/releases/tag/1.4.6', }, - releaseDate: '2023年5月25日发布', + releaseDate: '2023年10月19日发布', }, functions: { title: '功能', diff --git a/zh-cn/blog/2.3.0-beta-release.html b/zh-cn/blog/2.3.0-beta-release.html new file mode 100644 index 00000000000..ee79dfa8fbf --- /dev/null +++ b/zh-cn/blog/2.3.0-beta-release.html @@ -0,0 +1,170 @@ + + + + + + + + + + + + Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用 + + + + + +

Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用

+

新版本发布

+

经过4个多月社区的大量贡献,Nacos 2.3.0版本终于进入的Beta发布阶段,在经历了2.2.X的多个安全风险修复的版本后,又一次迎来了功能性上的更新版本。

+

Nacos 反脆弱插件

+

2.2.0版本加入ALPHA版本的反脆弱插件后,社区经过多次重构和修改,提升抽象度和稳定性,终于在2.3.0版本中正式公布了Nacos的反脆弱插件,用于避免Nacos集群在大量请求和连接时导致的容量问题而引发更大规模的故障。

+

默认反脆弱实现插件的使用以及自定义反脆弱插件的开发可参考文档反脆弱插件.

+

Nacos 配置变更插件

+

Nacos配置管理作为Nacos的核心功能,一直受到开发人员、运维人员和审计人员的关注,不少运维人员和审计人员向社区提出诉求,希望Nacos能够提供配置变更的审计、变更通知等功能;同时开发人员也希望Nacos提供一定的格式校验功能,避免修改配置时出现错误导致应用故障。

+

由于Nacos配置管理本身对配置内容是不感知的,且各个用户公司的审计、通知系统也各不相同,因此Nacos为了能够更广阔的适配各个用户的诉求,在去年的开源之夏中制定了配置变更插件的课题,希望以插件化的形式,来满足对配置变更过程的介入。

+

经过同学和导师长期的努力,该插件在2.3.0版本发布,用户可以开发自定义插件,在配置变更前和变更完成后,分别对接公司的审计系统和通知系统,并在变更前进行更多的校验操作,如格式校验,配置名字的规范化等。同时Nacos社区也提供了默认插件的实现,可接入webhook进行配置变更的通知以及导入配置的文件名校验等;详情可参考文档配置变更插件

+

Nacos 请求参数校验

+

2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。

+

在2.3.0版本中,Nacos明确了参数校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据校验规则对客户端向服务端发送的请求进行校验。用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。

+

详细情况可参考文档参数校验规则

+

Nacos 能力协商

+

随着Nacos功能越来越多,版本越来越多,客户端和服务端之间的兼容性愈发重要,如果继续保持尝试性的兼容,只会兼容能力愈发的困难。因此Nacos社区在去年开源之夏中,发布了课题,通过增加客户端和服务端之间的能力协商机制,在客户端连接到服务端时,让连接双方知道对方分别支持的功能,在支持对应功能的情况下,开启对应功能。避免通过尝试而增加通信开销。

+

在2.3.0版本中,该能力也被集成进来,为之后Nacos3.0支持更多优化功能提供基础。

+

其他重要改动

+

除了上述改动,Nacos2.3.0版本还支持了:客户端异常的metrics、grpc长连接的SSL功能、关闭开源控制台等功能,更多细节可以查看变更列表,欢迎试用:

+
## feature
+[#5698] Support nacos control plugin.
+[#8458] Support ability negotiations between server and clients.
+[#8460] Support config change hook plugin.
+[#10117] Support metrics for nacos client request server exception.
+[#10150] Support SSL for grpc connection.
+[#10223] Support auto build instance id when client request instance id is null.
+[#10288] Support get more module state and switches in console.
+[#10734] Support validate most of request parameters.
+[#10774] Support toml format for configuration in console ui.
+[#10831] Support batch deregister instances for service.
+[#10971] Support disable console ui and support add guide information.
+
+## Enhancement&Refactor
+[#6819] Add page size selector in service details page.
+[#8107][#9109][#10169][#10176] Enhance hint when console ui session expired for default auth plugin.
+[#9085] Add the Reachability Metadata required by native-image.
+[#9821] Enhance datasource plugin to make more datasource implementation easier.
+[#9881] Enhance configuration page to supports folding when editing configuration.
+[#10067] Enhance Windows compatibility for configuration snapshot.
+[#10155] Enhance hints for grpc request when request timeout.
+[#10343] Use CMS as default GC when jdk less 9.
+[#10361] Refactor module switches to make only load specified module but not only close in console ui.
+[#10520] Validate for namespace show name when create new namespace.
+[#10521] Enhance the hints for `No DataSourceSet` error by validate datasource after construction.
+[#10539] Enhance logs when opeation configuration failed.
+[#10730] Link to v2 document for console ui.
+[#10811] Enhance compatibility for colorful service healthy status in console ui.
+[#10891] Support setting maximum number of push retries.
+[#10930] Forward compatible old version secretKey for default auth plugin.
+[#11129] Remove the namespace information from the node list page.
+[#11231] Optimize the handleSpringBinder method in PropertiesUtil.
+
+## BugFix
+[#10056] Fix loss revision of client for distro sync.
+[#10128] Fix wrong judgement in raft stateMachine.
+[#10149] Fix dead lock on sending connection reset request on server over limit.
+[#10271] Fix nacos-client failover switch file path.
+[#10318] Fix import configuration problem.
+[#10347] Fix only admin role user can register service into default namespace when enabled default auth plugin.
+[#10406] Fix jraft install leader snapshot error after disconnection.
+[#10427] Fix nacos client no response when handle server request with exception.
+[#10464] Fix NPE when concurrent operations for client.
+[#10470] Fix some missed i18n for console ui.
+[#10509] Fix out data connection not be disconnect problem.
+[#10548] Fix switch domain might not load snapshot after restart.
+[#10556] Fix index loss for client and service in extreme scenarios.
+[#10583] Fix some new API loss auth check.
+[#10585] Fix selectInstances and selectOneHealthyInstance methods will not subscribe service problem.
+[#10593] Fix invalid create `file:` dir under `nacos.home`.
+[#10598] Fix nacos-client not random get server address when using address.
+[#10606] Fix memory leak for nacos client when user create and shutdown client frequently.
+[#10657] Fix NPE when using derby datasource for cluster mode.
+[#10935] Fix startsWith judgement wrong when ignoreCase is true.
+[#11056] Fix Batch register count size wrong, when batch register sereval time.
+[#11059] Fix RPC_CLIENT_TLS_PROTOCOLS setting error.
+[#11192] Fix batchRegisterInstance not recalculate revision prblem.
+[#11197] Fix frequent do query service when hit protect empty.
+
+## Dependency
+[#7698] Remove httpasyncclient version dependency management to avoid version conflicts.
+[#10416] Upgrade console yaml editor.
+[#10648] Optimize Guava Dependency.
+[#10893] Upgrade spring boot to 2.7.15.
+[#11199] Upgrade grpc version to 1.57.2.
+
+

开源之夏OSPP 2023以及编程夏令营GLCC 2023

+

经过夏天如火如荼的努力,Nacos今年的开源之夏也接近尾声,各项目进展稳步推进,均取得了不小的进展。

+

开源之夏 OSPP 2023

+

Nacos在2023年度的开源之夏 中,发布了9个项目,其中6个项目成功匹配同学,并基本完成了项目课题,其中不乏大量社区期待的新增功能和优化,将在后续版本中加入。

+

入选项目列表

+
    +
  • 客户端订阅者合并和数据选择功能
  • +
  • 为nacos-spring-boot-starter支持适配spring boot 3
  • +
  • ConfigMap到Nacos配置中心自动同步工具
  • +
  • 客户端可观测性指标建设
  • +
  • 支持Nacos注册中心的模糊订阅能力
  • +
  • 实现其他部分xDS协议
  • +
+

感谢导师和同学们的辛苦付出,也感谢主办方中科院软件所提供的优秀平台和活动。

+

Nacos 荣获编程夏令营 GLCC 2023优秀社区

+

Nacos在2023年度的编程夏令营 申报了一个Nacos3.0中呼声较高的一个功能作为课题 -- 支持分布式锁。 目前已在导师和同学的共同努力下,初步完成了功能,预计会在Nacos3.0的Alpha版本中在社区公布。

+

由于导师和同学的努力,Nacos社区荣获CCF Gitlink 颁发的开源编程夏令营的优秀社区奖,感谢主办方Gitlink为社区提供的平台和活动。

+ + + + + + + + +
奖杯现场
+

展望

+

2.X 后续计划

+

从2021年3月 2.0.0正式版发布至今,2.X版本已经走了接近2年时间,如今2.3.0版本发布,完成了大部分功能的插件化提炼,在之后的2.3.X版本中,会主要对当前版本的问题进行修复,并做出小范围的功能优化。同时对于2.4.0版本,会作为一个Nacos3.0的过度版本,对大量代码进行优化重构,在提升稳定性、健壮性的同时,提升易用性和可观测性,向Nacos3.0版本平稳过度。

+

3.0 计划

+

Nacos社区同时也开启了关于Nacos3.0 的畅想和规划,Nacos将会从统一控制面、支持国产化、存储计算分离等方向进一步演进Nacos的功能和架构,欢迎社区积极参与到新版本的建设中。

+

image.png

+

image.png

+

About Nacos

+

Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。

+

Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。 Nacos 是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施。

+

最后欢迎大家扫码加入Nacos社区群

+

image.png

+
+ + + + + + + + + + + + + diff --git a/zh-cn/blog/2.3.0-beta-release.json b/zh-cn/blog/2.3.0-beta-release.json new file mode 100644 index 00000000000..86489023819 --- /dev/null +++ b/zh-cn/blog/2.3.0-beta-release.json @@ -0,0 +1,11 @@ +{ + "filename": "2.3.0-beta-release.md", + "__html": "

Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用

\n

新版本发布

\n

经过4个多月社区的大量贡献,Nacos 2.3.0版本终于进入的Beta发布阶段,在经历了2.2.X的多个安全风险修复的版本后,又一次迎来了功能性上的更新版本。

\n

Nacos 反脆弱插件

\n

2.2.0版本加入ALPHA版本的反脆弱插件后,社区经过多次重构和修改,提升抽象度和稳定性,终于在2.3.0版本中正式公布了Nacos的反脆弱插件,用于避免Nacos集群在大量请求和连接时导致的容量问题而引发更大规模的故障。

\n

默认反脆弱实现插件的使用以及自定义反脆弱插件的开发可参考文档反脆弱插件.

\n

Nacos 配置变更插件

\n

Nacos配置管理作为Nacos的核心功能,一直受到开发人员、运维人员和审计人员的关注,不少运维人员和审计人员向社区提出诉求,希望Nacos能够提供配置变更的审计、变更通知等功能;同时开发人员也希望Nacos提供一定的格式校验功能,避免修改配置时出现错误导致应用故障。

\n

由于Nacos配置管理本身对配置内容是不感知的,且各个用户公司的审计、通知系统也各不相同,因此Nacos为了能够更广阔的适配各个用户的诉求,在去年的开源之夏中制定了配置变更插件的课题,希望以插件化的形式,来满足对配置变更过程的介入。

\n

经过同学和导师长期的努力,该插件在2.3.0版本发布,用户可以开发自定义插件,在配置变更前和变更完成后,分别对接公司的审计系统和通知系统,并在变更前进行更多的校验操作,如格式校验,配置名字的规范化等。同时Nacos社区也提供了默认插件的实现,可接入webhook进行配置变更的通知以及导入配置的文件名校验等;详情可参考文档配置变更插件

\n

Nacos 请求参数校验

\n

2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。

\n

在2.3.0版本中,Nacos明确了参数校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据校验规则对客户端向服务端发送的请求进行校验。用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。

\n

详细情况可参考文档参数校验规则

\n

Nacos 能力协商

\n

随着Nacos功能越来越多,版本越来越多,客户端和服务端之间的兼容性愈发重要,如果继续保持尝试性的兼容,只会兼容能力愈发的困难。因此Nacos社区在去年开源之夏中,发布了课题,通过增加客户端和服务端之间的能力协商机制,在客户端连接到服务端时,让连接双方知道对方分别支持的功能,在支持对应功能的情况下,开启对应功能。避免通过尝试而增加通信开销。

\n

在2.3.0版本中,该能力也被集成进来,为之后Nacos3.0支持更多优化功能提供基础。

\n

其他重要改动

\n

除了上述改动,Nacos2.3.0版本还支持了:客户端异常的metrics、grpc长连接的SSL功能、关闭开源控制台等功能,更多细节可以查看变更列表,欢迎试用:

\n
## feature\n[#5698] Support nacos control plugin.\n[#8458] Support ability negotiations between server and clients.\n[#8460] Support config change hook plugin.\n[#10117] Support metrics for nacos client request server exception.\n[#10150] Support SSL for grpc connection.\n[#10223] Support auto build instance id when client request instance id is null.\n[#10288] Support get more module state and switches in console.\n[#10734] Support validate most of request parameters.\n[#10774] Support toml format for configuration in console ui.\n[#10831] Support batch deregister instances for service.\n[#10971] Support disable console ui and support add guide information.\n\n## Enhancement&Refactor\n[#6819] Add page size selector in service details page.\n[#8107][#9109][#10169][#10176] Enhance hint when console ui session expired for default auth plugin.\n[#9085] Add the Reachability Metadata required by native-image.\n[#9821] Enhance datasource plugin to make more datasource implementation easier.\n[#9881] Enhance configuration page to supports folding when editing configuration.\n[#10067] Enhance Windows compatibility for configuration snapshot.\n[#10155] Enhance hints for grpc request when request timeout.\n[#10343] Use CMS as default GC when jdk less 9.\n[#10361] Refactor module switches to make only load specified module but not only close in console ui.\n[#10520] Validate for namespace show name when create new namespace.\n[#10521] Enhance the hints for `No DataSourceSet` error by validate datasource after construction.\n[#10539] Enhance logs when opeation configuration failed.\n[#10730] Link to v2 document for console ui.\n[#10811] Enhance compatibility for colorful service healthy status in console ui.\n[#10891] Support setting maximum number of push retries.\n[#10930] Forward compatible old version secretKey for default auth plugin.\n[#11129] Remove the namespace information from the node list page.\n[#11231] Optimize the handleSpringBinder method in PropertiesUtil.\n\n## BugFix\n[#10056] Fix loss revision of client for distro sync.\n[#10128] Fix wrong judgement in raft stateMachine.\n[#10149] Fix dead lock on sending connection reset request on server over limit.\n[#10271] Fix nacos-client failover switch file path.\n[#10318] Fix import configuration problem.\n[#10347] Fix only admin role user can register service into default namespace when enabled default auth plugin.\n[#10406] Fix jraft install leader snapshot error after disconnection.\n[#10427] Fix nacos client no response when handle server request with exception.\n[#10464] Fix NPE when concurrent operations for client.\n[#10470] Fix some missed i18n for console ui.\n[#10509] Fix out data connection not be disconnect problem.\n[#10548] Fix switch domain might not load snapshot after restart.\n[#10556] Fix index loss for client and service in extreme scenarios.\n[#10583] Fix some new API loss auth check.\n[#10585] Fix selectInstances and selectOneHealthyInstance methods will not subscribe service problem.\n[#10593] Fix invalid create `file:` dir under `nacos.home`.\n[#10598] Fix nacos-client not random get server address when using address.\n[#10606] Fix memory leak for nacos client when user create and shutdown client frequently.\n[#10657] Fix NPE when using derby datasource for cluster mode.\n[#10935] Fix startsWith judgement wrong when ignoreCase is true.\n[#11056] Fix Batch register count size wrong, when batch register sereval time.\n[#11059] Fix RPC_CLIENT_TLS_PROTOCOLS setting error.\n[#11192] Fix batchRegisterInstance not recalculate revision prblem.\n[#11197] Fix frequent do query service when hit protect empty.\n\n## Dependency\n[#7698] Remove httpasyncclient version dependency management to avoid version conflicts.\n[#10416] Upgrade console yaml editor.\n[#10648] Optimize Guava Dependency.\n[#10893] Upgrade spring boot to 2.7.15.\n[#11199] Upgrade grpc version to 1.57.2.\n
\n

开源之夏OSPP 2023以及编程夏令营GLCC 2023

\n

经过夏天如火如荼的努力,Nacos今年的开源之夏也接近尾声,各项目进展稳步推进,均取得了不小的进展。

\n

开源之夏 OSPP 2023

\n

Nacos在2023年度的开源之夏 中,发布了9个项目,其中6个项目成功匹配同学,并基本完成了项目课题,其中不乏大量社区期待的新增功能和优化,将在后续版本中加入。

\n

入选项目列表

\n
    \n
  • 客户端订阅者合并和数据选择功能
  • \n
  • 为nacos-spring-boot-starter支持适配spring boot 3
  • \n
  • ConfigMap到Nacos配置中心自动同步工具
  • \n
  • 客户端可观测性指标建设
  • \n
  • 支持Nacos注册中心的模糊订阅能力
  • \n
  • 实现其他部分xDS协议
  • \n
\n

感谢导师和同学们的辛苦付出,也感谢主办方中科院软件所提供的优秀平台和活动。

\n

Nacos 荣获编程夏令营 GLCC 2023优秀社区

\n

Nacos在2023年度的编程夏令营 申报了一个Nacos3.0中呼声较高的一个功能作为课题 -- 支持分布式锁。 目前已在导师和同学的共同努力下,初步完成了功能,预计会在Nacos3.0的Alpha版本中在社区公布。

\n

由于导师和同学的努力,Nacos社区荣获CCF Gitlink 颁发的开源编程夏令营的优秀社区奖,感谢主办方Gitlink为社区提供的平台和活动。

\n\n\n\n\n\n\n\n\n
\"奖杯\"\"现场\"
\n

展望

\n

2.X 后续计划

\n

从2021年3月 2.0.0正式版发布至今,2.X版本已经走了接近2年时间,如今2.3.0版本发布,完成了大部分功能的插件化提炼,在之后的2.3.X版本中,会主要对当前版本的问题进行修复,并做出小范围的功能优化。同时对于2.4.0版本,会作为一个Nacos3.0的过度版本,对大量代码进行优化重构,在提升稳定性、健壮性的同时,提升易用性和可观测性,向Nacos3.0版本平稳过度。

\n

3.0 计划

\n

Nacos社区同时也开启了关于Nacos3.0 的畅想和规划,Nacos将会从统一控制面、支持国产化、存储计算分离等方向进一步演进Nacos的功能和架构,欢迎社区积极参与到新版本的建设中。

\n

\"image.png\"

\n

\"image.png\"

\n

About Nacos

\n

Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。

\n

Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。 Nacos 是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施。

\n

最后欢迎大家扫码加入Nacos社区群

\n

\"image.png\"

\n", + "link": "/zh-cn/blog/2.3.0-beta-release.html", + "meta": { + "title": "Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用", + "keywords": "2.3.0", + "description": "Nacos 荣获GLCC优秀社区,同时2.3.0-BETA发布,欢迎试用", + "date": "2023-10-24" + } +} \ No newline at end of file diff --git a/zh-cn/blog/index.html b/zh-cn/blog/index.html index 701aed1f4c0..799a177ca53 100644 --- a/zh-cn/blog/index.html +++ b/zh-cn/blog/index.html @@ -15,7 +15,7 @@ -
博客

所有文章

+
博客

所有文章

diff --git a/zh-cn/docs/v2/guide/admin/console-guide.html b/zh-cn/docs/v2/guide/admin/console-guide.html index 70904f78ae1..d536335fbea 100644 --- a/zh-cn/docs/v2/guide/admin/console-guide.html +++ b/zh-cn/docs/v2/guide/admin/console-guide.html @@ -101,18 +101,17 @@

修改默认用户名/密码方法

INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHzz32dJN7jexM34MOeYirDdFAZm2kuWj7VEOJhhZkDrxfvUu', TRUE);
 INSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');
 
+

关闭登录功能

-

由于部分公司自己开发控制台,不希望被nacos的安全filter拦截。因此nacos支持定制关闭登录功能找到配置文件 ${nacoshome}/conf/application.properties, 替换以下内容即可。

-
## spring security config
-### turn off security
-spring.security.enabled=false
-management.security=false
-security.basic.enabled=false
-nacos.security.ignore.urls=/**
-
-#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/**
-
-
+

Nacos默认控制台在2.2.2版本前,无论是否开启鉴权功能,默认控制台都会跳转到登录页,导致用户被误导认为控制台存在鉴权功能,实际没有开启鉴权,存在安全隐患。

+

经过社区和安全工程师协商讨论,需要在使用Nacos默认控制台时,鉴权开关关闭时将会自动关闭控制台登录功能。

+

因此从2.2.2版本开始,当鉴权开关nacos.core.auth.enabled关闭时,Nacos默认控制台将不再跳转登录页,同时添加页面提示,提示当前集群未开启鉴权功能。

+

同时针对自定义的鉴权插件添加新接口com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled(默认返回false)来对自定义插件进行登录页控制。

+

+

关闭默认控制台

+

部分公司或用户希望关闭默认控制台,使用公司的统一平台进行Nacos的配置和服务管理;或将控制台鉴权和客户端访问的鉴权分离,即控制台操作进行鉴权但客户端请求不进行鉴权。

+

2.3.0版本开始,可以通过${nacoshome}/conf/application.properties中的nacos.console.ui.enabled来开启或关闭Nacos默认控制台,默认为开启。

+

同时在关闭默认控制台时,默认控制台会读取${nacoshome}/conf/console-guide.conf文件中的内容,并在默认控制台中生成引导页,让维护者自定义将使用默认控制台的用户引导向自定义的统一平台上进行操作。

会话时间

默认会话保持时间为30分钟。30分钟后需要重新登录认证。 暂时不支持修改该默认时间。

社区参与的前端共建

diff --git a/zh-cn/docs/v2/guide/admin/console-guide.json b/zh-cn/docs/v2/guide/admin/console-guide.json index 25c00518a86..06f6f5f51cd 100644 --- a/zh-cn/docs/v2/guide/admin/console-guide.json +++ b/zh-cn/docs/v2/guide/admin/console-guide.json @@ -1,6 +1,6 @@ { "filename": "console-guide.md", - "__html": "

控制台手册

\n

Nacos 控制台主要旨在于增强对于服务列表,健康状态管理,服务治理,分布式配置管理等方面的管控能力,以便进一步帮助用户降低管理微服务应用架构的成本,将提供包括下列基本功能:

\n
    \n
  • 服务管理\n
      \n
    • 服务列表及服务健康状态展示
    • \n
    • 服务元数据存储及编辑
    • \n
    • 服务流量权重的调整
    • \n
    • 服务优雅上下线
    • \n
    \n
  • \n
  • 配置管理\n
      \n
    • 多种配置格式编辑
    • \n
    • 编辑DIFF
    • \n
    • 示例代码
    • \n
    • 推送状态查询
    • \n
    • 配置版本及一键回滚
    • \n
    \n
  • \n
  • 命名空间
  • \n
  • 登录管理
  • \n
\n

特性详解

\n

服务管理

\n

开发者或者运维人员往往需要在服务注册后,通过友好的界面来查看服务的注册情况,包括当前系统注册的所有服务和每个服务的详情。并在有权限控制的情况下,进行服务的一些配置的编辑操作。Nacos在这个版本开放的控制台的服务发现部分,主要就是提供用户一个基本的运维页面,能够查看、编辑当前注册的服务。

\n

服务列表管理

\n

服务列表帮助用户以统一的视图管理其所有的微服务以及服务健康状态。整体界面布局是左上角有服务的搜索框和搜索按钮,页面中央是服务列表的展示。服务列表主要展示服务名、集群数目、实例数目、健康实例数目和详情按钮五个栏目。

\n

\"image.png

\n

在服务列表页面点击详情,可以看到服务的详情。可以查看服务、集群和实例的基本信息。

\n

服务流量权重支持及流量保护

\n

Nacos 为用户提供了流量权重控制的能力,同时开放了服务流量的阈值保护,以帮助用户更好的保护服务服务提供者集群不被意外打垮。如下图所以,可以点击实例的编辑按钮,修改实例的权重。如果想增加实例的流量,可以将权重调大,如果不想实例接收流量,则可以将权重设为0。

\n

\"image.png

\n

服务元数据管理

\n

Nacos提供多个维度的服务元数据的暴露,帮助用户存储自定义的信息。这些信息都是以K-V的数据结构存储,在控制台上,会以k1=v1,k2=v2这样的格式展示。类似的,编辑元数据可以通过相同的格式进行。例如服务的元数据编辑,首先点击服务详情页右上角的“编辑服务”按钮,然后在元数据输入框输入:version=1.0,env=prod。

\n

\"image.png

\n

点击确认,就可以在服务详情页面,看到服务的元数据已经更新了。

\n

\"image.png

\n

服务优雅上下线

\n

Nacos还提供服务实例的上下线操作,在服务详情页面,可以点击实例的“上线”或者“下线”按钮,被下线的实例,将不会包含在健康的实例列表里。

\n

\"image.png

\n

配置管理

\n

Nacos支持基于Namespace和Group的配置分组管理,以便用户更灵活的根据自己的需要按照环境或者应用、模块等分组管理微服务以及Spring的大量配置,在配置管理中主要提供了配置历史版本、回滚、订阅者查询等核心管理能力。

\n

\"image.png

\n

多配置格式编辑器

\n

Nacos支持 YAML、Properties、TEXT、JSON、XML、HTML 等常见配置格式在线编辑、语法高亮、格式校验,帮助用户高效编辑的同时大幅降低格式错误带来的风险。

\n

Nacos支持配置标签的能力,帮助用户更好、更灵活的做到基于标签的配置分类及管理。同时支持用户对配置及其变更进行描述,方便多人或者跨团队协作管理配置。

\n

\"image.png

\n

编辑DIFF

\n

Nacos支持编辑DIFF能力,帮助用户校验修改内容,降低改错带来的风险。

\n

\"image.png

\n

示例代码

\n

Nacos提供示例代码能力,能够让新手快速使用客户端编程消费该配置,大幅降低新手使用门槛。

\n

\"image.png

\n

\"image.png

\n

监听者查询

\n

Nacos提供配置订阅者即监听者查询能力,同时提供客户端当前配置的MD5校验值,以便帮助用户更好的检查配置变更是否推送到 Client 端。

\n

\"image.png

\n

配置的版本及一键回滚

\n

Nacos通过提供配置版本管理及其一键回滚能力,帮助用户改错配置的时候能够快速恢复,降低微服务系统在配置管理上的一定会遇到的可用性风险。

\n

\"image.png

\n

\"image.png

\n

命名空间管理

\n

Nacos 基于Namespace 帮助用户逻辑隔离多个命名空间,这可以帮助用户更好的管理测试、预发、生产等多环境服务和配置,让每个环境的同一个配置(如数据库数据源)可以定义不同的值。

\n

\"image.png

\n

\"image.png

\n

登录管理

\n

Nacos0.8 版本支持简单登录功能,默认用户名/密码为: nacos/nacos

\n

\"login\"

\n

修改默认用户名/密码方法

\n
    \n
  1. 生成加密密码, 在com.alibaba.nacos.console.utils.PasswordEncoderUtil.main函数中,将 nacos 改成你要改成的密码,运行即可得到加密有算法。注意盐值是随机的,所以生成密码每次可能不一样,请不要担心。
  2. \n
\n
public class PasswordEncoderUtil {\n\n    public static void main(String[] args) {\n        System.out.println(new BCryptPasswordEncoder().encode("nacos"));\n    }\n}\n
\n
    \n
  1. 创建用户名或者密码的时候,用指定用户名密码即可。
  2. \n
\n
INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHzz32dJN7jexM34MOeYirDdFAZm2kuWj7VEOJhhZkDrxfvUu', TRUE);\nINSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');\n
\n

关闭登录功能

\n

由于部分公司自己开发控制台,不希望被nacos的安全filter拦截。因此nacos支持定制关闭登录功能找到配置文件 ${nacoshome}/conf/application.properties, 替换以下内容即可。

\n
## spring security config\n### turn off security\nspring.security.enabled=false\nmanagement.security=false\nsecurity.basic.enabled=false\nnacos.security.ignore.urls=/**\n\n#nacos.security.ignore.urls=/,/**/*.css,/**/*.js,/**/*.html,/**/*.map,/**/*.svg,/**/*.png,/**/*.ico,/console-fe/public/**,/v1/auth/login,/v1/console/health,/v1/cs/**,/v1/ns/**,/v1/cmdb/**,/actuator/**\n\n
\n

会话时间

\n

默认会话保持时间为30分钟。30分钟后需要重新登录认证。 暂时不支持修改该默认时间。

\n

社区参与的前端共建

\n

在Nacos前端风格、布局的讨论中,社区踊跃投票,最终选择了这套经典黑白蓝风格的皮肤,并且通过我们UED程瑶同学的设计,布局,让交互变得十分自然流畅。

\n

在控制台的开发之前,我们通过社区招募到了很多前端同学一起参与了前端代码的开发,在此尤其感谢李晨、王庆、王彦民同学在Nacos前端开发过程中的大力支持!

\n

坚持社区化发展,欢迎加入并贡献社区

\n
\n

比吐槽更重要的是搭把手,参与社区一起发展Nacos!

\n
\n

要加入Nacos 微信社区讨论 Nacos 产品的演进,你可以通过扫超哥的微信二维码,让“超哥” 帮你拉入 “Nacos社区交流群”。

\n

\"Screen

\n

更多与 Nacos 相关的开源项目信息:

\n\n", + "__html": "

控制台手册

\n

Nacos 控制台主要旨在于增强对于服务列表,健康状态管理,服务治理,分布式配置管理等方面的管控能力,以便进一步帮助用户降低管理微服务应用架构的成本,将提供包括下列基本功能:

\n
    \n
  • 服务管理\n
      \n
    • 服务列表及服务健康状态展示
    • \n
    • 服务元数据存储及编辑
    • \n
    • 服务流量权重的调整
    • \n
    • 服务优雅上下线
    • \n
    \n
  • \n
  • 配置管理\n
      \n
    • 多种配置格式编辑
    • \n
    • 编辑DIFF
    • \n
    • 示例代码
    • \n
    • 推送状态查询
    • \n
    • 配置版本及一键回滚
    • \n
    \n
  • \n
  • 命名空间
  • \n
  • 登录管理
  • \n
\n

特性详解

\n

服务管理

\n

开发者或者运维人员往往需要在服务注册后,通过友好的界面来查看服务的注册情况,包括当前系统注册的所有服务和每个服务的详情。并在有权限控制的情况下,进行服务的一些配置的编辑操作。Nacos在这个版本开放的控制台的服务发现部分,主要就是提供用户一个基本的运维页面,能够查看、编辑当前注册的服务。

\n

服务列表管理

\n

服务列表帮助用户以统一的视图管理其所有的微服务以及服务健康状态。整体界面布局是左上角有服务的搜索框和搜索按钮,页面中央是服务列表的展示。服务列表主要展示服务名、集群数目、实例数目、健康实例数目和详情按钮五个栏目。

\n

\"image.png

\n

在服务列表页面点击详情,可以看到服务的详情。可以查看服务、集群和实例的基本信息。

\n

服务流量权重支持及流量保护

\n

Nacos 为用户提供了流量权重控制的能力,同时开放了服务流量的阈值保护,以帮助用户更好的保护服务服务提供者集群不被意外打垮。如下图所以,可以点击实例的编辑按钮,修改实例的权重。如果想增加实例的流量,可以将权重调大,如果不想实例接收流量,则可以将权重设为0。

\n

\"image.png

\n

服务元数据管理

\n

Nacos提供多个维度的服务元数据的暴露,帮助用户存储自定义的信息。这些信息都是以K-V的数据结构存储,在控制台上,会以k1=v1,k2=v2这样的格式展示。类似的,编辑元数据可以通过相同的格式进行。例如服务的元数据编辑,首先点击服务详情页右上角的“编辑服务”按钮,然后在元数据输入框输入:version=1.0,env=prod。

\n

\"image.png

\n

点击确认,就可以在服务详情页面,看到服务的元数据已经更新了。

\n

\"image.png

\n

服务优雅上下线

\n

Nacos还提供服务实例的上下线操作,在服务详情页面,可以点击实例的“上线”或者“下线”按钮,被下线的实例,将不会包含在健康的实例列表里。

\n

\"image.png

\n

配置管理

\n

Nacos支持基于Namespace和Group的配置分组管理,以便用户更灵活的根据自己的需要按照环境或者应用、模块等分组管理微服务以及Spring的大量配置,在配置管理中主要提供了配置历史版本、回滚、订阅者查询等核心管理能力。

\n

\"image.png

\n

多配置格式编辑器

\n

Nacos支持 YAML、Properties、TEXT、JSON、XML、HTML 等常见配置格式在线编辑、语法高亮、格式校验,帮助用户高效编辑的同时大幅降低格式错误带来的风险。

\n

Nacos支持配置标签的能力,帮助用户更好、更灵活的做到基于标签的配置分类及管理。同时支持用户对配置及其变更进行描述,方便多人或者跨团队协作管理配置。

\n

\"image.png

\n

编辑DIFF

\n

Nacos支持编辑DIFF能力,帮助用户校验修改内容,降低改错带来的风险。

\n

\"image.png

\n

示例代码

\n

Nacos提供示例代码能力,能够让新手快速使用客户端编程消费该配置,大幅降低新手使用门槛。

\n

\"image.png

\n

\"image.png

\n

监听者查询

\n

Nacos提供配置订阅者即监听者查询能力,同时提供客户端当前配置的MD5校验值,以便帮助用户更好的检查配置变更是否推送到 Client 端。

\n

\"image.png

\n

配置的版本及一键回滚

\n

Nacos通过提供配置版本管理及其一键回滚能力,帮助用户改错配置的时候能够快速恢复,降低微服务系统在配置管理上的一定会遇到的可用性风险。

\n

\"image.png

\n

\"image.png

\n

命名空间管理

\n

Nacos 基于Namespace 帮助用户逻辑隔离多个命名空间,这可以帮助用户更好的管理测试、预发、生产等多环境服务和配置,让每个环境的同一个配置(如数据库数据源)可以定义不同的值。

\n

\"image.png

\n

\"image.png

\n

登录管理

\n

Nacos0.8 版本支持简单登录功能,默认用户名/密码为: nacos/nacos

\n

\"login\"

\n

修改默认用户名/密码方法

\n
    \n
  1. 生成加密密码, 在com.alibaba.nacos.console.utils.PasswordEncoderUtil.main函数中,将 nacos 改成你要改成的密码,运行即可得到加密有算法。注意盐值是随机的,所以生成密码每次可能不一样,请不要担心。
  2. \n
\n
public class PasswordEncoderUtil {\n\n    public static void main(String[] args) {\n        System.out.println(new BCryptPasswordEncoder().encode("nacos"));\n    }\n}\n
\n
    \n
  1. 创建用户名或者密码的时候,用指定用户名密码即可。
  2. \n
\n
INSERT INTO users (username, password, enabled) VALUES ('nacos', '$2a$10$EuWPZHzz32dJN7jexM34MOeYirDdFAZm2kuWj7VEOJhhZkDrxfvUu', TRUE);\nINSERT INTO roles (username, role) VALUES ('nacos', 'ROLE_ADMIN');\n
\n

\n

关闭登录功能

\n

Nacos默认控制台在2.2.2版本前,无论是否开启鉴权功能,默认控制台都会跳转到登录页,导致用户被误导认为控制台存在鉴权功能,实际没有开启鉴权,存在安全隐患。

\n

经过社区和安全工程师协商讨论,需要在使用Nacos默认控制台时,鉴权开关关闭时将会自动关闭控制台登录功能。

\n

因此从2.2.2版本开始,当鉴权开关nacos.core.auth.enabled关闭时,Nacos默认控制台将不再跳转登录页,同时添加页面提示,提示当前集群未开启鉴权功能。

\n

同时针对自定义的鉴权插件添加新接口com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService#isLoginEnabled(默认返回false)来对自定义插件进行登录页控制。

\n

\n

关闭默认控制台

\n

部分公司或用户希望关闭默认控制台,使用公司的统一平台进行Nacos的配置和服务管理;或将控制台鉴权和客户端访问的鉴权分离,即控制台操作进行鉴权但客户端请求不进行鉴权。

\n

2.3.0版本开始,可以通过${nacoshome}/conf/application.properties中的nacos.console.ui.enabled来开启或关闭Nacos默认控制台,默认为开启。

\n

同时在关闭默认控制台时,默认控制台会读取${nacoshome}/conf/console-guide.conf文件中的内容,并在默认控制台中生成引导页,让维护者自定义将使用默认控制台的用户引导向自定义的统一平台上进行操作。

\n

会话时间

\n

默认会话保持时间为30分钟。30分钟后需要重新登录认证。 暂时不支持修改该默认时间。

\n

社区参与的前端共建

\n

在Nacos前端风格、布局的讨论中,社区踊跃投票,最终选择了这套经典黑白蓝风格的皮肤,并且通过我们UED程瑶同学的设计,布局,让交互变得十分自然流畅。

\n

在控制台的开发之前,我们通过社区招募到了很多前端同学一起参与了前端代码的开发,在此尤其感谢李晨、王庆、王彦民同学在Nacos前端开发过程中的大力支持!

\n

坚持社区化发展,欢迎加入并贡献社区

\n
\n

比吐槽更重要的是搭把手,参与社区一起发展Nacos!

\n
\n

要加入Nacos 微信社区讨论 Nacos 产品的演进,你可以通过扫超哥的微信二维码,让“超哥” 帮你拉入 “Nacos社区交流群”。

\n

\"Screen

\n

更多与 Nacos 相关的开源项目信息:

\n\n", "link": "/zh-cn/docs/v2/guide/admin/console-guide.html", "meta": { "title": "控制台手册", diff --git a/zh-cn/docs/v2/guide/user/auth.html b/zh-cn/docs/v2/guide/user/auth.html index c4aa16ee91c..ed5c849c02b 100644 --- a/zh-cn/docs/v2/guide/user/auth.html +++ b/zh-cn/docs/v2/guide/user/auth.html @@ -91,6 +91,9 @@

相关参数

+

默认控制台登录页

+

2.2.2版本之前的Nacos默认控制台,无论服务端是否开启鉴权,都会存在一个登录页;这导致很多用户被误导认为Nacos默认是存在鉴权的。在社区安全工程师的建议下,Nacos自2.2.2版本开始,在未开启鉴权时,默认控制台将不需要登录即可访问,同时在控制台中给予提示,提醒用户当前集群未开启鉴权。

+

在用户开启鉴权后,控制台才需要进行登录访问。 同时针对不同的鉴权插件,提供新的接口方法,用于提示控制台是否开启登录页;同时在2.2.3版本后,Nacos可支持关闭开源控制台,并引导到用户自定义的Nacos控制台,详情可查看Nacos鉴权插件-服务端插件控制台手册-关闭登录功能

服务端如何开启鉴权

非Docker环境

按照官方文档配置启动,默认是不需要登录的,这样会导致配置中心对外直接暴露。而启用鉴权之后,需要在使用用户名和密码登录之后,才能正常使用nacos。

diff --git a/zh-cn/docs/v2/guide/user/auth.json b/zh-cn/docs/v2/guide/user/auth.json index 902ce00c4a7..2e968a17488 100644 --- a/zh-cn/docs/v2/guide/user/auth.json +++ b/zh-cn/docs/v2/guide/user/auth.json @@ -1,6 +1,6 @@ { "filename": "auth.md", - "__html": "
\n

注意

\n
    \n
  • Nacos是一个内部微服务组件,需要在可信的内部网络中运行,不可暴露在公网环境,防止带来安全风险。
  • \n
  • Nacos提供简单的鉴权实现,为防止业务错用的弱鉴权体系,不是防止恶意攻击的强鉴权体系。
  • \n
  • 如果运行在不可信的网络环境或者有强鉴权诉求,请参考官方简单实现做进行自定义插件开发
  • \n
\n
\n

鉴权

\n

相关参数

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
参数名默认值启止版本说明
nacos.core.auth.enabledfalse1.2.0 ~ latest是否开启鉴权功能
nacos.core.auth.system.typenacos1.2.0 ~ latest鉴权类型
nacos.core.auth.plugin.nacos.token.secret.keySecretKey012345678901234567890123456789012345678901234567890123456789(2.2.0.1后无默认值)2.1.0 ~ latest默认鉴权插件用于生成用户登陆临时accessToken所使用的密钥,使用默认值有安全风险
nacos.core.auth.plugin.nacos.token.expire.seconds180002.1.0 ~ latest用户登陆临时accessToken的过期时间
nacos.core.auth.enable.userAgentAuthWhitefalse1.4.1 ~ latest是否使用useragent白名单,主要用于适配老版本升级,置为true时有安全风险
nacos.core.auth.server.identity.keyserverIdentity(2.2.1后无默认值)1.4.1 ~ latest用于替换useragent白名单的身份识别key,使用默认值有安全风险
nacos.core.auth.server.identity.valuesecurity(2.2.1后无默认值)1.4.1 ~ latest用于替换useragent白名单的身份识别value,使用默认值有安全风险
nacos.core.auth.default.token.secret.keySecretKey0123456789012345678901234567890123456789012345678901234567891.2.0 ~ 2.0.4nacos.core.auth.plugin.nacos.token.secret.key
nacos.core.auth.default.token.expire.seconds180001.2.0 ~ 2.0.4nacos.core.auth.plugin.nacos.token.expire.seconds
\n

服务端如何开启鉴权

\n

非Docker环境

\n

按照官方文档配置启动,默认是不需要登录的,这样会导致配置中心对外直接暴露。而启用鉴权之后,需要在使用用户名和密码登录之后,才能正常使用nacos。

\n

开启鉴权之前,application.properties中的配置信息为:

\n
### If turn on auth system:\nnacos.core.auth.enabled=false\n
\n

开启鉴权之后,application.properties中的配置信息为:

\n
### If turn on auth system:\nnacos.core.auth.system.type=nacos\nnacos.core.auth.enabled=true\n
\n

自定义密钥

\n

开启鉴权之后,你可以自定义用于生成JWT令牌的密钥,application.properties中的配置信息为:

\n
\n

注意:

\n
    \n
  1. 文档中提供的密钥为公开密钥,在实际部署时请更换为其他密钥内容,防止密钥泄漏导致安全风险。
  2. \n
  3. 在2.2.0.1版本后,社区发布版本将移除以文档如下值作为默认值,需要自行填充,否则无法启动节点。
  4. \n
  5. 密钥需要保持节点间一致,长时间不一致可能导致403 invalid token错误。
  6. \n
\n
\n
### The default token(Base64 String):\nnacos.core.auth.default.token.secret.key=SecretKey012345678901234567890123456789012345678901234567890123456789\n\n### 2.1.0 版本后\nnacos.core.auth.plugin.nacos.token.secret.key=SecretKey012345678901234567890123456789012345678901234567890123456789\n
\n

自定义密钥时,推荐将配置项设置为Base64编码的字符串,且原始密钥长度不得低于32字符。例如下面的的例子:

\n
### The default token(Base64 String):\nnacos.core.auth.default.token.secret.key=VGhpc0lzTXlDdXN0b21TZWNyZXRLZXkwMTIzNDU2Nzg=\n\n### 2.1.0 版本后\nnacos.core.auth.plugin.nacos.token.secret.key=VGhpc0lzTXlDdXN0b21TZWNyZXRLZXkwMTIzNDU2Nzg=\n
\n
\n

注意:鉴权开关是修改之后立马生效的,不需要重启服务端。动态修改token.secret.key时,请确保token是有效的,如果修改成无效值,会导致后续无法登录,请求访问异常。

\n
\n

Docker环境

\n

官方镜像

\n

如果使用官方镜像,请在启动docker容器时,添加如下环境变量

\n
NACOS_AUTH_ENABLE=true\n
\n

例如,可以通过如下命令运行开启了鉴权的容器:

\n
docker run --env PREFER_HOST_MODE=hostname --env MODE=standalone --env NACOS_AUTH_ENABLE=true -p 8848:8848 nacos/nacos-server\n
\n

除此之外,还可以添加其他鉴权相关的环境变量信息:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
namedescriptionoption
NACOS_AUTH_ENABLE是否开启权限系统默认:false
NACOS_AUTH_TOKEN_EXPIRE_SECONDStoken 失效时间默认:18000
NACOS_AUTH_TOKENtoken默认:SecretKey012345678901234567890123456789012345678901234567890123456789
NACOS_AUTH_CACHE_ENABLE权限缓存开关 ,开启后权限缓存的更新默认有15秒的延迟默认 : false
\n

然后运行docker-compose构建命令,例如

\n
docker-compose -f example/standalone-derby.yaml up\n
\n

自定义镜像

\n

如果选择自定义镜像,请在构建镜像之前,修改nacos工程中的application.properties文件,

\n

将下面这一行配置信息

\n
nacos.core.auth.enabled=false\n
\n

修改为

\n
nacos.core.auth.system.type=nacos\nnacos.core.auth.enabled=true\n
\n

然后再配置nacos启动命令。

\n

客户端如何进行鉴权

\n

Java SDK鉴权

\n

在构建“Properties”类时,需传入用户名和密码。

\n
properties.put(\"username\",\"${username}\");\nproperties.put(\"password\",\"${password}\");\n
\n

示例代码

\n
try {\n    // Initialize the configuration service, and the console automatically obtains the following parameters through the sample code.\n\tString serverAddr = \"{serverAddr}\";\n\tProperties properties = new Properties();\n\tproperties.put(\"serverAddr\", serverAddr);\n\n    // if need username and password to login\n        properties.put(\"username\",\"nacos\");\n        properties.put(\"password\",\"nacos\");\n\n\tConfigService configService = NacosFactory.createConfigService(properties);\n} catch (NacosException e) {\n    // TODO Auto-generated catch block\n    e.printStackTrace();\n}\n
\n

其他语言的SDK鉴权

\n

待补充

\n

Open-API鉴权

\n

首先需要使用用户名和密码登陆nacos。

\n
curl -X POST '127.0.0.1:8848/nacos/v1/auth/login' -d 'username=nacos&password=nacos'\n
\n

若用户名和密码正确,返回信息如下:

\n
{"accessToken":"eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyOTE2Nn0.2TogGhhr11_vLEjqKko1HJHUJEmsPuCxkur-CfNojDo","tokenTtl":18000,"globalAdmin":true}\n
\n

接下来进行配置信息或服务信息时,应当使用该accessToken鉴权,在url后添加参数accessToken={accessToken},其中{accessToken}为登录时返回的token信息,例如

\n
curl -X GET '127.0.0.1:8848/nacos/v1/cs/configs?accessToken=eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyMzkyM30.O-s2yWfDSUZ7Svd3Vs7jy9tsfDNHs1SuebJB4KlNY8Q&dataId=nacos.example.1&group=nacos_group'\n
\n
curl -X POST 'http://127.0.0.1:8848/nacos/v1/ns/instance?accessToken=eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyMzkyM30.O-s2yWfDSUZ7Svd3Vs7jy9tsfDNHs1SuebJB4KlNY8Q&port=8848&healthy=true&ip=11.11.11.11&weight=1.0&serviceName=nacos.test.3&encoding=GBK&namespaceId=n1'\n
\n

开启Token缓存功能

\n

服务端自2.2.1版本后,默认鉴权插件模块支持token缓存功能,可参见ISSUE #9906

\n
https://github.com/alibaba/nacos/issues/9906\n
\n

背景

\n

无论是客户端SDK还是OpenAPI,在调用login接口获取accessToken之后,携带accessToken访问服务端,服务端解析Token进行鉴权。解析的动作比较耗时,如果想要提升接口的性能,可以考虑开启缓存Token的功能,用字符串比较代替Token解析。

\n

开启方式

\n
nacos.core.auth.plugin.nacos.token.cache.enable=true\n
\n

注意事项

\n

在开启Token缓存功能之前,服务端对每一个携带用户名密码访问login接口的请求都会生成新的token,接口的返回值中的tokenTtl字段跟服务端配置文件中设置的值相等,配置如下:

\n
nacos.core.auth.plugin.nacos.token.expire.seconds=18000\n
\n

在开启Token缓存功能之后,服务端对每一个携带用户名密码访问login接口的请求,会先检查缓存中是否存在该用户名对应的token。若不存在,生成新的Token,插入缓存再返回;若存在,返回该token,此时tokenTtl字段的值为配置文件中设置的值减去该Token在缓存中存留的时长。\n如果Token在缓存中存留的时长超过配置文件设置的值的90%,当login接口收到请求时,尽管缓存中存在该用户名对应的Token,服务端会重新生成Token返回给请求方,并更新缓存。因此,最差情况下,请求方收到的tokenTtl只有配置文件设置的值的10%。

\n

开启服务身份识别功能

\n

开启鉴权功能后,服务端之间的请求也会通过鉴权系统的影响。考虑到服务端之间的通信应该是可信的,因此在1.2~1.4.0版本期间,通过User-Agent中是否包含Nacos-Server来进行判断请求是否来自其他服务端。

\n

但这种实现由于过于简单且固定,导致可能存在安全问题。因此从1.4.1版本开始,Nacos添加服务身份识别功能,用户可以自行配置服务端的Identity,不再使用User-Agent作为服务端请求的判断标准。

\n

开启方式:

\n
### 开启鉴权\nnacos.core.auth.enabled=true\n\n### 关闭使用user-agent判断服务端请求并放行鉴权的功能\nnacos.core.auth.enable.userAgentAuthWhite=false\n\n### 配置自定义身份识别的key(不可为空)和value(不可为空)\nnacos.core.auth.server.identity.key=example\nnacos.core.auth.server.identity.value=example\n
\n

** 注意 ** 所有集群均需要配置相同的server.identity信息,否则可能导致服务端之间数据不一致或无法删除实例等问题。

\n

旧版本升级

\n

考虑到旧版本用户需要升级,可以在升级期间,开启nacos.core.auth.enable.userAgentAuthWhite=true功能,待集群整体升级到1.4.1并稳定运行后,再关闭此功能。

\n", + "__html": "
\n

注意

\n
    \n
  • Nacos是一个内部微服务组件,需要在可信的内部网络中运行,不可暴露在公网环境,防止带来安全风险。
  • \n
  • Nacos提供简单的鉴权实现,为防止业务错用的弱鉴权体系,不是防止恶意攻击的强鉴权体系。
  • \n
  • 如果运行在不可信的网络环境或者有强鉴权诉求,请参考官方简单实现做进行自定义插件开发
  • \n
\n
\n

鉴权

\n

相关参数

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
参数名默认值启止版本说明
nacos.core.auth.enabledfalse1.2.0 ~ latest是否开启鉴权功能
nacos.core.auth.system.typenacos1.2.0 ~ latest鉴权类型
nacos.core.auth.plugin.nacos.token.secret.keySecretKey012345678901234567890123456789012345678901234567890123456789(2.2.0.1后无默认值)2.1.0 ~ latest默认鉴权插件用于生成用户登陆临时accessToken所使用的密钥,使用默认值有安全风险
nacos.core.auth.plugin.nacos.token.expire.seconds180002.1.0 ~ latest用户登陆临时accessToken的过期时间
nacos.core.auth.enable.userAgentAuthWhitefalse1.4.1 ~ latest是否使用useragent白名单,主要用于适配老版本升级,置为true时有安全风险
nacos.core.auth.server.identity.keyserverIdentity(2.2.1后无默认值)1.4.1 ~ latest用于替换useragent白名单的身份识别key,使用默认值有安全风险
nacos.core.auth.server.identity.valuesecurity(2.2.1后无默认值)1.4.1 ~ latest用于替换useragent白名单的身份识别value,使用默认值有安全风险
nacos.core.auth.default.token.secret.keySecretKey0123456789012345678901234567890123456789012345678901234567891.2.0 ~ 2.0.4nacos.core.auth.plugin.nacos.token.secret.key
nacos.core.auth.default.token.expire.seconds180001.2.0 ~ 2.0.4nacos.core.auth.plugin.nacos.token.expire.seconds
\n

默认控制台登录页

\n

2.2.2版本之前的Nacos默认控制台,无论服务端是否开启鉴权,都会存在一个登录页;这导致很多用户被误导认为Nacos默认是存在鉴权的。在社区安全工程师的建议下,Nacos自2.2.2版本开始,在未开启鉴权时,默认控制台将不需要登录即可访问,同时在控制台中给予提示,提醒用户当前集群未开启鉴权。

\n

在用户开启鉴权后,控制台才需要进行登录访问。 同时针对不同的鉴权插件,提供新的接口方法,用于提示控制台是否开启登录页;同时在2.2.3版本后,Nacos可支持关闭开源控制台,并引导到用户自定义的Nacos控制台,详情可查看Nacos鉴权插件-服务端插件控制台手册-关闭登录功能

\n

服务端如何开启鉴权

\n

非Docker环境

\n

按照官方文档配置启动,默认是不需要登录的,这样会导致配置中心对外直接暴露。而启用鉴权之后,需要在使用用户名和密码登录之后,才能正常使用nacos。

\n

开启鉴权之前,application.properties中的配置信息为:

\n
### If turn on auth system:\nnacos.core.auth.enabled=false\n
\n

开启鉴权之后,application.properties中的配置信息为:

\n
### If turn on auth system:\nnacos.core.auth.system.type=nacos\nnacos.core.auth.enabled=true\n
\n

自定义密钥

\n

开启鉴权之后,你可以自定义用于生成JWT令牌的密钥,application.properties中的配置信息为:

\n
\n

注意:

\n
    \n
  1. 文档中提供的密钥为公开密钥,在实际部署时请更换为其他密钥内容,防止密钥泄漏导致安全风险。
  2. \n
  3. 在2.2.0.1版本后,社区发布版本将移除以文档如下值作为默认值,需要自行填充,否则无法启动节点。
  4. \n
  5. 密钥需要保持节点间一致,长时间不一致可能导致403 invalid token错误。
  6. \n
\n
\n
### The default token(Base64 String):\nnacos.core.auth.default.token.secret.key=SecretKey012345678901234567890123456789012345678901234567890123456789\n\n### 2.1.0 版本后\nnacos.core.auth.plugin.nacos.token.secret.key=SecretKey012345678901234567890123456789012345678901234567890123456789\n
\n

自定义密钥时,推荐将配置项设置为Base64编码的字符串,且原始密钥长度不得低于32字符。例如下面的的例子:

\n
### The default token(Base64 String):\nnacos.core.auth.default.token.secret.key=VGhpc0lzTXlDdXN0b21TZWNyZXRLZXkwMTIzNDU2Nzg=\n\n### 2.1.0 版本后\nnacos.core.auth.plugin.nacos.token.secret.key=VGhpc0lzTXlDdXN0b21TZWNyZXRLZXkwMTIzNDU2Nzg=\n
\n
\n

注意:鉴权开关是修改之后立马生效的,不需要重启服务端。动态修改token.secret.key时,请确保token是有效的,如果修改成无效值,会导致后续无法登录,请求访问异常。

\n
\n

Docker环境

\n

官方镜像

\n

如果使用官方镜像,请在启动docker容器时,添加如下环境变量

\n
NACOS_AUTH_ENABLE=true\n
\n

例如,可以通过如下命令运行开启了鉴权的容器:

\n
docker run --env PREFER_HOST_MODE=hostname --env MODE=standalone --env NACOS_AUTH_ENABLE=true -p 8848:8848 nacos/nacos-server\n
\n

除此之外,还可以添加其他鉴权相关的环境变量信息:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
namedescriptionoption
NACOS_AUTH_ENABLE是否开启权限系统默认:false
NACOS_AUTH_TOKEN_EXPIRE_SECONDStoken 失效时间默认:18000
NACOS_AUTH_TOKENtoken默认:SecretKey012345678901234567890123456789012345678901234567890123456789
NACOS_AUTH_CACHE_ENABLE权限缓存开关 ,开启后权限缓存的更新默认有15秒的延迟默认 : false
\n

然后运行docker-compose构建命令,例如

\n
docker-compose -f example/standalone-derby.yaml up\n
\n

自定义镜像

\n

如果选择自定义镜像,请在构建镜像之前,修改nacos工程中的application.properties文件,

\n

将下面这一行配置信息

\n
nacos.core.auth.enabled=false\n
\n

修改为

\n
nacos.core.auth.system.type=nacos\nnacos.core.auth.enabled=true\n
\n

然后再配置nacos启动命令。

\n

客户端如何进行鉴权

\n

Java SDK鉴权

\n

在构建“Properties”类时,需传入用户名和密码。

\n
properties.put(\"username\",\"${username}\");\nproperties.put(\"password\",\"${password}\");\n
\n

示例代码

\n
try {\n    // Initialize the configuration service, and the console automatically obtains the following parameters through the sample code.\n\tString serverAddr = \"{serverAddr}\";\n\tProperties properties = new Properties();\n\tproperties.put(\"serverAddr\", serverAddr);\n\n    // if need username and password to login\n        properties.put(\"username\",\"nacos\");\n        properties.put(\"password\",\"nacos\");\n\n\tConfigService configService = NacosFactory.createConfigService(properties);\n} catch (NacosException e) {\n    // TODO Auto-generated catch block\n    e.printStackTrace();\n}\n
\n

其他语言的SDK鉴权

\n

待补充

\n

Open-API鉴权

\n

首先需要使用用户名和密码登陆nacos。

\n
curl -X POST '127.0.0.1:8848/nacos/v1/auth/login' -d 'username=nacos&password=nacos'\n
\n

若用户名和密码正确,返回信息如下:

\n
{"accessToken":"eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyOTE2Nn0.2TogGhhr11_vLEjqKko1HJHUJEmsPuCxkur-CfNojDo","tokenTtl":18000,"globalAdmin":true}\n
\n

接下来进行配置信息或服务信息时,应当使用该accessToken鉴权,在url后添加参数accessToken={accessToken},其中{accessToken}为登录时返回的token信息,例如

\n
curl -X GET '127.0.0.1:8848/nacos/v1/cs/configs?accessToken=eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyMzkyM30.O-s2yWfDSUZ7Svd3Vs7jy9tsfDNHs1SuebJB4KlNY8Q&dataId=nacos.example.1&group=nacos_group'\n
\n
curl -X POST 'http://127.0.0.1:8848/nacos/v1/ns/instance?accessToken=eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJuYWNvcyIsImV4cCI6MTYwNTYyMzkyM30.O-s2yWfDSUZ7Svd3Vs7jy9tsfDNHs1SuebJB4KlNY8Q&port=8848&healthy=true&ip=11.11.11.11&weight=1.0&serviceName=nacos.test.3&encoding=GBK&namespaceId=n1'\n
\n

开启Token缓存功能

\n

服务端自2.2.1版本后,默认鉴权插件模块支持token缓存功能,可参见ISSUE #9906

\n
https://github.com/alibaba/nacos/issues/9906\n
\n

背景

\n

无论是客户端SDK还是OpenAPI,在调用login接口获取accessToken之后,携带accessToken访问服务端,服务端解析Token进行鉴权。解析的动作比较耗时,如果想要提升接口的性能,可以考虑开启缓存Token的功能,用字符串比较代替Token解析。

\n

开启方式

\n
nacos.core.auth.plugin.nacos.token.cache.enable=true\n
\n

注意事项

\n

在开启Token缓存功能之前,服务端对每一个携带用户名密码访问login接口的请求都会生成新的token,接口的返回值中的tokenTtl字段跟服务端配置文件中设置的值相等,配置如下:

\n
nacos.core.auth.plugin.nacos.token.expire.seconds=18000\n
\n

在开启Token缓存功能之后,服务端对每一个携带用户名密码访问login接口的请求,会先检查缓存中是否存在该用户名对应的token。若不存在,生成新的Token,插入缓存再返回;若存在,返回该token,此时tokenTtl字段的值为配置文件中设置的值减去该Token在缓存中存留的时长。\n如果Token在缓存中存留的时长超过配置文件设置的值的90%,当login接口收到请求时,尽管缓存中存在该用户名对应的Token,服务端会重新生成Token返回给请求方,并更新缓存。因此,最差情况下,请求方收到的tokenTtl只有配置文件设置的值的10%。

\n

开启服务身份识别功能

\n

开启鉴权功能后,服务端之间的请求也会通过鉴权系统的影响。考虑到服务端之间的通信应该是可信的,因此在1.2~1.4.0版本期间,通过User-Agent中是否包含Nacos-Server来进行判断请求是否来自其他服务端。

\n

但这种实现由于过于简单且固定,导致可能存在安全问题。因此从1.4.1版本开始,Nacos添加服务身份识别功能,用户可以自行配置服务端的Identity,不再使用User-Agent作为服务端请求的判断标准。

\n

开启方式:

\n
### 开启鉴权\nnacos.core.auth.enabled=true\n\n### 关闭使用user-agent判断服务端请求并放行鉴权的功能\nnacos.core.auth.enable.userAgentAuthWhite=false\n\n### 配置自定义身份识别的key(不可为空)和value(不可为空)\nnacos.core.auth.server.identity.key=example\nnacos.core.auth.server.identity.value=example\n
\n

** 注意 ** 所有集群均需要配置相同的server.identity信息,否则可能导致服务端之间数据不一致或无法删除实例等问题。

\n

旧版本升级

\n

考虑到旧版本用户需要升级,可以在升级期间,开启nacos.core.auth.enable.userAgentAuthWhite=true功能,待集群整体升级到1.4.1并稳定运行后,再关闭此功能。

\n", "link": "/zh-cn/docs/v2/guide/user/auth.html", "meta": { "title": "Authorization", diff --git a/zh-cn/docs/v2/guide/user/parameters-check.html b/zh-cn/docs/v2/guide/user/parameters-check.html new file mode 100644 index 00000000000..0a8223eba54 --- /dev/null +++ b/zh-cn/docs/v2/guide/user/parameters-check.html @@ -0,0 +1,222 @@ + + + + + + + + + + + + 参数校验 + + + + + +
Nacos 文档

参数校验

+

2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。

+

在2.3.0版本中,Nacos明确了参数#### 校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据#### 校验规则对客户端向服务端发送的请求进行校验。

+

用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。

+

参数校验开关

+

服务端

+

服务端的参数校验功能默认开启,用户可以通过设置${nacos.home}/conf目录下的application.properties文件中的nacos.core.param.check.enabled值选择开启或者关闭服务端参数校验功能。

+

nacos.core.param.check.enabled=true时开启Nacos服务端参数校验,false关闭服务端参数校验

+

客户端

+

待实现

+

参数校验规则

+

开启参数校验后OpenAPI文档 和 SDK文档中的所有接口中的相关参数都会接受格式校验,现对相关参数以及#### 校验规则进行说明:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
参数描述最大字符长度#### 校验规则
命名空间名称256禁止@#$%^&*,对应正则表达式:[^@#$%^&*]+$
命名空间ID64只允许字母数字下划线以及"-"字符,对应正则表达式:^[\w-]+
配置名称256只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\.]*$
服务名称512禁止中文和@@且禁止以@开头,禁止空白字符,对应正则表达式^(?!@).((?!@@)[^\u4E00-\u9FA5])*$
分组名称128只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\.]*$
集群名称64只允许数字字母和-_,对应正则表达式^[0-9a-zA-Z-_]+$
IP地址128禁止中文字符和空白字符,对应正则表达式为^[^\u4E00-\u9FA5]*$
端口号-取值范围为0~65535
实例元数据1024字段名加字段值的总长度小于1024个字符
+

1. namespaceShowName

+

参数描述

+

命名空间名称

+

校验规则

+

字符长度最大为256,禁止@#$%^&*,对应正则表达式:[^@#$%^&*]+$

+

OpenAPI示例

+ +

校验失败报错信息

+

-超出长度:Param 'namespaceShowName' is illegal, the param length should not exceed 256. +-非法字符:Param 'namespaceShowName' is illegal, illegal characters should not appear in the param.

+

2. namespaceId/tenant/namespace

+

参数描述

+

命名空间ID(租户空间)

+

校验规则

+

字符长度最大为64,只允许字母数字下划线以及"-"字符,对应正则表达式:^[\w-]+

+

OpenAPI示例

+ +

校验失败报错信息

+

-超出长度:Param 'namespaceId/tenant' is illegal, the param length should not exceed 64. +-非法字符:Param 'namespaceId/tenant' is illegal, illegal characters should not appear in the param.

+

3. dataId

+

参数描述

+

配置名称

+

校验规则

+

字符长度最大为256,只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\.]*$

+

OpenAPI示例

+

发布配置

+

Java SDK示例

+

监听配置:public void addListener(String dataId, String group, Listener listener)

+

校验失败报错信息

+

-超出长度:Param 'dataId' is illegal, the param length should not exceed 512. +-非法字符:Param 'dataId' is illegal, illegal characters should not appear in the param.

+

4. service/serviceName

+

参数描述

+

服务名称

+

校验规则

+

字符长度最大为512,禁止中文和@@且禁止以@开头,对应正则表达式^(?!@).((?!@@)[^\u4E00-\u9FA5])*$

+

OpenAPI示例

+

注册实例

+

Java SDK示例

+

注册实例:void registerInstance(String serviceName, String ip, int port) throws NacosException;

+

校验失败报错信息

+

-超出长度:Param 'serviceName' is illegal, the param length should not exceed 512. +-非法字符:Param 'serviceName' is illegal, illegal characters should not appear in the param.

+

5. group/groupName

+

参数描述

+

分组名称

+

校验规则

+

字符长度最大为128,只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\.]*$

+

OpenAPI示例

+

查询实例列表

+

Java SDK示例

+

删除配置:public boolean removeConfig(String dataId, String group) throws NacosException

+

校验失败报错信息

+

-超出长度:Param 'group' is illegal, the param length should not exceed 512. +-非法字符:Param 'group' is illegal, illegal characters should not appear in the param.

+

6. cluster/clusterName

+

参数描述

+

集群名称

+

校验规则

+

字符长度最大为64,只允许数字字母和-_,对应正则表达式^[0-9a-zA-Z-_]+$

+

OpenAPI示例

+

更新实例

+

Java SDK示例

+

获取全部实例:List<Instance> getAllInstances(String serviceName, List<String> clusters) throws NacosException;

+

校验失败报错信息

+

-超出长度:Param 'cluster' is illegal, the param length should not exceed 64. +-非法字符:Param 'cluster' is illegal, illegal characters should not appear in the param.

+

7. ip

+

参数描述

+

IP地址

+

校验规则

+

字符长度最大为128,禁止中文字符,对应正则表达式为^[^\u4E00-\u9FA5]*$

+

OpenAPI示例

+

查询指定服务的实例列表

+

Java SDK示例

+

注销实例:void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;

+

校验失败报错信息

+

-超出长度:Param 'ip' is illegal, the param length should not exceed 128. +-非法字符:Param 'ip' is illegal, illegal characters should not appear in the param.

+

8. port

+

参数描述

+

端口号

+

校验规则

+

取值范围为0~65535

+

OpenAPI示例

+

更新实例

+

Java SDK示例

+

注销实例:void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;

+

校验失败报错信息

+

端口取值超出范围:Param 'port' is illegal, the value should be between 0 and 65535

+

9. metadata

+

参数描述

+

实例元数据

+

校验规则

+

字段名加字段值的总长度小于1024个字符

+

OpenAPI示例

+

更新实例

+

Java SDK示例

+

注册实例:void registerInstance(String serviceName, Instance instance) throws NacosException;

+

校验失败报错信息

+

实例总长度超出范围:Param 'Metadata' is illegal, the param length should not exceed %d.

+
+ + + + + + + + + + + + + diff --git a/zh-cn/docs/v2/guide/user/parameters-check.json b/zh-cn/docs/v2/guide/user/parameters-check.json new file mode 100644 index 00000000000..f804a34da5f --- /dev/null +++ b/zh-cn/docs/v2/guide/user/parameters-check.json @@ -0,0 +1,11 @@ +{ + "filename": "parameters-check.md", + "__html": "

参数校验

\n

2.3.0版本之前的Nacos的参数校验逻辑分散,由各类请求的处理方法单独进行校验,难以更改维护,经常出现参数校验的遗漏,参数校验的规则也没有明确统一;这使得用户使用时经常会因为一些特殊字符导致功能不符合预期或出现漏洞,甚至导致大量推送,导致带宽打满,内存占用过多,导致应用出现故障。

\n

在2.3.0版本中,Nacos明确了参数#### 校验规则,在服务端实现了统一的参数校验逻辑并添加了参数校验层,根据#### 校验规则对客户端向服务端发送的请求进行校验。

\n

用户可以选择开启参数校验功能,开启后Nacos将会对客户端向服务端发送的请求中的部分参数进行参数校验,确保参数的合法性,避免由于错误使用,导致的不符合预期以及性能问题。

\n

参数校验开关

\n

服务端

\n

服务端的参数校验功能默认开启,用户可以通过设置${nacos.home}/conf目录下的application.properties文件中的nacos.core.param.check.enabled值选择开启或者关闭服务端参数校验功能。

\n

nacos.core.param.check.enabled=true时开启Nacos服务端参数校验,false关闭服务端参数校验

\n

客户端

\n

待实现

\n

参数校验规则

\n

开启参数校验后OpenAPI文档 和 SDK文档中的所有接口中的相关参数都会接受格式校验,现对相关参数以及#### 校验规则进行说明:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
参数描述最大字符长度#### 校验规则
命名空间名称256禁止@#$%^&*,对应正则表达式:[^@#$%^&*]+$
命名空间ID64只允许字母数字下划线以及"-"字符,对应正则表达式:^[\\w-]+
配置名称256只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\\.]*$
服务名称512禁止中文和@@且禁止以@开头,禁止空白字符,对应正则表达式^(?!@).((?!@@)[^\\u4E00-\\u9FA5])*$
分组名称128只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\\.]*$
集群名称64只允许数字字母和-_,对应正则表达式^[0-9a-zA-Z-_]+$
IP地址128禁止中文字符和空白字符,对应正则表达式为^[^\\u4E00-\\u9FA5]*$
端口号-取值范围为0~65535
实例元数据1024字段名加字段值的总长度小于1024个字符
\n

1. namespaceShowName

\n

参数描述

\n

命名空间名称

\n

校验规则

\n

字符长度最大为256,禁止@#$%^&*,对应正则表达式:[^@#$%^&*]+$

\n

OpenAPI示例

\n\n

校验失败报错信息

\n

-超出长度:Param 'namespaceShowName' is illegal, the param length should not exceed 256.\n-非法字符:Param 'namespaceShowName' is illegal, illegal characters should not appear in the param.

\n

2. namespaceId/tenant/namespace

\n

参数描述

\n

命名空间ID(租户空间)

\n

校验规则

\n

字符长度最大为64,只允许字母数字下划线以及"-"字符,对应正则表达式:^[\\w-]+

\n

OpenAPI示例

\n\n

校验失败报错信息

\n

-超出长度:Param 'namespaceId/tenant' is illegal, the param length should not exceed 64.\n-非法字符:Param 'namespaceId/tenant' is illegal, illegal characters should not appear in the param.

\n

3. dataId

\n

参数描述

\n

配置名称

\n

校验规则

\n

字符长度最大为256,只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\\.]*$

\n

OpenAPI示例

\n

发布配置

\n

Java SDK示例

\n

监听配置:public void addListener(String dataId, String group, Listener listener)

\n

校验失败报错信息

\n

-超出长度:Param 'dataId' is illegal, the param length should not exceed 512.\n-非法字符:Param 'dataId' is illegal, illegal characters should not appear in the param.

\n

4. service/serviceName

\n

参数描述

\n

服务名称

\n

校验规则

\n

字符长度最大为512,禁止中文和@@且禁止以@开头,对应正则表达式^(?!@).((?!@@)[^\\u4E00-\\u9FA5])*$

\n

OpenAPI示例

\n

注册实例

\n

Java SDK示例

\n

注册实例:void registerInstance(String serviceName, String ip, int port) throws NacosException;

\n

校验失败报错信息

\n

-超出长度:Param 'serviceName' is illegal, the param length should not exceed 512.\n-非法字符:Param 'serviceName' is illegal, illegal characters should not appear in the param.

\n

5. group/groupName

\n

参数描述

\n

分组名称

\n

校验规则

\n

字符长度最大为128,只允许字母数字以及_-.:,对应正则表达式:^[a-zA-Z0-9-_:\\.]*$

\n

OpenAPI示例

\n

查询实例列表

\n

Java SDK示例

\n

删除配置:public boolean removeConfig(String dataId, String group) throws NacosException

\n

校验失败报错信息

\n

-超出长度:Param 'group' is illegal, the param length should not exceed 512.\n-非法字符:Param 'group' is illegal, illegal characters should not appear in the param.

\n

6. cluster/clusterName

\n

参数描述

\n

集群名称

\n

校验规则

\n

字符长度最大为64,只允许数字字母和-_,对应正则表达式^[0-9a-zA-Z-_]+$

\n

OpenAPI示例

\n

更新实例

\n

Java SDK示例

\n

获取全部实例:List<Instance> getAllInstances(String serviceName, List<String> clusters) throws NacosException;

\n

校验失败报错信息

\n

-超出长度:Param 'cluster' is illegal, the param length should not exceed 64.\n-非法字符:Param 'cluster' is illegal, illegal characters should not appear in the param.

\n

7. ip

\n

参数描述

\n

IP地址

\n

校验规则

\n

字符长度最大为128,禁止中文字符,对应正则表达式为^[^\\u4E00-\\u9FA5]*$

\n

OpenAPI示例

\n

查询指定服务的实例列表

\n

Java SDK示例

\n

注销实例:void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;

\n

校验失败报错信息

\n

-超出长度:Param 'ip' is illegal, the param length should not exceed 128.\n-非法字符:Param 'ip' is illegal, illegal characters should not appear in the param.

\n

8. port

\n

参数描述

\n

端口号

\n

校验规则

\n

取值范围为0~65535

\n

OpenAPI示例

\n

更新实例

\n

Java SDK示例

\n

注销实例:void deregisterInstance(String serviceName, String ip, int port, String clusterName) throws NacosException;

\n

校验失败报错信息

\n

端口取值超出范围:Param 'port' is illegal, the value should be between 0 and 65535

\n

9. metadata

\n

参数描述

\n

实例元数据

\n

校验规则

\n

字段名加字段值的总长度小于1024个字符

\n

OpenAPI示例

\n

更新实例

\n

Java SDK示例

\n

注册实例:void registerInstance(String serviceName, Instance instance) throws NacosException;

\n

校验失败报错信息

\n

实例总长度超出范围:Param 'Metadata' is illegal, the param length should not exceed %d.

\n", + "link": "/zh-cn/docs/v2/guide/user/parameters-check.html", + "meta": { + "title": "参数校验", + "keywords": "参数校验,使用规则", + "description": "参数校验", + "date": "2023-10-24" + } +} \ No newline at end of file diff --git a/zh-cn/docs/v2/plugin/auth-plugin.html b/zh-cn/docs/v2/plugin/auth-plugin.html index aa283c8a51d..98ef1d1d366 100644 --- a/zh-cn/docs/v2/plugin/auth-plugin.html +++ b/zh-cn/docs/v2/plugin/auth-plugin.html @@ -126,6 +126,12 @@

服务端插件

boolean 对权限进行验证,在validateIdentity返回为true时调用 + +isLoginEnabled +void +boolean +是否该插件开启开源控制台登录页,返回true时,访问开源控制台将需要通过登录页登录 +

加载服务端插件

diff --git a/zh-cn/docs/v2/plugin/auth-plugin.json b/zh-cn/docs/v2/plugin/auth-plugin.json index 421a44777a5..500c0400512 100644 --- a/zh-cn/docs/v2/plugin/auth-plugin.json +++ b/zh-cn/docs/v2/plugin/auth-plugin.json @@ -1,6 +1,6 @@ { "filename": "auth-plugin.md", - "__html": "

鉴权插件

\n

Nacos从2.1.0版本开始,支持通过SPI的方式注入鉴权相关插件,并在application.properties配置文件中选择某一种插件实现作为实际鉴权服务。本文档会详细介绍如何实现一个鉴权插件和如何使其生效。

\n
\n

注意:\n目前鉴权插件还处于Beta测试的阶段,其API及接口定义可能会随后续版本升级而有所修改,请注意您的插件适用版本。

\n
\n

鉴权插件中的概念

\n

鉴权,通俗的表达就是,验证 是否能够对 某个东西 进行 某种操作 ,因此Nacos在设计鉴权插件时,将鉴权信息主要抽象为身份信息资源操作类型3类主要概念。

\n

身份信息 IdentityContext

\n

身份信息(IdentityContext)是请求发起主体在Nacos鉴权插件中的抽象。由于不同的插件实现,身份信息可能不同,较为灵活;比如用户名和密码是一种身份信息,accessToken又是另一种身份信息。因此身份信息(IdentityContext)并没有限制具体的个数和名字,插件实现可以自定义任意个数和身份关键字,Nacos将会从请求中自动获取插件实现定义的身份关键字及其对应的值注入到身份信息(IdentityContext)中,供插件使用。

\n

其中必定会包含的内容有:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名描述
remote_ip请求来源ip
\n

资源 Resource

\n

资源(Resource)是请求所操作对象在Nacos鉴权插件中的抽象。它主要由Nacos来定义,具体可以是某个配置,某个服务,或者某个分组。

\n

资源(Resource)主要由以下内容组成:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名描述
namespaceId请求资源的命名空间ID,部分接口可能没有该值
group请求资源的分组名,部分接口可能没有该值
name请求资源的资源名,如服务名或配置的dataId,部分接口可能是定义的特殊值,如nacos/admin
type请求资源的类型,可能取值为SignType中的枚举值,主要表示该资源所相关的模块
properties请求资源的扩展配置,不属于上述的资源相关信息,会被放如properties中,比如Grpc请求的Request名称或@Secured注解上的tags等
\n

操作类型 Action

\n

操作类型(Action)是请求操作在Nacos鉴权插件中的抽象,主要有读操作R和写操作W,详情查看ActionTypes枚举。

\n

服务端插件

\n

开发Nacos服务端鉴权插件,首先需要依赖鉴权插件的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-auth-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本

\n

随后实现com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService接口, 并将您的实现添加到SPI的services当中。

\n

接口中需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
getAuthServiceNamevoidString插件的名称,当名字相同时,后加载的插件会覆盖先加载的插件。
identityNamesvoidCollection插件的身份信息关键字,Nacos会从请求中获取以这些关键字为key的参数,并注入到IdentityContext中。
enableAuthActionTypes,SignTypeboolean在调用validateIdentityvalidateAuthority前调用,插件可自行判断是否对此类型的操作或此类型的模块进行鉴权。
validateIdentityIdentityContext, Resourceboolean对身份信息进行验证,在validateAuthority前调用
validateAuthorityIdentityContext, Permissionboolean对权限进行验证,在validateIdentity返回为true时调用
\n

加载服务端插件

\n

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

\n

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
### 所启用的Nacos的鉴权插件的名称,与`com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService`的`getAuthServiceName`返回值对应\nnacos.core.auth.system.type=${authServiceName}\n\n### 开启鉴权功能\nnacos.core.auth.enabled=true\n
\n

随后重启nacos集群,在有请求访问到nacos节点后,可以从${nacos-server.path}/logs/nacos.log中看到如下日志:

\n
[AuthPluginManager] Load AuthPluginService(xxxx) AuthServiceName(xxx) successfully.\n
\n

使用Nacos自带的鉴权插件

\n

Nacos默认带有一个鉴权的简易实现,主要是为防止业务错用的弱鉴权体系,不是防止恶意攻击的强鉴权体系。开启和使用方式请查看文档用户指南-权限认证.

\n

客户端插件

\n

Nacos的客户端鉴权插件主要工作为将鉴权相关的身份信息,注入到请求中,让每个请求都能够被对应的服务端鉴权插件识别。

\n

在Nacos的Java客户端默认自带两个实现:

\n
    \n
  • 使用usernamepasswordaccessToken的简易鉴权实现;
  • \n
  • 使用accessKeysecretKey的阿里云鉴权实现;
  • \n
\n

Nacos简易鉴权实现

\n

当构造客户端实例时传入的properties中带有usernamepassword时,客户端会使用简易鉴权实现插件注入身份信息;\n如:

\n
Properties properties = new Properties();\nproperties.setProperty(PropertyKeyConst.SERVER_ADDR, \"localhost:8848\");\nproperties.setProperty(PropertyKeyConst.USERNAME, \"nacos\");\nproperties.setProperty(PropertyKeyConst.PASSWORD, \"nacos\");\nNamingFactory.createNamingService(properties);\nConfigFactory.createConfigService(properties);\n
\n

该插件会异步地通过usernamepassword进行登录,获取登录成功后的accessToken,并将accessToken注入到所有客户端请求中,开发者可以根据accessToken在实现的服务端插件中进行身份验证及后续的权限验证。

\n

阿里云鉴权实现

\n

当properties中带有accessKeysecretKey时,则会使用阿里云鉴权实现注入身份信息,如:

\n
Properties properties = new Properties();\nproperties.setProperty(PropertyKeyConst.SERVER_ADDR, \"localhost:8848\");\nproperties.setProperty(PropertyKeyConst.ACCESS_KEY, \"nacos\");\nproperties.setProperty(PropertyKeyConst.SECRET_KEY, \"nacos\");\nNamingFactory.createNamingService(properties);\nConfigFactory.createConfigService(properties);\n
\n

该插件会根据accessKeysecretKey以及请求的资源内容,自动生成对应的请求签名,并注入到请求中,根据资源类型的不同,请求中的身份信息关键字可能不同:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
类型身份关键字描述
NamingServiceakaccessKey
NamingServicesignature注册中心模块的签名信息
NamingServicedata签名数据,主要是时间戳
ConfigServiceSpas-AccessKeyaccessKey
ConfigServiceSpas-Signature配置中心模块的签名信息
ConfigServiceTimestamp请求的时间戳
ConfigServiceSpas-SecurityToken临时token(启用阿里云STS功能时使用)
\n

开发者可以根据以上信息,在实现的服务端插件中进行身份验证及后续的权限验证。

\n

其他自定义插件

\n

考虑到开发者的鉴权插件可能有自定义的身份信息关键字,因此Nacos的Java客户端同样可以使用SPI方式注入对应的插件实现。

\n

开发Nacos客户端鉴权插件,首先需要依赖鉴权插件的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-auth-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本

\n

随后实现com.alibaba.nacos.plugin.auth.spi.client.ClientAuthService接口, 并将您的实现添加到SPI的services当中。

\n

接口中需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
setServerListList,Nacos服务端地址列表void初始化时会调用此接口注入Nacos的服务列表,方便插件访问nacos服务端,如调用登录接口等
setNacosRestTemplateNacosRestTemplate,Nacos的http客户端void初始化时会调用此接口注入Nacos的http客户端,方便插件访问nacos服务端,如调用登录接口等
loginProperties,即初始化Nacos客户端时传入的参数boolean登录接口,主要执行的是身份信息的转换工作,如usernamepassword转换为accessToken
getLoginIdentityContextResourceIdentityContext获取经过登录接口转换后的身份信息,客户端会将该返回对象的内容全部注入到请求中
\n

您也可以选择继承com.alibaba.nacos.plugin.auth.spi.client.AbstractClientAuthService,该父类默认实现了setServerListsetNacosRestTemplate

\n

将开发完成的客户端插件打包成jar/zip,放入到您应用的classpath中即可自动生效。

\n

其他语言客户端鉴权插件

\n

待社区贡献。

\n", + "__html": "

鉴权插件

\n

Nacos从2.1.0版本开始,支持通过SPI的方式注入鉴权相关插件,并在application.properties配置文件中选择某一种插件实现作为实际鉴权服务。本文档会详细介绍如何实现一个鉴权插件和如何使其生效。

\n
\n

注意:\n目前鉴权插件还处于Beta测试的阶段,其API及接口定义可能会随后续版本升级而有所修改,请注意您的插件适用版本。

\n
\n

鉴权插件中的概念

\n

鉴权,通俗的表达就是,验证 是否能够对 某个东西 进行 某种操作 ,因此Nacos在设计鉴权插件时,将鉴权信息主要抽象为身份信息资源操作类型3类主要概念。

\n

身份信息 IdentityContext

\n

身份信息(IdentityContext)是请求发起主体在Nacos鉴权插件中的抽象。由于不同的插件实现,身份信息可能不同,较为灵活;比如用户名和密码是一种身份信息,accessToken又是另一种身份信息。因此身份信息(IdentityContext)并没有限制具体的个数和名字,插件实现可以自定义任意个数和身份关键字,Nacos将会从请求中自动获取插件实现定义的身份关键字及其对应的值注入到身份信息(IdentityContext)中,供插件使用。

\n

其中必定会包含的内容有:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名描述
remote_ip请求来源ip
\n

资源 Resource

\n

资源(Resource)是请求所操作对象在Nacos鉴权插件中的抽象。它主要由Nacos来定义,具体可以是某个配置,某个服务,或者某个分组。

\n

资源(Resource)主要由以下内容组成:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名描述
namespaceId请求资源的命名空间ID,部分接口可能没有该值
group请求资源的分组名,部分接口可能没有该值
name请求资源的资源名,如服务名或配置的dataId,部分接口可能是定义的特殊值,如nacos/admin
type请求资源的类型,可能取值为SignType中的枚举值,主要表示该资源所相关的模块
properties请求资源的扩展配置,不属于上述的资源相关信息,会被放如properties中,比如Grpc请求的Request名称或@Secured注解上的tags等
\n

操作类型 Action

\n

操作类型(Action)是请求操作在Nacos鉴权插件中的抽象,主要有读操作R和写操作W,详情查看ActionTypes枚举。

\n

服务端插件

\n

开发Nacos服务端鉴权插件,首先需要依赖鉴权插件的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-auth-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本

\n

随后实现com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService接口, 并将您的实现添加到SPI的services当中。

\n

接口中需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
getAuthServiceNamevoidString插件的名称,当名字相同时,后加载的插件会覆盖先加载的插件。
identityNamesvoidCollection插件的身份信息关键字,Nacos会从请求中获取以这些关键字为key的参数,并注入到IdentityContext中。
enableAuthActionTypes,SignTypeboolean在调用validateIdentityvalidateAuthority前调用,插件可自行判断是否对此类型的操作或此类型的模块进行鉴权。
validateIdentityIdentityContext, Resourceboolean对身份信息进行验证,在validateAuthority前调用
validateAuthorityIdentityContext, Permissionboolean对权限进行验证,在validateIdentity返回为true时调用
isLoginEnabledvoidboolean是否该插件开启开源控制台登录页,返回true时,访问开源控制台将需要通过登录页登录
\n

加载服务端插件

\n

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

\n

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
### 所启用的Nacos的鉴权插件的名称,与`com.alibaba.nacos.plugin.auth.spi.server.AuthPluginService`的`getAuthServiceName`返回值对应\nnacos.core.auth.system.type=${authServiceName}\n\n### 开启鉴权功能\nnacos.core.auth.enabled=true\n
\n

随后重启nacos集群,在有请求访问到nacos节点后,可以从${nacos-server.path}/logs/nacos.log中看到如下日志:

\n
[AuthPluginManager] Load AuthPluginService(xxxx) AuthServiceName(xxx) successfully.\n
\n

使用Nacos自带的鉴权插件

\n

Nacos默认带有一个鉴权的简易实现,主要是为防止业务错用的弱鉴权体系,不是防止恶意攻击的强鉴权体系。开启和使用方式请查看文档用户指南-权限认证.

\n

客户端插件

\n

Nacos的客户端鉴权插件主要工作为将鉴权相关的身份信息,注入到请求中,让每个请求都能够被对应的服务端鉴权插件识别。

\n

在Nacos的Java客户端默认自带两个实现:

\n
    \n
  • 使用usernamepasswordaccessToken的简易鉴权实现;
  • \n
  • 使用accessKeysecretKey的阿里云鉴权实现;
  • \n
\n

Nacos简易鉴权实现

\n

当构造客户端实例时传入的properties中带有usernamepassword时,客户端会使用简易鉴权实现插件注入身份信息;\n如:

\n
Properties properties = new Properties();\nproperties.setProperty(PropertyKeyConst.SERVER_ADDR, \"localhost:8848\");\nproperties.setProperty(PropertyKeyConst.USERNAME, \"nacos\");\nproperties.setProperty(PropertyKeyConst.PASSWORD, \"nacos\");\nNamingFactory.createNamingService(properties);\nConfigFactory.createConfigService(properties);\n
\n

该插件会异步地通过usernamepassword进行登录,获取登录成功后的accessToken,并将accessToken注入到所有客户端请求中,开发者可以根据accessToken在实现的服务端插件中进行身份验证及后续的权限验证。

\n

阿里云鉴权实现

\n

当properties中带有accessKeysecretKey时,则会使用阿里云鉴权实现注入身份信息,如:

\n
Properties properties = new Properties();\nproperties.setProperty(PropertyKeyConst.SERVER_ADDR, \"localhost:8848\");\nproperties.setProperty(PropertyKeyConst.ACCESS_KEY, \"nacos\");\nproperties.setProperty(PropertyKeyConst.SECRET_KEY, \"nacos\");\nNamingFactory.createNamingService(properties);\nConfigFactory.createConfigService(properties);\n
\n

该插件会根据accessKeysecretKey以及请求的资源内容,自动生成对应的请求签名,并注入到请求中,根据资源类型的不同,请求中的身份信息关键字可能不同:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
类型身份关键字描述
NamingServiceakaccessKey
NamingServicesignature注册中心模块的签名信息
NamingServicedata签名数据,主要是时间戳
ConfigServiceSpas-AccessKeyaccessKey
ConfigServiceSpas-Signature配置中心模块的签名信息
ConfigServiceTimestamp请求的时间戳
ConfigServiceSpas-SecurityToken临时token(启用阿里云STS功能时使用)
\n

开发者可以根据以上信息,在实现的服务端插件中进行身份验证及后续的权限验证。

\n

其他自定义插件

\n

考虑到开发者的鉴权插件可能有自定义的身份信息关键字,因此Nacos的Java客户端同样可以使用SPI方式注入对应的插件实现。

\n

开发Nacos客户端鉴权插件,首先需要依赖鉴权插件的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-auth-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本

\n

随后实现com.alibaba.nacos.plugin.auth.spi.client.ClientAuthService接口, 并将您的实现添加到SPI的services当中。

\n

接口中需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
setServerListList,Nacos服务端地址列表void初始化时会调用此接口注入Nacos的服务列表,方便插件访问nacos服务端,如调用登录接口等
setNacosRestTemplateNacosRestTemplate,Nacos的http客户端void初始化时会调用此接口注入Nacos的http客户端,方便插件访问nacos服务端,如调用登录接口等
loginProperties,即初始化Nacos客户端时传入的参数boolean登录接口,主要执行的是身份信息的转换工作,如usernamepassword转换为accessToken
getLoginIdentityContextResourceIdentityContext获取经过登录接口转换后的身份信息,客户端会将该返回对象的内容全部注入到请求中
\n

您也可以选择继承com.alibaba.nacos.plugin.auth.spi.client.AbstractClientAuthService,该父类默认实现了setServerListsetNacosRestTemplate

\n

将开发完成的客户端插件打包成jar/zip,放入到您应用的classpath中即可自动生效。

\n

其他语言客户端鉴权插件

\n

待社区贡献。

\n", "link": "/zh-cn/docs/v2/plugin/auth-plugin.html", "meta": { "title": "鉴权插件", diff --git a/zh-cn/docs/v2/plugin/config-change-plugin.html b/zh-cn/docs/v2/plugin/config-change-plugin.html new file mode 100644 index 00000000000..1dbd736df14 --- /dev/null +++ b/zh-cn/docs/v2/plugin/config-change-plugin.html @@ -0,0 +1,266 @@ + + + + + + + + + + + + 配置变更 + + + + + +
Nacos 文档

配置变更插件

+

社区中一直以来都希望Nacos配置中心能在配置发生变更时,通知一些特定系统,用于发送记录、警告等审计功能。在2.3.0版本前,只能通过模拟Nacos客户端订阅配置的方式,对核心配置的变更操作进行订阅,在收到变更通知后,进行发送记录、警告等功能的执行。

+

这种实现方式有几个比较大的问题,第一是监听的配置需要逐个添加,难以对所有配置变更进行获取;第二是只能在配置变更后执行功能逻辑,无法做到前置的操作,如格式校验,白名单校验等。

+

因此Nacos在2.3.0版本后,支持通过SPI注入配置变更插件,允许用户通过自定义插件的方式,对配置变更前,和变更完成后分别执行一些自定义逻辑,如格式校验,白名单校验,webhook等。

+

配置变更插件中的概念

+

Nacos的配置变更插件,参考了面向切面编程AOP的设计思想,将配置的变更操作(如新增,更新,删除)作为切点(PointCut),并在这些切点前(Before)后(After)织入插件。

+

配置变更切点(ConfigChangePointCutTypes)

+

Nacos根据配置变更的行为和来源,将配置变更操作在com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes中定位为了数个配置变更切点(ConfigChangePointCutTypes),具体内容如下:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
切点名称描述起始版本
PUBLISH_BY_HTTP配置通过HTTP接口进行发布,包含了创建配置及修改配置2.3.0
PUBLISH_BY_RPC配置通过GRPC接口进行发布,包含了创建配置及修改配置2.3.0
REMOVE_BY_HTTP配置通过HTTP接口进行删除2.3.0
REMOVE_BY_RPC配置通过GRPC接口进行删除2.3.0
IMPORT_BY_HTTP配置通过HTTP接口进行导入2.3.0
REMOVE_BATCH_HTTP配置通过HTTP接口进行批量删除2.3.0
+

配置变更织入类型(ConfigChangeExecuteTypes)

+

Nacos的配置变更插件需要在配置变更切点之前或之后进行执行,即需要选择配置变更织入类型(ConfigChangeExecuteTypes),定义在com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes中,具体内容如下:

+ + + + + + + + + + + + + + + + + + + + +
织入类型描述起始版本
EXECUTE_BEFORE_TYPE插件实现在配置变更切点执行2.3.0
EXECUTE_AFTER_TYPE插件实现在配置变更切点执行2.3.0
+

插件开发

+

开发Nacos服务端配置变更插件,首先需要依赖配置变更插件的的相关API

+
        <dependency>
+            <groupId>com.alibaba.nacos</groupId>
+            <artifactId>nacos-config-plugin</artifactId>
+            <version>${project.version}</version>
+        </dependency>
+
+

${project.version} 为您开发插件所对应的Nacos版本,2.3.0及以上。

+

随后实现com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService接口,该接口需要实现的方法如下:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
方法名入参内容返回内容描述
getServiceTypevoidString插件的名称,用于区分不同类型的插件实现
getOrdervoidint插件的执行顺序,配置变更插件采用链式插件设计,多个插件实现时会按照顺序执行,getOrder越小,顺序越靠前
executeTypevoidConfigChangeExecuteTypes插件实现的配置变更织入类型
pointcutMethodNamesvoidConfigChangePointCutTypes[]插件实现织入的配置变更切点
executeConfigChangeRequest,ConfigChangeResponsevoid实际插件执行的逻辑
+

其中ConfigChangeRequestConfigChangeResponse分别为执行逻辑时传入的内容及执行结果,

+

ConfigChangeRequest的具体内容如下:

+ + + + + + + + + + + + + + + + + + + + +
字段名字段类型描述
requestTypeConfigChangePointCutTypes本次配置变更的切点类型
requestArgsHashMap<String, Object>本次配置变更的实际参数,主要包含有namespace,group,dataId,content等内容,不同的切点类型参数存在不同
+

ConfigChangeResponse的具体内容如下:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
字段名字段类型描述
responseTypeConfigChangePointCutTypes本次配置变更的切点类型
isSuccessboolean执行是否成功,当返回值为false时,将会拦截本次配置变更,并直接返回失败的结果
retValObject返回内容,预留字段,暂未启用
msgString执行结果信息,在isSuccessfalse时获取,用于返回给客户端的信息
argsObject[]配置变更操作的执行参数,在EXECUTE_BEFORE_TYPE的插件类型时生效,可用于修改实际执行的配置变更时的内容,如将content中的某些内容修改为其他值
+

加载插件

+

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

+

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

+
### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应
+nacos.core.config.plugin.${configChangePluginName}.enabled=true
+
+

随后重启nacos集群,启动完成后,可在${nacos-server.path}/logs/nacos.log日志中看到如下日志。

+
[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully.
+
+

插件自定义参数传递

+

部分插件可能希望通过配置文件设置一些参数,自定义插件可以通过修改${nacos-server.path}/conf/application.properties中的以下配置完成:

+
### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应
+nacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue}
+
+

之后能在ConfigChangeRequest中,通过下述方法获取该参数:

+
final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES);
+final String ${propertyKey} = properties.getProperty("${propertyKey}");
+
+

插件DEMO实现

+

nacos-group/nacos-plugin中,有一个demo的配置变更插件的实现,该demo插件实现了对配置内容格式的校验、配置导入名称白名单的校验、以及变更后回调webhook。打包成jar/zip,放置到nacos服务端的classpath中,在${nacos-server.path}/conf/application.properties中的加入以下配置:

+
# webhook
+#nacos.core.config.plugin.webhook.enabled=true
+# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html
+#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=***
+# The content push max capacity ,byte
+#nacos.core.config.plugin.webhook.contentMaxCapacity=102400
+
+# whitelist
+#nacos.core.config.plugin.whitelist.enabled=true
+# The import file suffixs
+#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html
+
+# fileformatcheck,which validate the import file of type and content
+#nacos.core.config.plugin.fileformatcheck.enabled=true
+
+
+ + + + + + + + + + + + + diff --git a/zh-cn/docs/v2/plugin/config-change-plugin.json b/zh-cn/docs/v2/plugin/config-change-plugin.json new file mode 100644 index 00000000000..e8712d1620a --- /dev/null +++ b/zh-cn/docs/v2/plugin/config-change-plugin.json @@ -0,0 +1,10 @@ +{ + "filename": "config-change-plugin.md", + "__html": "

配置变更插件

\n

社区中一直以来都希望Nacos配置中心能在配置发生变更时,通知一些特定系统,用于发送记录、警告等审计功能。在2.3.0版本前,只能通过模拟Nacos客户端订阅配置的方式,对核心配置的变更操作进行订阅,在收到变更通知后,进行发送记录、警告等功能的执行。

\n

这种实现方式有几个比较大的问题,第一是监听的配置需要逐个添加,难以对所有配置变更进行获取;第二是只能在配置变更后执行功能逻辑,无法做到前置的操作,如格式校验,白名单校验等。

\n

因此Nacos在2.3.0版本后,支持通过SPI注入配置变更插件,允许用户通过自定义插件的方式,对配置变更前,和变更完成后分别执行一些自定义逻辑,如格式校验,白名单校验,webhook等。

\n

配置变更插件中的概念

\n

Nacos的配置变更插件,参考了面向切面编程AOP的设计思想,将配置的变更操作(如新增,更新,删除)作为切点(PointCut),并在这些切点前(Before)后(After)织入插件。

\n

配置变更切点(ConfigChangePointCutTypes)

\n

Nacos根据配置变更的行为和来源,将配置变更操作在com.alibaba.nacos.plugin.config.constants.ConfigChangePointCutTypes中定位为了数个配置变更切点(ConfigChangePointCutTypes),具体内容如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
切点名称描述起始版本
PUBLISH_BY_HTTP配置通过HTTP接口进行发布,包含了创建配置及修改配置2.3.0
PUBLISH_BY_RPC配置通过GRPC接口进行发布,包含了创建配置及修改配置2.3.0
REMOVE_BY_HTTP配置通过HTTP接口进行删除2.3.0
REMOVE_BY_RPC配置通过GRPC接口进行删除2.3.0
IMPORT_BY_HTTP配置通过HTTP接口进行导入2.3.0
REMOVE_BATCH_HTTP配置通过HTTP接口进行批量删除2.3.0
\n

配置变更织入类型(ConfigChangeExecuteTypes)

\n

Nacos的配置变更插件需要在配置变更切点之前或之后进行执行,即需要选择配置变更织入类型(ConfigChangeExecuteTypes),定义在com.alibaba.nacos.plugin.config.constants.ConfigChangeExecuteTypes中,具体内容如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
织入类型描述起始版本
EXECUTE_BEFORE_TYPE插件实现在配置变更切点执行2.3.0
EXECUTE_AFTER_TYPE插件实现在配置变更切点执行2.3.0
\n

插件开发

\n

开发Nacos服务端配置变更插件,首先需要依赖配置变更插件的的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-config-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本,2.3.0及以上。

\n

随后实现com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService接口,该接口需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
getServiceTypevoidString插件的名称,用于区分不同类型的插件实现
getOrdervoidint插件的执行顺序,配置变更插件采用链式插件设计,多个插件实现时会按照顺序执行,getOrder越小,顺序越靠前
executeTypevoidConfigChangeExecuteTypes插件实现的配置变更织入类型
pointcutMethodNamesvoidConfigChangePointCutTypes[]插件实现织入的配置变更切点
executeConfigChangeRequest,ConfigChangeResponsevoid实际插件执行的逻辑
\n

其中ConfigChangeRequestConfigChangeResponse分别为执行逻辑时传入的内容及执行结果,

\n

ConfigChangeRequest的具体内容如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名字段类型描述
requestTypeConfigChangePointCutTypes本次配置变更的切点类型
requestArgsHashMap<String, Object>本次配置变更的实际参数,主要包含有namespace,group,dataId,content等内容,不同的切点类型参数存在不同
\n

ConfigChangeResponse的具体内容如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名字段类型描述
responseTypeConfigChangePointCutTypes本次配置变更的切点类型
isSuccessboolean执行是否成功,当返回值为false时,将会拦截本次配置变更,并直接返回失败的结果
retValObject返回内容,预留字段,暂未启用
msgString执行结果信息,在isSuccessfalse时获取,用于返回给客户端的信息
argsObject[]配置变更操作的执行参数,在EXECUTE_BEFORE_TYPE的插件类型时生效,可用于修改实际执行的配置变更时的内容,如将content中的某些内容修改为其他值
\n

加载插件

\n

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

\n

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应\nnacos.core.config.plugin.${configChangePluginName}.enabled=true\n
\n

随后重启nacos集群,启动完成后,可在${nacos-server.path}/logs/nacos.log日志中看到如下日志。

\n
[ConfigChangePluginManager] Load ${className}(${classFullName}) ConfigChangeServiceName(${configChangePluginName}) successfully.\n
\n

插件自定义参数传递

\n

部分插件可能希望通过配置文件设置一些参数,自定义插件可以通过修改${nacos-server.path}/conf/application.properties中的以下配置完成:

\n
### 所启用的Nacos的配置变更插件的名称,与com.alibaba.nacos.plugin.config.spi.ConfigChangePluginService 的getServiceType 返回值对应\nnacos.core.config.plugin.${configChangePluginName}.${propertyKey}=${propertyValue}\n
\n

之后能在ConfigChangeRequest中,通过下述方法获取该参数:

\n
final Properties properties = (Properties) configChangeRequest.getArg(ConfigChangeConstants.PLUGIN_PROPERTIES);\nfinal String ${propertyKey} = properties.getProperty(\"${propertyKey}\");\n
\n

插件DEMO实现

\n

nacos-group/nacos-plugin中,有一个demo的配置变更插件的实现,该demo插件实现了对配置内容格式的校验、配置导入名称白名单的校验、以及变更后回调webhook。打包成jar/zip,放置到nacos服务端的classpath中,在${nacos-server.path}/conf/application.properties中的加入以下配置:

\n
# webhook\n#nacos.core.config.plugin.webhook.enabled=true\n# It is recommended to use EB https://help.aliyun.com/document_detail/413974.html\n#nacos.core.config.plugin.webhook.url=http://${webhookIp}:${webhookPort}/${webhookUri}?token=***\n# The content push max capacity ,byte\n#nacos.core.config.plugin.webhook.contentMaxCapacity=102400\n\n# whitelist\n#nacos.core.config.plugin.whitelist.enabled=true\n# The import file suffixs\n#nacos.core.config.plugin.whitelist.suffixs=xml,text,properties,yaml,html\n\n# fileformatcheck,which validate the import file of type and content\n#nacos.core.config.plugin.fileformatcheck.enabled=true\n
\n", + "link": "/zh-cn/docs/v2/plugin/config-change-plugin.html", + "meta": { + "title": "配置变更", + "keywords": "配置变更,配置审计,配置格式校验,webhook", + "description": "Nacos 支持配置变更插件,可支持配置审计、配置格式校验、及变更后webhook回调" + } +} \ No newline at end of file diff --git a/zh-cn/docs/v2/plugin/control-plugin.html b/zh-cn/docs/v2/plugin/control-plugin.html new file mode 100644 index 00000000000..b4ada1b0cdb --- /dev/null +++ b/zh-cn/docs/v2/plugin/control-plugin.html @@ -0,0 +1,439 @@ + + + + + + + + + + + + 反脆弱 + + + + + +
Nacos 文档

反脆弱插件

+

Nacos 从2.3.0版本开始,支持通过SPI的方式注入反脆弱相关插件,并在application.properties配置文件中选择某一种插件实现作为实际反脆弱能力。本文档会详细介绍如何实现一个反脆弱插件和如何使其生效。

+

反脆弱插件中的概念

+

反脆弱是对访问服务端的某种资源频率和次数达到一定程度时进行的限制访问的策略,用于保护服务端在高压情况下能快速拒绝请求,防止过多的资源访问导致服务端资源耗尽引起的大面积不可用;Nacos反脆弱插件,将信息主要抽象为监控点反脆弱规则

+

监控点(ControlPoint)

+

监控点对应的请求服务端时所占用的资源的映射,目前主要针对的是连接(Connection)以及每秒请求数(TPS)

+
    +
  • 连接(Connection)监控点主要监控Nacos 服务端中使用Nacos2.X客户端的长连接数量以及使用Nacos1.X客户端的配置长轮询数量,两者独立监控。
  • +
  • 每秒请求数(TPS)监控点主要是监控Nacos 服务端中各核心接口被访问的频率,同类型的操作接口会被视为相同的监控点,如注册服务的v1接口和v2接口,具体的每秒请求数(TPS)监控点可查看本文档下文监控点名称
  • +
+

反脆弱规则(ControlRule)

+

反脆弱规则是针对每个监控点而执行的不同的限制规则,具体又分为连接数规则(ConnectionControlRule)每秒请求数规则(TpsControlRule)

+

连接数规则(ConnectionControlRule)主要包含如下内容:

+ + + + + + + + + + + + + + + + + + + + +
字段名类型描述
countLimitint连接数总数限制,默认为-1,不限制
monitorIpListSettrace监控的Ip列表,用于详细观察对应ip的连接做了哪些操作,添加后,对应ip的连接请求会被详细打印在remote-digest.log日志中
+

每秒请求数规则(TpsControlRule)主要包含如下内容:

+ + + + + + + + + + + + + + + + + + + + +
字段名类型描述
pointNameString规则所对应的监控点名称
pointRuleRuleDetail规则内容的具体细节
+

其中RuleDetail又包含如下内容:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
字段名类型描述
ruleNameString规则的名称,区别于监控点名称,同一个监控点可以有多个规则名
maxCountintTPS总数限制,默认为-1,不限制
periodTimeUnit规则生效的周期,即统计到秒级/分钟级等,默认TimeUnit.SECONDS秒级
monitorTypeString监控类型,取值为monitorintercept,对应为监控模式(只统计和打印tps,即使触发规则也不拦截)和拦截模式
+

插件开发

+

开发Nacos服务端反脆弱插件,首先需要依赖反脆弱插件的相关API

+
        <dependency>
+            <groupId>com.alibaba.nacos</groupId>
+            <artifactId>nacos-control-plugin</artifactId>
+            <version>${project.version}</version>
+        </dependency>
+
+

${project.version} 为您开发插件所对应的Nacos版本,2.3.0及以上。

+

随后继承com.alibaba.nacos.plugin.control.connection.ConnectionControlManager抽象类和com.alibaba.nacos.plugin.control.tps.TpsControlManager抽象类,实现缺失的方法;然后实现com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 接口,创建上述实现的两个抽象类;最后将您的实现添加到SPI的services当中。

+

com.alibaba.nacos.plugin.control.connection.ConnectionControlManager需要实现的方法如下:

+ + + + + + + + + + + + + + + + + + + + + + + +
方法名入参内容返回内容描述
applyConnectionLimitRuleConnectionControlRulevoid应用新的连接数规则
checkConnectionCheckRequestConnectionCheckResponse判断是否命中连接数规则,如果ConnectionCheckResponse中的sucess为false,将会拒绝新连接的建立
+

com.alibaba.nacos.plugin.control.tps.TpsControlManager需要实现的方法如下:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
方法名入参内容返回内容描述
registerTpsPointStringvoid注册TPS监控点,Nacos服务会在启动时向插件注册当前的TPS监控买点,入参为TPS监控点的名字,具体的监控点名称可查看本文档下文监控点名称;插件需要在方法内,维护一个用于记录TPS和规则内容的TpsBarrier,详情查看自定义TPS时间窗口
applyTpsRuleString,TpsControlRulevoid应用新的TPS规则,根据TPS监控点名称关联及更新。
checkTpsCheckRequestTpsCheckResponse判断是否命中TPS规则,如果TpsCheckResponse中的sucess为false,将会拒绝新的请求。
+

com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 需要实现的方法如下:

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
方法名入参内容返回内容描述
getNamevoidString插件的名称,和配置文件中指定的类型进行匹配,使用命中的的插件。
buildConnectionControlManagervoidConnectionControlManager创建插件对应的ConnectionControlManager实现,为null时会使用no limit实现。
buildTpsControlManagervoidTpsControlManager创建插件对应的TpsControlManager实现,为null时会使用no limit实现。
+

加载插件

+

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

+

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

+
### 所启用的Nacos的反脆弱插件的名称,与com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 的getName 返回值对应
+nacos.plugin.control.manager.type=${controlPluginName}
+
+

随后重启nacos集群,启动完成后,可以从${nacos-server.path}/logs/plugin-control.log中看到如下日志:

+
Found control manager plugin of name=${controlPluginName}
+
+Build connection control manager, class=${your plugin ConnectionControlManager class}
+
+Build tps control manager, class=${your plugin TpsControlManager class}
+
+

使用Nacos自带的反脆弱插件

+

Nacos2.3.0版本起,自带一个简易的反脆弱插件实现,可以做到对Nacos服务端的连接数及指定接口TPS进行限制。

+

启用Nacos自带的反脆弱插件

+

需要修改${nacos-server.path}/conf/application.properties中的以下配置

+
nacos.plugin.control.manager.type=nacos
+
+

设置反脆弱规则

+

可通过创建和修改反脆弱规则文件的方式,修改和设置反脆弱规则,默认反脆弱插件的规则是通过json格式定义的;例如想要设置连接限制为100,可执行如下操作:

+
mkdir -p ${nacos.home}/data/connection/
+echo '{"countLimit": 100}' > ${nacos.home}/data/connection/limitRule
+
+

随后重启Nacos节点即可。

+

又例如想要设置配置查询接口的TPS为100,可执行如下操作:

+
mkdir -p ${nacos.home}/data/tps/
+# ConfigQuery 为配置查询接口的监控点名称(pointName)
+echo '{"pointName":"ConfigQuery","pointRule":{"maxCount":100,"monitorType":"intercept"}}' > ${nacos.home}/data/tps/ConfigQuery 
+
+

随后重启Nacos节点即可。

+

其他更多反脆弱规则,以及具体的反脆弱监控点名称,请查看下文监控点名称

+

反脆弱规则存储位置

+

Nacos自带的简易反脆弱插件实现,Nacos服务端会通过本地文件系统,存储和读取反脆弱规则,默认所在目录的为${nacos.home}/data/connection${nacos.home}/data/tps中,如果想将规则文件更换目录存储,可以在${nacos-server.path}/conf/application.properties中修改以下配置:

+
nacos.plugin.control.rule.local.basedir=${expectedDir}
+
+

这样规则将会被存储在${expectedDir}/data/connection${expectedDir}/data/tps中。

+

+

当前支持的监控点名称

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
监控点名称对应内容描述起始版本
connection节点总连接数指定节点最大可支持连接数限制2.3.0
ConfigPublish配置发布接口TPS指定节点最大可支持配置发布的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigQuery配置查询接口TPS指定节点最大可支持配置查询的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigRemove配置移除接口TPS指定节点最大可支持配置移除的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigListen配置监听接口TPS指定节点最大可支持配置监听的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingInstanceRegisterDeregister服务实例注册及注销接口TPS服务实例注册或注销的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingInstanceBatchRegister服务实例批量注册接口TPS服务实例批量注册的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceListQuery服务列表查询接口TPS服务列表查询的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceQuery服务查询接口TPS服务查询的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceSubscribeUnSubscribe服务订阅和取消订阅接口TPS服务订阅和取消订阅的TPS限制,仅包含通过grpc访问的来源2.3.0
NamingInstanceRegister服务实例注册接口TPS服务实例注册的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceDeregister服务实例注销接口TPS服务实例注销的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceUpdate服务实例元数据更新接口TPS服务实例更新的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceMetadataUpdate服务实例元数据批量更新接口TPS服务实例元数据批量更新的TPS限制,仅包含通过http访问的来源2.3.0
NamingServiceSubscribe服务实例查询及订阅接口TPS服务订阅及查询的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceQuery单个服务实例查询接口TPS单个服务实例查询的TPS限制,仅包含通过http访问的来源2.3.0
HttpHealthCheck服务实例心跳续约接口TPS服务实例心跳续约的TPS限制,仅包含通过http访问的来源2.3.0
NamingServiceRegister服务创建接口TPS服务创建的TPS限制,与NamingInstanceRegister不同,此监控点表示的是创建空服务接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceDeregister服务删除接口TPS服务删除的TPS限制,与NamingInstanceDeregister不同,此监控点表示的是删除服务接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceQuery服务查询接口TPS服务查询的TPS限制,与NamingInstanceQuery不同,此监控点表示的是查询服务信息接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceListQuery服务列表查询接口TPS服务列表查询的TPS限制,与NamingServiceSubscribe不同,此监控点表示的是服务列表查询接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceUpdate服务元数据更新接口TPS服务元数据更新的TPS限制,与NamingInstanceUpdate不同,此监控点表示的是服务元数据更新接口所对应的TPS,仅包含通过http访问的来源2.3.0
+

反脆弱插件进阶开发

+

Nacos反脆弱插件还支持一些进阶式的拓展,以满足对此方面有更高要求的开发者和用户。

+

反脆弱规则外部存储

+

Nacos反脆弱插件的默认情况下,仅支持通过本地文件系统来存储和修改单节点的反脆弱规则,对于一些集群规模较大或集群较多的用户,逐个节点进行调整会消耗大量时间和操作;同时本地文件系统在许多容器化环境中,存在磁盘挂载和持久化的问题。因此Nacos反脆弱插件允许增加一个可选的外部存储进行反脆弱规则的统一存储和下发,外部存储可有插件自行实现对接,例如数据库,配置中心等。

+

要实现反脆弱规则的外部存储,需要在开发插件时,实现com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder接口,并随插件jar文件一起放置在${nacos-server.path}/plugins下。

+

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

+
nacos.plugin.control.rule.external.storage=${controlPluginName}
+
+

随后重启Nacos节点即可。

+

动态加载反脆弱规则

+

在自定义插件实现中,可以通过两种方式进行反脆弱规则的动态加载:

+
    +
  1. 调用com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule方法或com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule方法。
  2. +
  3. 通过NotifyCenter.publishEvent()发布ConnectionLimitRuleChangeEventTpsControlRuleChangeEvent事件。
  4. +
+

反脆弱规则的自定义格式解析

+

Nacos 默认使用Json格式作为反脆弱规则的文本格式,插件开发者也可以使用其他的格式,如Yaml或其他自定义格式进行解析。

+

重写com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParsercom.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser,实现自定义格式规则解析器RuleParser,Nacos将使用自定义的规则解析器RuleParser进行规则文本的解析。

+

同时,也可以解析成默认自定义规则的增强规则,配合自定义插件的定制逻辑实现更高程度的反脆弱控制。

+

+

自定义TPS时间窗口

+

众所周知,在统计TPS时,存在时间窗口算法的区别,不同的时间窗口对TPS的统计结果会有较大的区别。

+

Nacos默认使用简单的同秒统计方式,即按照时钟的秒来进行统计。对于大多数场景来说是足够使用的,但对于一些精确度要求高的用户而言,可能需要使用滑动窗口等更精确的方式进行统计。

+

此时需要插件开发者,继承com.alibaba.nacos.plugin.control.tps.barrier.TpsBarriercom.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier,自定义实现TPS的时间窗口和统计方式。并重写com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator,在初始化插件和动态加载反脆弱规则时,生成对应的自定义实现。

+
+ + + + + + + + + + + + + diff --git a/zh-cn/docs/v2/plugin/control-plugin.json b/zh-cn/docs/v2/plugin/control-plugin.json new file mode 100644 index 00000000000..e745e19eed9 --- /dev/null +++ b/zh-cn/docs/v2/plugin/control-plugin.json @@ -0,0 +1,10 @@ +{ + "filename": "control-plugin.md", + "__html": "

反脆弱插件

\n

Nacos 从2.3.0版本开始,支持通过SPI的方式注入反脆弱相关插件,并在application.properties配置文件中选择某一种插件实现作为实际反脆弱能力。本文档会详细介绍如何实现一个反脆弱插件和如何使其生效。

\n

反脆弱插件中的概念

\n

反脆弱是对访问服务端的某种资源频率和次数达到一定程度时进行的限制访问的策略,用于保护服务端在高压情况下能快速拒绝请求,防止过多的资源访问导致服务端资源耗尽引起的大面积不可用;Nacos反脆弱插件,将信息主要抽象为监控点反脆弱规则

\n

监控点(ControlPoint)

\n

监控点对应的请求服务端时所占用的资源的映射,目前主要针对的是连接(Connection)以及每秒请求数(TPS)

\n
    \n
  • 连接(Connection)监控点主要监控Nacos 服务端中使用Nacos2.X客户端的长连接数量以及使用Nacos1.X客户端的配置长轮询数量,两者独立监控。
  • \n
  • 每秒请求数(TPS)监控点主要是监控Nacos 服务端中各核心接口被访问的频率,同类型的操作接口会被视为相同的监控点,如注册服务的v1接口和v2接口,具体的每秒请求数(TPS)监控点可查看本文档下文监控点名称
  • \n
\n

反脆弱规则(ControlRule)

\n

反脆弱规则是针对每个监控点而执行的不同的限制规则,具体又分为连接数规则(ConnectionControlRule)每秒请求数规则(TpsControlRule)

\n

连接数规则(ConnectionControlRule)主要包含如下内容:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名类型描述
countLimitint连接数总数限制,默认为-1,不限制
monitorIpListSettrace监控的Ip列表,用于详细观察对应ip的连接做了哪些操作,添加后,对应ip的连接请求会被详细打印在remote-digest.log日志中
\n

每秒请求数规则(TpsControlRule)主要包含如下内容:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名类型描述
pointNameString规则所对应的监控点名称
pointRuleRuleDetail规则内容的具体细节
\n

其中RuleDetail又包含如下内容:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
字段名类型描述
ruleNameString规则的名称,区别于监控点名称,同一个监控点可以有多个规则名
maxCountintTPS总数限制,默认为-1,不限制
periodTimeUnit规则生效的周期,即统计到秒级/分钟级等,默认TimeUnit.SECONDS秒级
monitorTypeString监控类型,取值为monitorintercept,对应为监控模式(只统计和打印tps,即使触发规则也不拦截)和拦截模式
\n

插件开发

\n

开发Nacos服务端反脆弱插件,首先需要依赖反脆弱插件的相关API

\n
        <dependency>\n            <groupId>com.alibaba.nacos</groupId>\n            <artifactId>nacos-control-plugin</artifactId>\n            <version>${project.version}</version>\n        </dependency>\n
\n

${project.version} 为您开发插件所对应的Nacos版本,2.3.0及以上。

\n

随后继承com.alibaba.nacos.plugin.control.connection.ConnectionControlManager抽象类和com.alibaba.nacos.plugin.control.tps.TpsControlManager抽象类,实现缺失的方法;然后实现com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 接口,创建上述实现的两个抽象类;最后将您的实现添加到SPI的services当中。

\n

com.alibaba.nacos.plugin.control.connection.ConnectionControlManager需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
applyConnectionLimitRuleConnectionControlRulevoid应用新的连接数规则
checkConnectionCheckRequestConnectionCheckResponse判断是否命中连接数规则,如果ConnectionCheckResponse中的sucess为false,将会拒绝新连接的建立
\n

com.alibaba.nacos.plugin.control.tps.TpsControlManager需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
registerTpsPointStringvoid注册TPS监控点,Nacos服务会在启动时向插件注册当前的TPS监控买点,入参为TPS监控点的名字,具体的监控点名称可查看本文档下文监控点名称;插件需要在方法内,维护一个用于记录TPS和规则内容的TpsBarrier,详情查看自定义TPS时间窗口
applyTpsRuleString,TpsControlRulevoid应用新的TPS规则,根据TPS监控点名称关联及更新。
checkTpsCheckRequestTpsCheckResponse判断是否命中TPS规则,如果TpsCheckResponse中的sucess为false,将会拒绝新的请求。
\n

com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 需要实现的方法如下:

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
方法名入参内容返回内容描述
getNamevoidString插件的名称,和配置文件中指定的类型进行匹配,使用命中的的插件。
buildConnectionControlManagervoidConnectionControlManager创建插件对应的ConnectionControlManager实现,为null时会使用no limit实现。
buildTpsControlManagervoidTpsControlManager创建插件对应的TpsControlManager实现,为null时会使用no limit实现。
\n

加载插件

\n

插件开发完成后,需要打包成jar/zip,放置到nacos服务端的classpath中,如果您不知道如何修改classpath,请直接放置到${nacos-server.path}/plugins

\n

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
### 所启用的Nacos的反脆弱插件的名称,与com.alibaba.nacos.plugin.control.spi.ControlManagerBuilder 的getName 返回值对应\nnacos.plugin.control.manager.type=${controlPluginName}\n
\n

随后重启nacos集群,启动完成后,可以从${nacos-server.path}/logs/plugin-control.log中看到如下日志:

\n
Found control manager plugin of name=${controlPluginName}\n\nBuild connection control manager, class=${your plugin ConnectionControlManager class}\n\nBuild tps control manager, class=${your plugin TpsControlManager class}\n
\n

使用Nacos自带的反脆弱插件

\n

Nacos2.3.0版本起,自带一个简易的反脆弱插件实现,可以做到对Nacos服务端的连接数及指定接口TPS进行限制。

\n

启用Nacos自带的反脆弱插件

\n

需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
nacos.plugin.control.manager.type=nacos\n
\n

设置反脆弱规则

\n

可通过创建和修改反脆弱规则文件的方式,修改和设置反脆弱规则,默认反脆弱插件的规则是通过json格式定义的;例如想要设置连接限制为100,可执行如下操作:

\n
mkdir -p ${nacos.home}/data/connection/\necho '{\"countLimit\": 100}' > ${nacos.home}/data/connection/limitRule\n
\n

随后重启Nacos节点即可。

\n

又例如想要设置配置查询接口的TPS为100,可执行如下操作:

\n
mkdir -p ${nacos.home}/data/tps/\n# ConfigQuery 为配置查询接口的监控点名称(pointName)\necho '{\"pointName\":\"ConfigQuery\",\"pointRule\":{\"maxCount\":100,\"monitorType\":\"intercept\"}}' > ${nacos.home}/data/tps/ConfigQuery \n
\n

随后重启Nacos节点即可。

\n

其他更多反脆弱规则,以及具体的反脆弱监控点名称,请查看下文监控点名称

\n

反脆弱规则存储位置

\n

Nacos自带的简易反脆弱插件实现,Nacos服务端会通过本地文件系统,存储和读取反脆弱规则,默认所在目录的为${nacos.home}/data/connection${nacos.home}/data/tps中,如果想将规则文件更换目录存储,可以在${nacos-server.path}/conf/application.properties中修改以下配置:

\n
nacos.plugin.control.rule.local.basedir=${expectedDir}\n
\n

这样规则将会被存储在${expectedDir}/data/connection${expectedDir}/data/tps中。

\n

\n

当前支持的监控点名称

\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n
监控点名称对应内容描述起始版本
connection节点总连接数指定节点最大可支持连接数限制2.3.0
ConfigPublish配置发布接口TPS指定节点最大可支持配置发布的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigQuery配置查询接口TPS指定节点最大可支持配置查询的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigRemove配置移除接口TPS指定节点最大可支持配置移除的TPS限制,同时包含了通过http访问和grpc访问的来源2.3.0
ConfigListen配置监听接口TPS指定节点最大可支持配置监听的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingInstanceRegisterDeregister服务实例注册及注销接口TPS服务实例注册或注销的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingInstanceBatchRegister服务实例批量注册接口TPS服务实例批量注册的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceListQuery服务列表查询接口TPS服务列表查询的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceQuery服务查询接口TPS服务查询的TPS限制,仅包含通过grpc访问的来源2.3.0
RemoteNamingServiceSubscribeUnSubscribe服务订阅和取消订阅接口TPS服务订阅和取消订阅的TPS限制,仅包含通过grpc访问的来源2.3.0
NamingInstanceRegister服务实例注册接口TPS服务实例注册的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceDeregister服务实例注销接口TPS服务实例注销的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceUpdate服务实例元数据更新接口TPS服务实例更新的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceMetadataUpdate服务实例元数据批量更新接口TPS服务实例元数据批量更新的TPS限制,仅包含通过http访问的来源2.3.0
NamingServiceSubscribe服务实例查询及订阅接口TPS服务订阅及查询的TPS限制,仅包含通过http访问的来源2.3.0
NamingInstanceQuery单个服务实例查询接口TPS单个服务实例查询的TPS限制,仅包含通过http访问的来源2.3.0
HttpHealthCheck服务实例心跳续约接口TPS服务实例心跳续约的TPS限制,仅包含通过http访问的来源2.3.0
NamingServiceRegister服务创建接口TPS服务创建的TPS限制,与NamingInstanceRegister不同,此监控点表示的是创建空服务接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceDeregister服务删除接口TPS服务删除的TPS限制,与NamingInstanceDeregister不同,此监控点表示的是删除服务接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceQuery服务查询接口TPS服务查询的TPS限制,与NamingInstanceQuery不同,此监控点表示的是查询服务信息接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceListQuery服务列表查询接口TPS服务列表查询的TPS限制,与NamingServiceSubscribe不同,此监控点表示的是服务列表查询接口所对应的TPS,仅包含通过http访问的来源2.3.0
NamingServiceUpdate服务元数据更新接口TPS服务元数据更新的TPS限制,与NamingInstanceUpdate不同,此监控点表示的是服务元数据更新接口所对应的TPS,仅包含通过http访问的来源2.3.0
\n

反脆弱插件进阶开发

\n

Nacos反脆弱插件还支持一些进阶式的拓展,以满足对此方面有更高要求的开发者和用户。

\n

反脆弱规则外部存储

\n

Nacos反脆弱插件的默认情况下,仅支持通过本地文件系统来存储和修改单节点的反脆弱规则,对于一些集群规模较大或集群较多的用户,逐个节点进行调整会消耗大量时间和操作;同时本地文件系统在许多容器化环境中,存在磁盘挂载和持久化的问题。因此Nacos反脆弱插件允许增加一个可选的外部存储进行反脆弱规则的统一存储和下发,外部存储可有插件自行实现对接,例如数据库,配置中心等。

\n

要实现反脆弱规则的外部存储,需要在开发插件时,实现com.alibaba.nacos.plugin.control.spi.ExternalRuleStorageBuilder接口,并随插件jar文件一起放置在${nacos-server.path}/plugins下。

\n

放置后,需要修改${nacos-server.path}/conf/application.properties中的以下配置

\n
nacos.plugin.control.rule.external.storage=${controlPluginName}\n
\n

随后重启Nacos节点即可。

\n

动态加载反脆弱规则

\n

在自定义插件实现中,可以通过两种方式进行反脆弱规则的动态加载:

\n
    \n
  1. 调用com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadTpsControlRule方法或com.alibaba.nacos.plugin.control.ControlManagerCenter#reloadConnectionControlRule方法。
  2. \n
  3. 通过NotifyCenter.publishEvent()发布ConnectionLimitRuleChangeEventTpsControlRuleChangeEvent事件。
  4. \n
\n

反脆弱规则的自定义格式解析

\n

Nacos 默认使用Json格式作为反脆弱规则的文本格式,插件开发者也可以使用其他的格式,如Yaml或其他自定义格式进行解析。

\n

重写com.alibaba.nacos.plugin.control.connection.ConnectionControlManager#buildConnectionControlRuleParsercom.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsControlRuleParser,实现自定义格式规则解析器RuleParser,Nacos将使用自定义的规则解析器RuleParser进行规则文本的解析。

\n

同时,也可以解析成默认自定义规则的增强规则,配合自定义插件的定制逻辑实现更高程度的反脆弱控制。

\n

\n

自定义TPS时间窗口

\n

众所周知,在统计TPS时,存在时间窗口算法的区别,不同的时间窗口对TPS的统计结果会有较大的区别。

\n

Nacos默认使用简单的同秒统计方式,即按照时钟的秒来进行统计。对于大多数场景来说是足够使用的,但对于一些精确度要求高的用户而言,可能需要使用滑动窗口等更精确的方式进行统计。

\n

此时需要插件开发者,继承com.alibaba.nacos.plugin.control.tps.barrier.TpsBarriercom.alibaba.nacos.plugin.control.tps.barrier.RuleBarrier,自定义实现TPS的时间窗口和统计方式。并重写com.alibaba.nacos.plugin.control.tps.TpsControlManager#buildTpsBarrierCreator,在初始化插件和动态加载反脆弱规则时,生成对应的自定义实现。

\n", + "link": "/zh-cn/docs/v2/plugin/control-plugin.html", + "meta": { + "title": "反脆弱", + "keywords": "反脆弱,限流,连接数限制,TPS", + "description": "Nacos 支持反脆弱插件,避免高压下的集群容量问题。" + } +} \ No newline at end of file diff --git a/zh-cn/docs/what-is-nacos.html b/zh-cn/docs/what-is-nacos.html index d24c3fecfe0..5ec5b1be5df 100644 --- a/zh-cn/docs/what-is-nacos.html +++ b/zh-cn/docs/what-is-nacos.html @@ -74,11 +74,11 @@

Nacos 生态图

使用 Nacos 简化服务发现、配置管理、服务治理及管理的解决方案,让微服务的发现、管理、共享、组合更加容易。

关于如何在这些生态中使用 Nacos,请参考以下文档:

-

Nacos与Spring Cloud一起使用

+

Nacos与Spring Cloud一起使用

Nacos与Kubernetes一起使用

Nacos与Dubbo一起使用

-

Nacos与gRPC一起使用

-

Nacos与Istio一起使用

+

Nacos与gRPC一起使用

+

Nacos与Istio一起使用

下一步

继续阅读 快速开始 以快速上手 Nacos。

diff --git a/zh-cn/docs/what-is-nacos.json b/zh-cn/docs/what-is-nacos.json index 757512de12c..dbea7f2c897 100644 --- a/zh-cn/docs/what-is-nacos.json +++ b/zh-cn/docs/what-is-nacos.json @@ -1,6 +1,6 @@ { "filename": "what-is-nacos.md", - "__html": "

什么是 Nacos

\n

概览

\n

欢迎来到 Nacos 的世界!

\n

Nacos /nɑ:kəʊs/ 是 Dynamic Naming and Configuration Service的首字母简称,一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。

\n

Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。

\n

Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。 Nacos 是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施。

\n

什么是 Nacos?

\n

服务(Service)是 Nacos 世界的一等公民。Nacos 支持几乎所有主流类型的“服务”的发现、配置和管理:

\n

Kubernetes Service

\n

gRPC & Dubbo RPC Service

\n

Spring Cloud RESTful Service

\n

Nacos 的关键特性包括:

\n
    \n
  • \n

    服务发现和服务健康监测

    \n

    Nacos 支持基于 DNS 和基于 RPC 的服务发现。服务提供者使用 原生SDKOpenAPI、或一个独立的Agent TODO注册 Service 后,服务消费者可以使用DNS TODOHTTP&API查找和发现服务。

    \n

    Nacos 提供对服务的实时的健康检查,阻止向不健康的主机或服务实例发送请求。Nacos 支持传输层 (PING 或 TCP)和应用层 (如 HTTP、MySQL、用户自定义)的健康检查。 对于复杂的云环境和网络拓扑环境中(如 VPC、边缘网络等)服务的健康检查,Nacos 提供了 agent 上报模式和服务端主动检测2种健康检查模式。Nacos 还提供了统一的健康检查仪表盘,帮助您根据健康状态管理服务的可用性及流量。

    \n
  • \n
  • \n

    动态配置服务

    \n

    动态配置服务可以让您以中心化、外部化和动态化的方式管理所有环境的应用配置和服务配置。

    \n

    动态配置消除了配置变更时重新部署应用和服务的需要,让配置管理变得更加高效和敏捷。

    \n

    配置中心化管理让实现无状态服务变得更简单,让服务按需弹性扩展变得更容易。

    \n

    Nacos 提供了一个简洁易用的UI (控制台样例 Demo) 帮助您管理所有的服务和应用的配置。Nacos 还提供包括配置版本跟踪、金丝雀发布、一键回滚配置以及客户端配置更新状态跟踪在内的一系列开箱即用的配置管理特性,帮助您更安全地在生产环境中管理配置变更和降低配置变更带来的风险。

    \n
  • \n
  • \n

    动态 DNS 服务

    \n

    动态 DNS 服务支持权重路由,让您更容易地实现中间层负载均衡、更灵活的路由策略、流量控制以及数据中心内网的简单DNS解析服务。动态DNS服务还能让您更容易地实现以 DNS 协议为基础的服务发现,以帮助您消除耦合到厂商私有服务发现 API 上的风险。

    \n

    Nacos 提供了一些简单的 DNS APIs TODO 帮助您管理服务的关联域名和可用的 IP:PORT 列表.

    \n
  • \n
  • \n

    服务及其元数据管理

    \n

    Nacos 能让您从微服务平台建设的视角管理数据中心的所有服务及元数据,包括管理服务的描述、生命周期、服务的静态依赖分析、服务的健康状态、服务的流量管理、路由及安全策略、服务的 SLA 以及最首要的 metrics 统计数据。

    \n
  • \n
  • \n

    更多的特性列表 ...

    \n
  • \n
\n

Nacos 地图

\n

一图看懂 Nacos,下面架构部分会详细介绍。\n\"nacos_map\"

\n
    \n
  • 特性大图:要从功能特性,非功能特性,全面介绍我们要解的问题域的特性诉求
  • \n
  • 架构大图:通过清晰架构,让您快速进入 Nacos 世界
  • \n
  • 业务大图:利用当前特性可以支持的业务场景,及其最佳实践
  • \n
  • 生态大图:系统梳理 Nacos 和主流技术生态的关系
  • \n
  • 优势大图:展示 Nacos 核心竞争力
  • \n
  • 战略大图:要从战略到战术层面讲 Nacos 的宏观优势
  • \n
\n

Nacos 生态图

\n

\"nacos_landscape.png\"

\n

如 Nacos 全景图所示,Nacos 无缝支持一些主流的开源生态,例如

\n\n

使用 Nacos 简化服务发现、配置管理、服务治理及管理的解决方案,让微服务的发现、管理、共享、组合更加容易。

\n

关于如何在这些生态中使用 Nacos,请参考以下文档:

\n

Nacos与Spring Cloud一起使用

\n

Nacos与Kubernetes一起使用

\n

Nacos与Dubbo一起使用

\n

Nacos与gRPC一起使用

\n

Nacos与Istio一起使用

\n

下一步

\n

继续阅读 快速开始 以快速上手 Nacos。

\n", + "__html": "

什么是 Nacos

\n

概览

\n

欢迎来到 Nacos 的世界!

\n

Nacos /nɑ:kəʊs/ 是 Dynamic Naming and Configuration Service的首字母简称,一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。

\n

Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集,帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。

\n

Nacos 帮助您更敏捷和容易地构建、交付和管理微服务平台。 Nacos 是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施。

\n

什么是 Nacos?

\n

服务(Service)是 Nacos 世界的一等公民。Nacos 支持几乎所有主流类型的“服务”的发现、配置和管理:

\n

Kubernetes Service

\n

gRPC & Dubbo RPC Service

\n

Spring Cloud RESTful Service

\n

Nacos 的关键特性包括:

\n
    \n
  • \n

    服务发现和服务健康监测

    \n

    Nacos 支持基于 DNS 和基于 RPC 的服务发现。服务提供者使用 原生SDKOpenAPI、或一个独立的Agent TODO注册 Service 后,服务消费者可以使用DNS TODOHTTP&API查找和发现服务。

    \n

    Nacos 提供对服务的实时的健康检查,阻止向不健康的主机或服务实例发送请求。Nacos 支持传输层 (PING 或 TCP)和应用层 (如 HTTP、MySQL、用户自定义)的健康检查。 对于复杂的云环境和网络拓扑环境中(如 VPC、边缘网络等)服务的健康检查,Nacos 提供了 agent 上报模式和服务端主动检测2种健康检查模式。Nacos 还提供了统一的健康检查仪表盘,帮助您根据健康状态管理服务的可用性及流量。

    \n
  • \n
  • \n

    动态配置服务

    \n

    动态配置服务可以让您以中心化、外部化和动态化的方式管理所有环境的应用配置和服务配置。

    \n

    动态配置消除了配置变更时重新部署应用和服务的需要,让配置管理变得更加高效和敏捷。

    \n

    配置中心化管理让实现无状态服务变得更简单,让服务按需弹性扩展变得更容易。

    \n

    Nacos 提供了一个简洁易用的UI (控制台样例 Demo) 帮助您管理所有的服务和应用的配置。Nacos 还提供包括配置版本跟踪、金丝雀发布、一键回滚配置以及客户端配置更新状态跟踪在内的一系列开箱即用的配置管理特性,帮助您更安全地在生产环境中管理配置变更和降低配置变更带来的风险。

    \n
  • \n
  • \n

    动态 DNS 服务

    \n

    动态 DNS 服务支持权重路由,让您更容易地实现中间层负载均衡、更灵活的路由策略、流量控制以及数据中心内网的简单DNS解析服务。动态DNS服务还能让您更容易地实现以 DNS 协议为基础的服务发现,以帮助您消除耦合到厂商私有服务发现 API 上的风险。

    \n

    Nacos 提供了一些简单的 DNS APIs TODO 帮助您管理服务的关联域名和可用的 IP:PORT 列表.

    \n
  • \n
  • \n

    服务及其元数据管理

    \n

    Nacos 能让您从微服务平台建设的视角管理数据中心的所有服务及元数据,包括管理服务的描述、生命周期、服务的静态依赖分析、服务的健康状态、服务的流量管理、路由及安全策略、服务的 SLA 以及最首要的 metrics 统计数据。

    \n
  • \n
  • \n

    更多的特性列表 ...

    \n
  • \n
\n

Nacos 地图

\n

一图看懂 Nacos,下面架构部分会详细介绍。\n\"nacos_map\"

\n
    \n
  • 特性大图:要从功能特性,非功能特性,全面介绍我们要解的问题域的特性诉求
  • \n
  • 架构大图:通过清晰架构,让您快速进入 Nacos 世界
  • \n
  • 业务大图:利用当前特性可以支持的业务场景,及其最佳实践
  • \n
  • 生态大图:系统梳理 Nacos 和主流技术生态的关系
  • \n
  • 优势大图:展示 Nacos 核心竞争力
  • \n
  • 战略大图:要从战略到战术层面讲 Nacos 的宏观优势
  • \n
\n

Nacos 生态图

\n

\"nacos_landscape.png\"

\n

如 Nacos 全景图所示,Nacos 无缝支持一些主流的开源生态,例如

\n\n

使用 Nacos 简化服务发现、配置管理、服务治理及管理的解决方案,让微服务的发现、管理、共享、组合更加容易。

\n

关于如何在这些生态中使用 Nacos,请参考以下文档:

\n

Nacos与Spring Cloud一起使用

\n

Nacos与Kubernetes一起使用

\n

Nacos与Dubbo一起使用

\n

Nacos与gRPC一起使用

\n

Nacos与Istio一起使用

\n

下一步

\n

继续阅读 快速开始 以快速上手 Nacos。

\n", "link": "/zh-cn/docs/what-is-nacos.html", "meta": { "title": "什么是 Nacos", diff --git a/zh-cn/index.html b/zh-cn/index.html index e520a300965..40a5b1dcaab 100644 --- a/zh-cn/index.html +++ b/zh-cn/index.html @@ -15,7 +15,7 @@ -

一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。

2023年5月25日发布

功能

动态配置服务

动态配置服务让您能够以中心化、外部化和动态化的方式管理所有环境的配置。动态配置消除了配置变更时重新部署应用和服务的需要。配置中心化管理让实现无状态服务更简单,也让按需弹性扩展服务更容易。

服务发现及管理

动态服务发现对以服务为中心的(例如微服务和云原生)应用架构方式非常关键。Nacos支持DNS-Based和RPC-Based(Dubbo、gRPC)模式的服务发现。Nacos也提供实时健康检查,以防止将请求发往不健康的主机或服务实例。借助Nacos,您可以更容易地为您的服务实现断路器。

动态DNS服务

通过支持权重路由,动态DNS服务能让您轻松实现中间层负载均衡、更灵活的路由策略、流量控制以及简单数据中心内网的简单DNS解析服务。动态DNS服务还能让您更容易地实现以DNS协议为基础的服务发现,以消除耦合到厂商私有服务发现API上的风险。

微服务生态全景图

特性一览

  • 易于使用

    • 动态配置管理、服务发现和动态的一站式解决方案
    • 20多种开箱即用的以服务为中心的架构特性
    • 基本符合生产要求的轻量级易用控制台
  • 更适应云架构

    • 无缝支持Kubernetes和Spring Cloud
    • 在主流公共云上更容易部署和运行(例如阿里云和AWS)
    • 多租户和多环境支持
  • 生产等级

    • 脱胎于历经阿里巴巴10年生产验证的内部产品
    • 支持具有数百万服务的大规模场景
    • 具备企业级SLA的开源产品
  • 丰富的应用场景

    • 支持限流、大促销预案和异地多活
    • 直接支持或稍作扩展即可支持大量有用的互联网应用场景
    • 流量调度和服务治理

谁在使用Nacos

请在 谁在使用Nacos 上提供信息来帮助Nacos做的更好。

+

一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。

2023年10月19日发布

功能

动态配置服务

动态配置服务让您能够以中心化、外部化和动态化的方式管理所有环境的配置。动态配置消除了配置变更时重新部署应用和服务的需要。配置中心化管理让实现无状态服务更简单,也让按需弹性扩展服务更容易。

服务发现及管理

动态服务发现对以服务为中心的(例如微服务和云原生)应用架构方式非常关键。Nacos支持DNS-Based和RPC-Based(Dubbo、gRPC)模式的服务发现。Nacos也提供实时健康检查,以防止将请求发往不健康的主机或服务实例。借助Nacos,您可以更容易地为您的服务实现断路器。

动态DNS服务

通过支持权重路由,动态DNS服务能让您轻松实现中间层负载均衡、更灵活的路由策略、流量控制以及简单数据中心内网的简单DNS解析服务。动态DNS服务还能让您更容易地实现以DNS协议为基础的服务发现,以消除耦合到厂商私有服务发现API上的风险。

微服务生态全景图

特性一览

  • 易于使用

    • 动态配置管理、服务发现和动态的一站式解决方案
    • 20多种开箱即用的以服务为中心的架构特性
    • 基本符合生产要求的轻量级易用控制台
  • 更适应云架构

    • 无缝支持Kubernetes和Spring Cloud
    • 在主流公共云上更容易部署和运行(例如阿里云和AWS)
    • 多租户和多环境支持
  • 生产等级

    • 脱胎于历经阿里巴巴10年生产验证的内部产品
    • 支持具有数百万服务的大规模场景
    • 具备企业级SLA的开源产品
  • 丰富的应用场景

    • 支持限流、大促销预案和异地多活
    • 直接支持或稍作扩展即可支持大量有用的互联网应用场景
    • 流量调度和服务治理

谁在使用Nacos

请在 谁在使用Nacos 上提供信息来帮助Nacos做的更好。