mirror of
https://gitee.com/mirrors/Spring-Cloud-Alibaba.git
synced 2021-06-26 13:25:11 +08:00
Merge pull request #1972 from alibaba/master
merge 2.2.5.RELEASE to 2020.0.0
This commit is contained in:
@@ -19,7 +19,7 @@ Spring Cloud Alibaba 致力于提供微服务开发的一站式解决方案。
|
|||||||
* **服务注册与发现**:适配 Spring Cloud 服务注册与发现标准,默认集成了 Ribbon 的支持。
|
* **服务注册与发现**:适配 Spring Cloud 服务注册与发现标准,默认集成了 Ribbon 的支持。
|
||||||
* **分布式配置管理**:支持分布式系统中的外部化配置,配置更改时自动刷新。
|
* **分布式配置管理**:支持分布式系统中的外部化配置,配置更改时自动刷新。
|
||||||
* **消息驱动能力**:基于 Spring Cloud Stream 为微服务应用构建消息驱动能力。
|
* **消息驱动能力**:基于 Spring Cloud Stream 为微服务应用构建消息驱动能力。
|
||||||
* **分布式事务**:使用 @GlobalTransactional 注解, 高效并且对业务零侵入地解决分布式事务问题。。
|
* **分布式事务**:使用 @GlobalTransactional 注解, 高效并且对业务零侵入地解决分布式事务问题。
|
||||||
* **阿里云对象存储**:阿里云提供的海量、安全、低成本、高可靠的云存储服务。支持在任何应用、任何时间、任何地点存储和访问任意类型的数据。
|
* **阿里云对象存储**:阿里云提供的海量、安全、低成本、高可靠的云存储服务。支持在任何应用、任何时间、任何地点存储和访问任意类型的数据。
|
||||||
* **分布式任务调度**:提供秒级、精准、高可靠、高可用的定时(基于 Cron 表达式)任务调度服务。同时提供分布式的任务执行模型,如网格任务。网格任务支持海量子任务均匀分配到所有 Worker(schedulerx-client)上执行。
|
* **分布式任务调度**:提供秒级、精准、高可靠、高可用的定时(基于 Cron 表达式)任务调度服务。同时提供分布式的任务执行模型,如网格任务。网格任务支持海量子任务均匀分配到所有 Worker(schedulerx-client)上执行。
|
||||||
* **阿里云短信服务**:覆盖全球的短信服务,友好、高效、智能的互联化通讯能力,帮助企业迅速搭建客户触达通道。
|
* **阿里云短信服务**:覆盖全球的短信服务,友好、高效、智能的互联化通讯能力,帮助企业迅速搭建客户触达通道。
|
||||||
@@ -70,7 +70,7 @@ Spring Cloud 使用 Maven 来构建,最快的使用方式是将本项目 clone
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.alibaba.cloud</groupId>
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
||||||
<version>2.2.3.RELEASE</version>
|
<version>2.2.5.RELEASE</version>
|
||||||
<type>pom</type>
|
<type>pom</type>
|
||||||
<scope>import</scope>
|
<scope>import</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
@@ -27,6 +27,7 @@ With Spring Cloud Alibaba, you only need to add some annotations and a small amo
|
|||||||
|
|
||||||
For more features, please refer to [Roadmap](https://github.com/alibaba/spring-cloud-alibaba/blob/master/Roadmap.md).
|
For more features, please refer to [Roadmap](https://github.com/alibaba/spring-cloud-alibaba/blob/master/Roadmap.md).
|
||||||
|
|
||||||
|
|
||||||
## Components
|
## Components
|
||||||
|
|
||||||
**[Sentinel](https://github.com/alibaba/Sentinel)**: Sentinel takes "traffic flow" as the breakthrough point, and provides solutions in areas such as flow control, concurrency, circuit breaking, and load protection to protect service stability.
|
**[Sentinel](https://github.com/alibaba/Sentinel)**: Sentinel takes "traffic flow" as the breakthrough point, and provides solutions in areas such as flow control, concurrency, circuit breaking, and load protection to protect service stability.
|
||||||
@@ -71,7 +72,7 @@ These artifacts are available from Maven Central and Spring Release repository v
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.alibaba.cloud</groupId>
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
||||||
<version>2.2.3.RELEASE</version>
|
<version>2.2.5.RELEASE</version>
|
||||||
<type>pom</type>
|
<type>pom</type>
|
||||||
<scope>import</scope>
|
<scope>import</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
2
pom.xml
2
pom.xml
@@ -80,7 +80,7 @@
|
|||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<!-- Project revision -->
|
<!-- Project revision -->
|
||||||
<revision>2.2.4-SNAPSHOT</revision>
|
<revision>2.2.5.RC2</revision>
|
||||||
|
|
||||||
<!-- Dependency Versions -->
|
<!-- Dependency Versions -->
|
||||||
<spring-cloud-commons.version>2.2.5.RELEASE</spring-cloud-commons.version>
|
<spring-cloud-commons.version>2.2.5.RELEASE</spring-cloud-commons.version>
|
||||||
|
@@ -18,10 +18,10 @@
|
|||||||
<description>Spring Cloud Alibaba Dependencies</description>
|
<description>Spring Cloud Alibaba Dependencies</description>
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<revision>2.2.4-SNAPSHOT</revision>
|
<revision>2.2.5.RC2</revision>
|
||||||
<sentinel.version>1.8.0</sentinel.version>
|
<sentinel.version>1.8.0</sentinel.version>
|
||||||
<seata.version>1.3.0</seata.version>
|
<seata.version>1.3.0</seata.version>
|
||||||
<nacos.client.version>1.4.0</nacos.client.version>
|
<nacos.client.version>1.4.1</nacos.client.version>
|
||||||
<nacos.config.version>0.8.0</nacos.config.version>
|
<nacos.config.version>0.8.0</nacos.config.version>
|
||||||
<spring.context.support.version>1.0.10</spring.context.support.version>
|
<spring.context.support.version>1.0.10</spring.context.support.version>
|
||||||
|
|
||||||
@@ -245,6 +245,12 @@
|
|||||||
<version>${revision}</version>
|
<version>${revision}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
<artifactId>spring-cloud-alibaba-commons</artifactId>
|
||||||
|
<version>${revision}</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<!-- Dubbo -->
|
<!-- Dubbo -->
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.alibaba.cloud</groupId>
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
@@ -10,7 +10,7 @@ Spring Cloud Alibaba BOM 包含了它所使用的所有依赖的版本。
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.alibaba.cloud</groupId>
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
||||||
<version>2.2.3.RELEASE</version>
|
<version>2.2.4.RELEASE</version>
|
||||||
<type>pom</type>
|
<type>pom</type>
|
||||||
<scope>import</scope>
|
<scope>import</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
@@ -8,7 +8,7 @@ If you’re a Maven Central user, add our BOM to your pom.xml <dependencyManagem
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.alibaba.cloud</groupId>
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
<artifactId>spring-cloud-alibaba-dependencies</artifactId>
|
||||||
<version>2.2.3.RELEASE</version>
|
<version>2.2.4.RELEASE</version>
|
||||||
<type>pom</type>
|
<type>pom</type>
|
||||||
<scope>import</scope>
|
<scope>import</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
@@ -161,29 +161,16 @@ class UserConfig {
|
|||||||
@Component
|
@Component
|
||||||
class SampleRunner implements ApplicationRunner {
|
class SampleRunner implements ApplicationRunner {
|
||||||
|
|
||||||
@Value("${user.name:zz}")
|
|
||||||
String userName;
|
|
||||||
|
|
||||||
@Value("${user.age:25}")
|
|
||||||
int userAge;
|
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private NacosConfigManager nacosConfigManager;
|
private NacosConfigManager nacosConfigManager;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run(ApplicationArguments args) throws Exception {
|
public void run(ApplicationArguments args) throws Exception {
|
||||||
System.out.println(
|
|
||||||
String.format("Initial username=%s, userAge=%d", userName, userAge));
|
|
||||||
|
|
||||||
nacosConfigManager.getConfigService().addListener(
|
nacosConfigManager.getConfigService().addListener(
|
||||||
"nacos-config-example.properties", "DEFAULT_GROUP", new Listener() {
|
"nacos-config-custom.properties", "DEFAULT_GROUP", new Listener() {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Callback with latest config data.
|
* Callback with latest config data.
|
||||||
*
|
|
||||||
* For example, config data in Nacos is:
|
|
||||||
*
|
|
||||||
* user.name=Nacos user.age=25
|
|
||||||
* @param configInfo latest config data for specific dataId in Nacos
|
* @param configInfo latest config data for specific dataId in Nacos
|
||||||
* server
|
* server
|
||||||
*/
|
*/
|
||||||
|
@@ -5,21 +5,27 @@ spring.cloud.nacos.config.server-addr=127.0.0.1:8848
|
|||||||
spring.cloud.nacos.username=nacos
|
spring.cloud.nacos.username=nacos
|
||||||
spring.cloud.nacos.password=nacos
|
spring.cloud.nacos.password=nacos
|
||||||
|
|
||||||
|
## nacos-namespace cannot user 'public',cause by 'public' has special handing inside.
|
||||||
#spring.cloud.nacos.config.namespace=public
|
#spring.cloud.nacos.config.namespace=public
|
||||||
spring.cloud.nacos.config.name=test-aaa
|
|
||||||
spring.cloud.nacos.config.file-extension=yaml
|
|
||||||
|
|
||||||
|
## you can specify a custom name if you don't want to use the application name.
|
||||||
|
#spring.cloud.nacos.config.name=test-aaa
|
||||||
|
#spring.cloud.nacos.config.file-extension=yaml
|
||||||
|
|
||||||
|
## not recommended.
|
||||||
#spring.cloud.nacos.config.refreshable-dataids=common.properties
|
#spring.cloud.nacos.config.refreshable-dataids=common.properties
|
||||||
|
## not recommended.
|
||||||
#spring.cloud.nacos.config.shared-data-ids=common.properties,base-common.properties
|
#spring.cloud.nacos.config.shared-data-ids=common.properties,base-common.properties
|
||||||
#spring.cloud.nacos.config.shared-configs[0]= common333.properties
|
|
||||||
#spring.cloud.nacos.config.shared-configs[1].data-id= common111.properties
|
## recommended.
|
||||||
#spring.cloud.nacos.config.shared-configs[1].group= GROUP_APP1
|
|
||||||
#spring.cloud.nacos.config.shared-configs[1].refresh= true
|
|
||||||
#spring.cloud.nacos.config.shared-configs[2]= common222.properties
|
|
||||||
spring.cloud.nacos.config.shared-configs[0].data-id= test2.yaml
|
spring.cloud.nacos.config.shared-configs[0].data-id= test2.yaml
|
||||||
spring.cloud.nacos.config.shared-configs[0].refresh=true
|
spring.cloud.nacos.config.shared-configs[0].refresh=true
|
||||||
|
## the default value is 'DEFAULT_GROUP' , if not specified.
|
||||||
|
spring.cloud.nacos.config.shared-configs[0].group= GROUP_APP1
|
||||||
|
|
||||||
|
## not recommended.
|
||||||
#spring.cloud.nacos.config.ext-config[0]=ext.properties
|
#spring.cloud.nacos.config.ext-config[0]=ext.properties
|
||||||
|
## recommended.
|
||||||
spring.cloud.nacos.config.extension-configs[0].data-id= extension1.properties
|
spring.cloud.nacos.config.extension-configs[0].data-id= extension1.properties
|
||||||
spring.cloud.nacos.config.extension-configs[0].refresh=true
|
spring.cloud.nacos.config.extension-configs[0].refresh=true
|
||||||
spring.cloud.nacos.config.extension-configs[1].data-id= test1.yml
|
spring.cloud.nacos.config.extension-configs[1].data-id= test1.yml
|
||||||
|
@@ -72,7 +72,7 @@ Before we start the demo, let's learn how to connect Nacos Config to a Spring Cl
|
|||||||
|
|
||||||
#### Query Service
|
#### Query Service
|
||||||
|
|
||||||
Enter `http://127.0.0.1:8848/nacos/v1/ns/instances?serviceName=service-provider` in the browser address bar and click Go to, we can see that the service node has been successfully registered to Nacos Server.
|
Enter `http://127.0.0.1:8848/nacos/#/serviceDetail?name=service-provider&groupName=DEFAULT_GROUP` in the browser address bar and click Go to, we can see that the service node has been successfully registered to Nacos Server.
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
@@ -73,7 +73,7 @@ spring.cloud.stream.bindings.input.group=test-group
|
|||||||
|
|
||||||
You should startup Name Server and Broker before using RocketMQ Binder.
|
You should startup Name Server and Broker before using RocketMQ Binder.
|
||||||
|
|
||||||
1. Download [RocketMQ](https://www.apache.org/dyn/closer.cgi?path=rocketmq/4.3.2/rocketmq-all-4.3.2-bin-release.zip) and unzip it.
|
1. Download [RocketMQ](https://archive.apache.org/dist/rocketmq/4.3.2/rocketmq-all-4.3.2-bin-release.zip) and unzip it.
|
||||||
|
|
||||||
2. Startup Name Server
|
2. Startup Name Server
|
||||||
|
|
||||||
|
@@ -74,10 +74,14 @@ public class RocketMQConsumerApplication {
|
|||||||
while (true) {
|
while (true) {
|
||||||
mySink.input5().poll(m -> {
|
mySink.input5().poll(m -> {
|
||||||
String payload = (String) m.getPayload();
|
String payload = (String) m.getPayload();
|
||||||
|
if (payload.contains("0")) {
|
||||||
|
throw new IllegalArgumentException(
|
||||||
|
"111111111111111111111111111111111111111111");
|
||||||
|
}
|
||||||
System.out.println("pull msg: " + payload);
|
System.out.println("pull msg: " + payload);
|
||||||
}, new ParameterizedTypeReference<String>() {
|
}, new ParameterizedTypeReference<String>() {
|
||||||
});
|
});
|
||||||
Thread.sleep(2_000);
|
Thread.sleep(5_00);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -15,6 +15,11 @@
|
|||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
<artifactId>spring-cloud-alibaba-commons</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-web</artifactId>
|
<artifactId>spring-boot-starter-web</artifactId>
|
||||||
|
@@ -52,8 +52,8 @@ import org.springframework.util.StreamUtils;
|
|||||||
import org.springframework.web.servlet.HttpServletBean;
|
import org.springframework.web.servlet.HttpServletBean;
|
||||||
import org.springframework.web.util.UriComponents;
|
import org.springframework.web.util.UriComponents;
|
||||||
|
|
||||||
import static org.apache.commons.lang3.StringUtils.substringAfter;
|
import static com.alibaba.cloud.commons.lang.StringUtils.substringAfter;
|
||||||
import static org.apache.commons.lang3.StringUtils.substringBetween;
|
import static com.alibaba.cloud.commons.lang.StringUtils.substringBetween;
|
||||||
import static org.springframework.web.util.UriComponentsBuilder.fromUriString;
|
import static org.springframework.web.util.UriComponentsBuilder.fromUriString;
|
||||||
|
|
||||||
@WebServlet(urlPatterns = "/dsc/*")
|
@WebServlet(urlPatterns = "/dsc/*")
|
||||||
|
@@ -27,6 +27,7 @@
|
|||||||
<module>spring-cloud-starter-alibaba-sentinel</module>
|
<module>spring-cloud-starter-alibaba-sentinel</module>
|
||||||
<module>spring-cloud-alibaba-sentinel-datasource</module>
|
<module>spring-cloud-alibaba-sentinel-datasource</module>
|
||||||
<module>spring-cloud-alibaba-sentinel-gateway</module>
|
<module>spring-cloud-alibaba-sentinel-gateway</module>
|
||||||
|
<module>spring-cloud-alibaba-commons</module>
|
||||||
</modules>
|
</modules>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
|
@@ -0,0 +1,18 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<parent>
|
||||||
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
<artifactId>spring-cloud-alibaba-starters</artifactId>
|
||||||
|
<version>${revision}</version>
|
||||||
|
<relativePath>../pom.xml</relativePath>
|
||||||
|
</parent>
|
||||||
|
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<artifactId>spring-cloud-alibaba-commons</artifactId>
|
||||||
|
<name>Spring Cloud Alibaba Commons</name>
|
||||||
|
|
||||||
|
|
||||||
|
</project>
|
@@ -0,0 +1,182 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2019 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.commons.io;
|
||||||
|
|
||||||
|
import java.nio.charset.Charset;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.SortedMap;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The Charsets constants, copy from apache commons-io.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">theonefx</a>
|
||||||
|
*/
|
||||||
|
public final class Charsets {
|
||||||
|
|
||||||
|
private Charsets() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a sorted map from canonical charset names to charset objects required of
|
||||||
|
* every implementation of the Java platform.
|
||||||
|
* <p>
|
||||||
|
* From the Java documentation
|
||||||
|
* <a href="https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">
|
||||||
|
* Standard charsets</a>:
|
||||||
|
* </p>
|
||||||
|
* @return An immutable, case-insensitive map from canonical charset names to charset
|
||||||
|
* objects.
|
||||||
|
* @see Charset#availableCharsets()
|
||||||
|
*/
|
||||||
|
public static SortedMap<String, Charset> requiredCharsets() {
|
||||||
|
// maybe cache?
|
||||||
|
final TreeMap<String, Charset> m = new TreeMap<>(String.CASE_INSENSITIVE_ORDER);
|
||||||
|
m.put(StandardCharsets.ISO_8859_1.name(), StandardCharsets.ISO_8859_1);
|
||||||
|
m.put(StandardCharsets.US_ASCII.name(), StandardCharsets.US_ASCII);
|
||||||
|
m.put(StandardCharsets.UTF_16.name(), StandardCharsets.UTF_16);
|
||||||
|
m.put(StandardCharsets.UTF_16BE.name(), StandardCharsets.UTF_16BE);
|
||||||
|
m.put(StandardCharsets.UTF_16LE.name(), StandardCharsets.UTF_16LE);
|
||||||
|
m.put(StandardCharsets.UTF_8.name(), StandardCharsets.UTF_8);
|
||||||
|
return Collections.unmodifiableSortedMap(m);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the given Charset or the default Charset if the given Charset is null.
|
||||||
|
* @param charset A charset or null.
|
||||||
|
* @return the given Charset or the default Charset if the given Charset is null
|
||||||
|
*/
|
||||||
|
public static Charset toCharset(final Charset charset) {
|
||||||
|
return charset == null ? Charset.defaultCharset() : charset;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a Charset for the named charset. If the name is null, return the default
|
||||||
|
* Charset.
|
||||||
|
* @param charset The name of the requested charset, may be null.
|
||||||
|
* @return a Charset for the named charset
|
||||||
|
* @throws java.nio.charset.UnsupportedCharsetException If the named charset is
|
||||||
|
* unavailable
|
||||||
|
*/
|
||||||
|
public static Charset toCharset(final String charset) {
|
||||||
|
return charset == null ? Charset.defaultCharset() : Charset.forName(charset);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CharEncodingISO Latin Alphabet No. 1, a.k.a. ISO-LATIN-1.
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset ISO_8859_1 = StandardCharsets.ISO_8859_1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Seven-bit ASCII, also known as ISO646-US, also known as the Basic Latin block of
|
||||||
|
* the Unicode character set.
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset US_ASCII = StandardCharsets.US_ASCII;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Sixteen-bit Unicode Transformation Format, The byte order specified by a mandatory
|
||||||
|
* initial byte-order mark (either order accepted on input, big-endian used on output)
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset UTF_16 = StandardCharsets.UTF_16;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Sixteen-bit Unicode Transformation Format, big-endian byte order.
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset UTF_16BE = StandardCharsets.UTF_16BE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Sixteen-bit Unicode Transformation Format, little-endian byte order.
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset UTF_16LE = StandardCharsets.UTF_16LE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Eight-bit Unicode Transformation Format.
|
||||||
|
* </p>
|
||||||
|
* <p>
|
||||||
|
* Every implementation of the Java platform is required to support this character
|
||||||
|
* encoding.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* @see <a href=
|
||||||
|
* "https://docs.oracle.com/javase/7/docs/api/java/nio/charset/Charset.html">Standard
|
||||||
|
* charsets</a>
|
||||||
|
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static final Charset UTF_8 = StandardCharsets.UTF_8;
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,113 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2019 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.commons.io;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.nio.charset.Charset;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FileUtils. copy from apache commons-io.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">theonefx</a>
|
||||||
|
*/
|
||||||
|
public final class FileUtils {
|
||||||
|
|
||||||
|
private FileUtils() {
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Opens a {@link java.io.FileInputStream} for the specified file, providing better
|
||||||
|
* error messages than simply calling <code>new FileInputStream(file)</code>.
|
||||||
|
* <p>
|
||||||
|
* At the end of the method either the stream will be successfully opened, or an
|
||||||
|
* exception will have been thrown.
|
||||||
|
* <p>
|
||||||
|
* An exception is thrown if the file does not exist. An exception is thrown if the
|
||||||
|
* file object exists but is a directory. An exception is thrown if the file exists
|
||||||
|
* but cannot be read.
|
||||||
|
* @param file the file to open for input, must not be {@code null}
|
||||||
|
* @return a new {@link java.io.FileInputStream} for the specified file
|
||||||
|
* @throws java.io.FileNotFoundException if the file does not exist
|
||||||
|
* @throws IOException if the file object is a directory
|
||||||
|
* @throws IOException if the file cannot be read
|
||||||
|
* @since 1.3
|
||||||
|
*/
|
||||||
|
public static FileInputStream openInputStream(final File file) throws IOException {
|
||||||
|
if (file.exists()) {
|
||||||
|
if (file.isDirectory()) {
|
||||||
|
throw new IOException("File '" + file + "' exists but is a directory");
|
||||||
|
}
|
||||||
|
if (!file.canRead()) {
|
||||||
|
throw new IOException("File '" + file + "' cannot be read");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new FileNotFoundException("File '" + file + "' does not exist");
|
||||||
|
}
|
||||||
|
return new FileInputStream(file);
|
||||||
|
}
|
||||||
|
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Reads the contents of a file into a String. The file is always closed.
|
||||||
|
* @param file the file to read, must not be {@code null}
|
||||||
|
* @param encoding the encoding to use, {@code null} means platform default
|
||||||
|
* @return the file contents, never {@code null}
|
||||||
|
* @throws IOException in case of an I/O error
|
||||||
|
*/
|
||||||
|
public static String readFileToString(final File file, final Charset encoding)
|
||||||
|
throws IOException {
|
||||||
|
try (InputStream in = openInputStream(file)) {
|
||||||
|
return IOUtils.toString(in, Charsets.toCharset(encoding));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the contents of a file into a String. The file is always closed.
|
||||||
|
* @param file the file to read, must not be {@code null}
|
||||||
|
* @param encoding the encoding to use, {@code null} means platform default
|
||||||
|
* @return the file contents, never {@code null}
|
||||||
|
* @throws java.io.IOException in case of an I/O error
|
||||||
|
* @throws java.nio.charset.UnsupportedCharsetException thrown instead of
|
||||||
|
* {@link java.io .UnsupportedEncodingException} in version 2.2 if the encoding is not
|
||||||
|
* supported.
|
||||||
|
*/
|
||||||
|
public static String readFileToString(final File file, final String encoding)
|
||||||
|
throws IOException {
|
||||||
|
return readFileToString(file, Charsets.toCharset(encoding));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the contents of a file into a String using the default encoding for the VM.
|
||||||
|
* The file is always closed.
|
||||||
|
* @param file the file to read, must not be {@code null}
|
||||||
|
* @return the file contents, never {@code null}
|
||||||
|
* @throws IOException in case of an I/O error
|
||||||
|
* @deprecated 2.5 use {@link #readFileToString(File, String)} instead (and specify
|
||||||
|
* the appropriate encoding)
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public static String readFileToString(final File file) throws IOException {
|
||||||
|
return readFileToString(file, Charset.defaultCharset());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,233 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2019 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.commons.io;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.io.Writer;
|
||||||
|
import java.nio.charset.Charset;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The IOUtils. copy from apache commons-io.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">theonefx</a>
|
||||||
|
*/
|
||||||
|
public final class IOUtils {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the end-of-file (or stream).
|
||||||
|
* @since 2.5 (made public)
|
||||||
|
*/
|
||||||
|
public static final int EOF = -1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default buffer size ({@value}) to use for.
|
||||||
|
* {@link #copyLarge(InputStream, java.io.OutputStream)} and
|
||||||
|
* {@link #copyLarge(Reader, Writer)}
|
||||||
|
*/
|
||||||
|
private static final int DEFAULT_BUFFER_SIZE = 1024 * 4;
|
||||||
|
|
||||||
|
private IOUtils() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the contents of an <code>InputStream</code> as a String using the specified
|
||||||
|
* character encoding.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedInputStream</code>.
|
||||||
|
* </p>
|
||||||
|
* @param input the <code>InputStream</code> to read from
|
||||||
|
* @param encoding the encoding to use, null means platform default
|
||||||
|
* @return the requested String
|
||||||
|
* @throws NullPointerException if the input is null
|
||||||
|
* @throws java.io.IOException if an I/O error occurs
|
||||||
|
* @since 2.3
|
||||||
|
*/
|
||||||
|
public static String toString(final InputStream input, final Charset encoding)
|
||||||
|
throws IOException {
|
||||||
|
try (StringBuilderWriter sw = new StringBuilderWriter()) {
|
||||||
|
copy(input, sw, encoding);
|
||||||
|
return sw.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// copy from Reader
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies chars from a <code>Reader</code> to a <code>Writer</code>.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedReader</code>.
|
||||||
|
* <p>
|
||||||
|
* Large streams (over 2GB) will return a chars copied value of <code>-1</code> after
|
||||||
|
* the copy has completed since the correct number of chars cannot be returned as an
|
||||||
|
* int. For large streams use the <code>copyLarge(Reader, Writer)</code> method.
|
||||||
|
* @param input the <code>Reader</code> to read from
|
||||||
|
* @param output the <code>Writer</code> to write to
|
||||||
|
* @return the number of characters copied, or -1 if > Integer.MAX_VALUE
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 1.1
|
||||||
|
*/
|
||||||
|
public static int copy(final Reader input, final Writer output) throws IOException {
|
||||||
|
final long count = copyLarge(input, output);
|
||||||
|
if (count > Integer.MAX_VALUE) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return (int) count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies bytes from an <code>InputStream</code> to chars on a <code>Writer</code>
|
||||||
|
* using the specified character encoding.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedInputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* This method uses {@link java.io.InputStreamReader}.
|
||||||
|
* @param input the <code>InputStream</code> to read from
|
||||||
|
* @param output the <code>Writer</code> to write to
|
||||||
|
* @param inputEncoding the encoding to use for the input stream, null means platform
|
||||||
|
* default
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 2.3
|
||||||
|
*/
|
||||||
|
public static void copy(final InputStream input, final Writer output,
|
||||||
|
final Charset inputEncoding) throws IOException {
|
||||||
|
final InputStreamReader in = new InputStreamReader(input,
|
||||||
|
Charsets.toCharset(inputEncoding));
|
||||||
|
copy(in, output);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies bytes from an <code>InputStream</code> to an <code>OutputStream</code> using
|
||||||
|
* an internal buffer of the given size.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedInputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* @param input the <code>InputStream</code> to read from
|
||||||
|
* @param output the <code>OutputStream</code> to write to
|
||||||
|
* @param bufferSize the bufferSize used to copy from the input to the output
|
||||||
|
* @return the number of bytes copied
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 2.5
|
||||||
|
*/
|
||||||
|
public static long copy(final InputStream input, final OutputStream output,
|
||||||
|
final int bufferSize) throws IOException {
|
||||||
|
return copyLarge(input, output, new byte[bufferSize]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies chars from a large (over 2GB) <code>Reader</code> to a <code>Writer</code>.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedReader</code>.
|
||||||
|
* <p>
|
||||||
|
* The buffer size is given by {@link #DEFAULT_BUFFER_SIZE}.
|
||||||
|
* @param input the <code>Reader</code> to read from
|
||||||
|
* @param output the <code>Writer</code> to write to
|
||||||
|
* @return the number of characters copied
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 1.3
|
||||||
|
*/
|
||||||
|
public static long copyLarge(final Reader input, final Writer output)
|
||||||
|
throws IOException {
|
||||||
|
return copyLarge(input, output, new char[DEFAULT_BUFFER_SIZE]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies chars from a large (over 2GB) <code>Reader</code> to a <code>Writer</code>.
|
||||||
|
* <p>
|
||||||
|
* This method uses the provided buffer, so there is no need to use a
|
||||||
|
* <code>BufferedReader</code>.
|
||||||
|
* <p>
|
||||||
|
* @param input the <code>Reader</code> to read from
|
||||||
|
* @param output the <code>Writer</code> to write to
|
||||||
|
* @param buffer the buffer to be used for the copy
|
||||||
|
* @return the number of characters copied
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 2.2
|
||||||
|
*/
|
||||||
|
public static long copyLarge(final Reader input, final Writer output,
|
||||||
|
final char[] buffer) throws IOException {
|
||||||
|
long count = 0;
|
||||||
|
int n;
|
||||||
|
while (EOF != (n = input.read(buffer))) {
|
||||||
|
output.write(buffer, 0, n);
|
||||||
|
count += n;
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies bytes from a large (over 2GB) <code>InputStream</code> to an
|
||||||
|
* <code>OutputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* This method buffers the input internally, so there is no need to use a
|
||||||
|
* <code>BufferedInputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* The buffer size is given by {@link #DEFAULT_BUFFER_SIZE}.
|
||||||
|
* @param input the <code>InputStream</code> to read from
|
||||||
|
* @param output the <code>OutputStream</code> to write to
|
||||||
|
* @return the number of bytes copied
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 1.3
|
||||||
|
*/
|
||||||
|
public static long copyLarge(final InputStream input, final OutputStream output)
|
||||||
|
throws IOException {
|
||||||
|
return copy(input, output, DEFAULT_BUFFER_SIZE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copies bytes from a large (over 2GB) <code>InputStream</code> to an
|
||||||
|
* <code>OutputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* This method uses the provided buffer, so there is no need to use a
|
||||||
|
* <code>BufferedInputStream</code>.
|
||||||
|
* <p>
|
||||||
|
* @param input the <code>InputStream</code> to read from
|
||||||
|
* @param output the <code>OutputStream</code> to write to
|
||||||
|
* @param buffer the buffer to use for the copy
|
||||||
|
* @return the number of bytes copied
|
||||||
|
* @throws NullPointerException if the input or output is null
|
||||||
|
* @throws IOException if an I/O error occurs
|
||||||
|
* @since 2.2
|
||||||
|
*/
|
||||||
|
public static long copyLarge(final InputStream input, final OutputStream output,
|
||||||
|
final byte[] buffer) throws IOException {
|
||||||
|
long count = 0;
|
||||||
|
int n;
|
||||||
|
while (EOF != (n = input.read(buffer))) {
|
||||||
|
output.write(buffer, 0, n);
|
||||||
|
count += n;
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,152 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2019 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.commons.io;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.io.Writer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copy from apache commons-io.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">theonefx</a>
|
||||||
|
*/
|
||||||
|
public class StringBuilderWriter extends Writer implements Serializable {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -146927496096066153L;
|
||||||
|
|
||||||
|
private final StringBuilder builder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new {@link StringBuilder} instance with default capacity.
|
||||||
|
*/
|
||||||
|
public StringBuilderWriter() {
|
||||||
|
this.builder = new StringBuilder();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new {@link StringBuilder} instance with the specified capacity.
|
||||||
|
* @param capacity The initial capacity of the underlying {@link StringBuilder}
|
||||||
|
*/
|
||||||
|
public StringBuilderWriter(final int capacity) {
|
||||||
|
this.builder = new StringBuilder(capacity);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new instance with the specified {@link StringBuilder}.
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* If {@code builder} is null a new instance with default capacity will be created.
|
||||||
|
* </p>
|
||||||
|
* @param builder The String builder. May be null.
|
||||||
|
*/
|
||||||
|
public StringBuilderWriter(final StringBuilder builder) {
|
||||||
|
this.builder = builder != null ? builder : new StringBuilder();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appends a single character to this Writer.
|
||||||
|
* @param value The character to append
|
||||||
|
* @return This writer instance
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Writer append(final char value) {
|
||||||
|
builder.append(value);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appends a character sequence to this Writer.
|
||||||
|
* @param value The character to append
|
||||||
|
* @return This writer instance
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Writer append(final CharSequence value) {
|
||||||
|
builder.append(value);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appends a portion of a character sequence to the {@link StringBuilder}.
|
||||||
|
* @param value The character to append
|
||||||
|
* @param start The index of the first character
|
||||||
|
* @param end The index of the last character + 1
|
||||||
|
* @return This writer instance
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Writer append(final CharSequence value, final int start, final int end) {
|
||||||
|
builder.append(value, start, end);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closing this writer has no effect.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void close() {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flushing this writer has no effect.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void flush() {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes a String to the {@link StringBuilder}.
|
||||||
|
* @param value The value to write
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void write(final String value) {
|
||||||
|
if (value != null) {
|
||||||
|
builder.append(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes a portion of a character array to the {@link StringBuilder}.
|
||||||
|
* @param value The value to write
|
||||||
|
* @param offset The index of the first character
|
||||||
|
* @param length The number of characters to write
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void write(final char[] value, final int offset, final int length) {
|
||||||
|
if (value != null) {
|
||||||
|
builder.append(value, offset, length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the underlying builder.
|
||||||
|
* @return The underlying builder
|
||||||
|
*/
|
||||||
|
public StringBuilder getBuilder() {
|
||||||
|
return builder;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns {@link StringBuilder#toString()}.
|
||||||
|
* @return The contents of the String builder.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return builder.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,372 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2019 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.commons.lang;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* StringUtils. copy from apache common-lang3.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">theonefx</a>
|
||||||
|
*/
|
||||||
|
public final class StringUtils {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The empty String {@code ""}.
|
||||||
|
*
|
||||||
|
* @since 2.0
|
||||||
|
*/
|
||||||
|
public static final String EMPTY = "";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a failed index search.
|
||||||
|
* @since 2.1
|
||||||
|
*/
|
||||||
|
public static final int INDEX_NOT_FOUND = -1;
|
||||||
|
|
||||||
|
private StringUtils() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Checks if a CharSequence is empty ("") or null.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.isEmpty(null) = true
|
||||||
|
* StringUtils.isEmpty("") = true
|
||||||
|
* StringUtils.isEmpty(" ") = false
|
||||||
|
* StringUtils.isEmpty("bob") = false
|
||||||
|
* StringUtils.isEmpty(" bob ") = false
|
||||||
|
* </pre>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* NOTE: This method changed in Lang version 2.0. It no longer trims the CharSequence.
|
||||||
|
* That functionality is available in isBlank().
|
||||||
|
* </p>
|
||||||
|
* @param cs the CharSequence to check, may be null
|
||||||
|
* @return {@code true} if the CharSequence is empty or null
|
||||||
|
* @since 3.0 Changed signature from isEmpty(String) to isEmpty(CharSequence)
|
||||||
|
*/
|
||||||
|
public static boolean isEmpty(final CharSequence cs) {
|
||||||
|
return cs == null || cs.length() == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Checks if a CharSequence is not empty ("") and not null.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.isNotEmpty(null) = false
|
||||||
|
* StringUtils.isNotEmpty("") = false
|
||||||
|
* StringUtils.isNotEmpty(" ") = true
|
||||||
|
* StringUtils.isNotEmpty("bob") = true
|
||||||
|
* StringUtils.isNotEmpty(" bob ") = true
|
||||||
|
* </pre>
|
||||||
|
* @param cs the CharSequence to check, may be null
|
||||||
|
* @return {@code true} if the CharSequence is not empty and not null
|
||||||
|
* @since 3.0 Changed signature from isNotEmpty(String) to isNotEmpty(CharSequence)
|
||||||
|
*/
|
||||||
|
public static boolean isNotEmpty(final CharSequence cs) {
|
||||||
|
return !isEmpty(cs);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Checks if a CharSequence is whitespace, empty ("") or null.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.isBlank(null) = true
|
||||||
|
* StringUtils.isBlank("") = true
|
||||||
|
* StringUtils.isBlank(" ") = true
|
||||||
|
* StringUtils.isBlank("bob") = false
|
||||||
|
* StringUtils.isBlank(" bob ") = false
|
||||||
|
* </pre>
|
||||||
|
* @param cs the CharSequence to check, may be null
|
||||||
|
* @return {@code true} if the CharSequence is null, empty or whitespace
|
||||||
|
*/
|
||||||
|
public static boolean isBlank(final CharSequence cs) {
|
||||||
|
if (cs == null || cs.length() == 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
int strLen = cs.length();
|
||||||
|
for (int i = 0; i < strLen; i++) {
|
||||||
|
if (!Character.isWhitespace(cs.charAt(i))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Checks if a CharSequence is not empty (""), not null and not whitespace only.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* Whitespace is defined by {@link Character#isWhitespace(char)}.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.isNotBlank(null) = false
|
||||||
|
* StringUtils.isNotBlank("") = false
|
||||||
|
* StringUtils.isNotBlank(" ") = false
|
||||||
|
* StringUtils.isNotBlank("bob") = true
|
||||||
|
* StringUtils.isNotBlank(" bob ") = true
|
||||||
|
* </pre>
|
||||||
|
* @param cs the CharSequence to check, may be null
|
||||||
|
* @return {@code true} if the CharSequence is not empty and not null and not
|
||||||
|
* whitespace only
|
||||||
|
* @since 2.0
|
||||||
|
* @since 3.0 Changed signature from isNotBlank(String) to isNotBlank(CharSequence)
|
||||||
|
*/
|
||||||
|
public static boolean isNotBlank(final CharSequence cs) {
|
||||||
|
return !isBlank(cs);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trim
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Removes control characters (char <= 32) from both ends of this String, handling
|
||||||
|
* {@code null} by returning {@code null}.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* The String is trimmed using {@link String#trim()}. Trim removes start and end
|
||||||
|
* characters <= 32.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.trim(null) = null
|
||||||
|
* StringUtils.trim("") = ""
|
||||||
|
* StringUtils.trim(" ") = ""
|
||||||
|
* StringUtils.trim("abc") = "abc"
|
||||||
|
* StringUtils.trim(" abc ") = "abc"
|
||||||
|
* </pre>
|
||||||
|
* @param str the String to be trimmed, may be null
|
||||||
|
* @return the trimmed string, {@code null} if null String input
|
||||||
|
*/
|
||||||
|
public static String trim(final String str) {
|
||||||
|
return str == null ? null : str.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equals
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Compares two CharSequences, returning {@code true} if they represent equal
|
||||||
|
* sequences of characters.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* {@code null}s are handled without exceptions. Two {@code null} references are
|
||||||
|
* considered to be equal. The comparison is case sensitive.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.equals(null, null) = true
|
||||||
|
* StringUtils.equals(null, "abc") = false
|
||||||
|
* StringUtils.equals("abc", null) = false
|
||||||
|
* StringUtils.equals("abc", "abc") = true
|
||||||
|
* StringUtils.equals("abc", "ABC") = false
|
||||||
|
* </pre>
|
||||||
|
* @param cs1 the first CharSequence, may be {@code null}
|
||||||
|
* @param cs2 the second CharSequence, may be {@code null}
|
||||||
|
* @return {@code true} if the CharSequences are equal (case-sensitive), or both
|
||||||
|
* {@code null}
|
||||||
|
* @see Object#equals(Object)
|
||||||
|
*/
|
||||||
|
public static boolean equals(final CharSequence cs1, final CharSequence cs2) {
|
||||||
|
if (cs1 == cs2) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (cs1 == null || cs2 == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (cs1 instanceof String && cs2 instanceof String) {
|
||||||
|
return cs1.equals(cs2);
|
||||||
|
}
|
||||||
|
return StringUtils.regionMatches(cs1, false, 0, cs2, 0,
|
||||||
|
Math.max(cs1.length(), cs2.length()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Green implementation of regionMatches.
|
||||||
|
* @param cs the {@code CharSequence} to be processed
|
||||||
|
* @param ignoreCase whether or not to be case insensitive
|
||||||
|
* @param thisStart the index to start on the {@code cs} CharSequence
|
||||||
|
* @param substring the {@code CharSequence} to be looked for
|
||||||
|
* @param start the index to start on the {@code substring} CharSequence
|
||||||
|
* @param length character length of the region
|
||||||
|
* @return whether the region matched
|
||||||
|
*/
|
||||||
|
public static boolean regionMatches(final CharSequence cs, final boolean ignoreCase,
|
||||||
|
final int thisStart, final CharSequence substring, final int start,
|
||||||
|
final int length) {
|
||||||
|
if (cs instanceof String && substring instanceof String) {
|
||||||
|
return ((String) cs).regionMatches(ignoreCase, thisStart, (String) substring,
|
||||||
|
start, length);
|
||||||
|
}
|
||||||
|
int index1 = thisStart;
|
||||||
|
int index2 = start;
|
||||||
|
int tmpLen = length;
|
||||||
|
|
||||||
|
while (tmpLen-- > 0) {
|
||||||
|
final char c1 = cs.charAt(index1++);
|
||||||
|
final char c2 = substring.charAt(index2++);
|
||||||
|
|
||||||
|
if (c1 == c2) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ignoreCase) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// The same check as in String.regionMatches():
|
||||||
|
if (Character.toUpperCase(c1) != Character.toUpperCase(c2)
|
||||||
|
&& Character.toLowerCase(c1) != Character.toLowerCase(c2)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Gets the substring after the first occurrence of a separator. The separator is not
|
||||||
|
* returned.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* A <code>null</code> string input will return <code>null</code>. An empty ("")
|
||||||
|
* string input will return the empty string. A <code>null</code> separator will
|
||||||
|
* return the empty string if the input string is not <code>null</code>.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* If nothing is found, the empty string is returned.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.substringAfter(null, *) = null
|
||||||
|
* StringUtils.substringAfter("", *) = ""
|
||||||
|
* StringUtils.substringAfter(*, null) = ""
|
||||||
|
* StringUtils.substringAfter("abc", "a") = "bc"
|
||||||
|
* StringUtils.substringAfter("abcba", "b") = "cba"
|
||||||
|
* StringUtils.substringAfter("abc", "c") = ""
|
||||||
|
* StringUtils.substringAfter("abc", "d") = ""
|
||||||
|
* StringUtils.substringAfter("abc", "") = "abc"
|
||||||
|
* </pre>
|
||||||
|
* @param str the String to get a substring from, may be null
|
||||||
|
* @param separator the String to search for, may be null
|
||||||
|
* @return the substring after the first occurrence of the separator,
|
||||||
|
* <code>null</code> if null String input
|
||||||
|
* @since 2.0
|
||||||
|
*/
|
||||||
|
public static String substringAfter(String str, String separator) {
|
||||||
|
if (isEmpty(str)) {
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
if (separator == null) {
|
||||||
|
return EMPTY;
|
||||||
|
}
|
||||||
|
int pos = str.indexOf(separator);
|
||||||
|
if (pos == INDEX_NOT_FOUND) {
|
||||||
|
return EMPTY;
|
||||||
|
}
|
||||||
|
return str.substring(pos + separator.length());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Substring between
|
||||||
|
// -----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Gets the String that is nested in between two instances of the same String.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* A <code>null</code> input String returns <code>null</code>. A <code>null</code> tag
|
||||||
|
* returns <code>null</code>.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.substringBetween(null, *) = null
|
||||||
|
* StringUtils.substringBetween("", "") = ""
|
||||||
|
* StringUtils.substringBetween("", "tag") = null
|
||||||
|
* StringUtils.substringBetween("tagabctag", null) = null
|
||||||
|
* StringUtils.substringBetween("tagabctag", "") = ""
|
||||||
|
* StringUtils.substringBetween("tagabctag", "tag") = "abc"
|
||||||
|
* </pre>
|
||||||
|
* @param str the String containing the substring, may be null
|
||||||
|
* @param tag the String before and after the substring, may be null
|
||||||
|
* @return the substring, <code>null</code> if no match
|
||||||
|
* @since 2.0
|
||||||
|
*/
|
||||||
|
public static String substringBetween(String str, String tag) {
|
||||||
|
return substringBetween(str, tag, tag);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <p>
|
||||||
|
* Gets the String that is nested in between two Strings. Only the first match is
|
||||||
|
* returned.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <p>
|
||||||
|
* A <code>null</code> input String returns <code>null</code>. A <code>null</code>
|
||||||
|
* open/close returns <code>null</code> (no match). An empty ("") open and close
|
||||||
|
* returns an empty string.
|
||||||
|
* </p>
|
||||||
|
*
|
||||||
|
* <pre>
|
||||||
|
* StringUtils.substringBetween("wx[b]yz", "[", "]") = "b"
|
||||||
|
* StringUtils.substringBetween(null, *, *) = null
|
||||||
|
* StringUtils.substringBetween(*, null, *) = null
|
||||||
|
* StringUtils.substringBetween(*, *, null) = null
|
||||||
|
* StringUtils.substringBetween("", "", "") = ""
|
||||||
|
* StringUtils.substringBetween("", "", "]") = null
|
||||||
|
* StringUtils.substringBetween("", "[", "]") = null
|
||||||
|
* StringUtils.substringBetween("yabcz", "", "") = ""
|
||||||
|
* StringUtils.substringBetween("yabcz", "y", "z") = "abc"
|
||||||
|
* StringUtils.substringBetween("yabczyabcz", "y", "z") = "abc"
|
||||||
|
* </pre>
|
||||||
|
* @param str the String containing the substring, may be null
|
||||||
|
* @param open the String before the substring, may be null
|
||||||
|
* @param close the String after the substring, may be null
|
||||||
|
* @return the substring, <code>null</code> if no match
|
||||||
|
* @since 2.0
|
||||||
|
*/
|
||||||
|
public static String substringBetween(String str, String open, String close) {
|
||||||
|
if (str == null || open == null || close == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
int start = str.indexOf(open);
|
||||||
|
if (start != INDEX_NOT_FOUND) {
|
||||||
|
int end = str.indexOf(close, start + open.length());
|
||||||
|
if (end != INDEX_NOT_FOUND) {
|
||||||
|
return str.substring(start + open.length(), end);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -15,6 +15,10 @@
|
|||||||
<name>Spring Cloud Alibaba Sentinel DataSource</name>
|
<name>Spring Cloud Alibaba Sentinel DataSource</name>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
<artifactId>spring-cloud-alibaba-commons</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<!--spring boot -->
|
<!--spring boot -->
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@@ -16,9 +16,9 @@
|
|||||||
|
|
||||||
package com.alibaba.cloud.sentinel.datasource.factorybean;
|
package com.alibaba.cloud.sentinel.datasource.factorybean;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.commons.lang.StringUtils;
|
||||||
import com.alibaba.csp.sentinel.datasource.Converter;
|
import com.alibaba.csp.sentinel.datasource.Converter;
|
||||||
import com.alibaba.csp.sentinel.datasource.zookeeper.ZookeeperDataSource;
|
import com.alibaba.csp.sentinel.datasource.zookeeper.ZookeeperDataSource;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
|
||||||
|
|
||||||
import org.springframework.beans.factory.FactoryBean;
|
import org.springframework.beans.factory.FactoryBean;
|
||||||
|
|
||||||
|
@@ -19,13 +19,13 @@ package com.alibaba.cloud.sentinel.datasource;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.commons.io.FileUtils;
|
||||||
import com.alibaba.cloud.sentinel.datasource.converter.JsonConverter;
|
import com.alibaba.cloud.sentinel.datasource.converter.JsonConverter;
|
||||||
import com.alibaba.cloud.sentinel.datasource.converter.XmlConverter;
|
import com.alibaba.cloud.sentinel.datasource.converter.XmlConverter;
|
||||||
import com.alibaba.csp.sentinel.slots.block.RuleConstant;
|
import com.alibaba.csp.sentinel.slots.block.RuleConstant;
|
||||||
import com.alibaba.csp.sentinel.slots.block.flow.FlowRule;
|
import com.alibaba.csp.sentinel.slots.block.flow.FlowRule;
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
|
import com.fasterxml.jackson.dataformat.xml.XmlMapper;
|
||||||
import org.apache.commons.io.FileUtils;
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import org.springframework.util.ResourceUtils;
|
import org.springframework.util.ResourceUtils;
|
||||||
|
@@ -18,8 +18,6 @@ package com.alibaba.cloud.nacos;
|
|||||||
|
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
import javax.annotation.PreDestroy;
|
|
||||||
|
|
||||||
import com.alibaba.cloud.nacos.diagnostics.analyzer.NacosConnectionFailureException;
|
import com.alibaba.cloud.nacos.diagnostics.analyzer.NacosConnectionFailureException;
|
||||||
import com.alibaba.nacos.api.NacosFactory;
|
import com.alibaba.nacos.api.NacosFactory;
|
||||||
import com.alibaba.nacos.api.config.ConfigService;
|
import com.alibaba.nacos.api.config.ConfigService;
|
||||||
@@ -75,14 +73,6 @@ public class NacosConfigManager {
|
|||||||
return service;
|
return service;
|
||||||
}
|
}
|
||||||
|
|
||||||
@PreDestroy
|
|
||||||
public void destroy() throws NacosException {
|
|
||||||
if (service != null) {
|
|
||||||
service.shutDown();
|
|
||||||
service = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public NacosConfigProperties getNacosConfigProperties() {
|
public NacosConfigProperties getNacosConfigProperties() {
|
||||||
return nacosConfigProperties;
|
return nacosConfigProperties;
|
||||||
}
|
}
|
||||||
|
@@ -30,8 +30,6 @@ import org.springframework.core.env.PropertySource;
|
|||||||
import org.springframework.core.io.Resource;
|
import org.springframework.core.io.Resource;
|
||||||
import org.springframework.util.StringUtils;
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
import static com.alibaba.cloud.nacos.parser.NacosDataParserHandler.DOT;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Nacos-specific loader, If need to support other methods of parsing,you need to do the
|
* Nacos-specific loader, If need to support other methods of parsing,you need to do the
|
||||||
* following steps:
|
* following steps:
|
||||||
@@ -47,6 +45,11 @@ import static com.alibaba.cloud.nacos.parser.NacosDataParserHandler.DOT;
|
|||||||
*/
|
*/
|
||||||
public abstract class AbstractPropertySourceLoader implements PropertySourceLoader {
|
public abstract class AbstractPropertySourceLoader implements PropertySourceLoader {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* symbol: dot.
|
||||||
|
*/
|
||||||
|
static final String DOT = ".";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Prevent interference with other loaders.Nacos-specific loader, unless the reload
|
* Prevent interference with other loaders.Nacos-specific loader, unless the reload
|
||||||
* changes it.
|
* changes it.
|
||||||
|
@@ -25,7 +25,10 @@ import java.util.Map;
|
|||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.nacos.utils.NacosConfigUtils;
|
||||||
|
|
||||||
import org.springframework.boot.env.OriginTrackedMapPropertySource;
|
import org.springframework.boot.env.OriginTrackedMapPropertySource;
|
||||||
|
import org.springframework.boot.env.PropertiesPropertySourceLoader;
|
||||||
import org.springframework.boot.env.PropertySourceLoader;
|
import org.springframework.boot.env.PropertySourceLoader;
|
||||||
import org.springframework.core.env.EnumerablePropertySource;
|
import org.springframework.core.env.EnumerablePropertySource;
|
||||||
import org.springframework.core.env.PropertySource;
|
import org.springframework.core.env.PropertySource;
|
||||||
@@ -33,25 +36,17 @@ import org.springframework.core.io.support.SpringFactoriesLoader;
|
|||||||
import org.springframework.util.CollectionUtils;
|
import org.springframework.util.CollectionUtils;
|
||||||
import org.springframework.util.StringUtils;
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
|
import static com.alibaba.cloud.nacos.parser.AbstractPropertySourceLoader.DOT;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author zkz
|
* @author zkz
|
||||||
*/
|
*/
|
||||||
public final class NacosDataParserHandler {
|
public final class NacosDataParserHandler {
|
||||||
|
|
||||||
/**
|
|
||||||
* symbol: dot.
|
|
||||||
*/
|
|
||||||
public static final String DOT = ".";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* constant.
|
|
||||||
*/
|
|
||||||
public static final String VALUE = "value";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* default extension.
|
* default extension.
|
||||||
*/
|
*/
|
||||||
public static final String DEFAULT_EXTENSION = "properties";
|
private static final String DEFAULT_EXTENSION = "properties";
|
||||||
|
|
||||||
private static List<PropertySourceLoader> propertySourceLoaders;
|
private static List<PropertySourceLoader> propertySourceLoaders;
|
||||||
|
|
||||||
@@ -80,8 +75,18 @@ public final class NacosDataParserHandler {
|
|||||||
if (!canLoadFileExtension(propertySourceLoader, extension)) {
|
if (!canLoadFileExtension(propertySourceLoader, extension)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
NacosByteArrayResource nacosByteArrayResource = new NacosByteArrayResource(
|
NacosByteArrayResource nacosByteArrayResource;
|
||||||
|
if (propertySourceLoader instanceof PropertiesPropertySourceLoader) {
|
||||||
|
// PropertiesPropertySourceLoader internal is to use the ISO_8859_1,
|
||||||
|
// the Chinese will be garbled, needs to transform into unicode.
|
||||||
|
nacosByteArrayResource = new NacosByteArrayResource(
|
||||||
|
NacosConfigUtils.selectiveConvertUnicode(configValue).getBytes(),
|
||||||
|
configName);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
nacosByteArrayResource = new NacosByteArrayResource(
|
||||||
configValue.getBytes(), configName);
|
configValue.getBytes(), configName);
|
||||||
|
}
|
||||||
nacosByteArrayResource.setFilename(getFileName(configName, extension));
|
nacosByteArrayResource.setFilename(getFileName(configName, extension));
|
||||||
List<PropertySource<?>> propertySourceList = propertySourceLoader
|
List<PropertySource<?>> propertySourceList = propertySourceLoader
|
||||||
.load(configName, nacosByteArrayResource);
|
.load(configName, nacosByteArrayResource);
|
||||||
|
@@ -28,14 +28,16 @@ import org.springframework.boot.env.OriginTrackedMapPropertySource;
|
|||||||
import org.springframework.core.env.PropertySource;
|
import org.springframework.core.env.PropertySource;
|
||||||
import org.springframework.core.io.Resource;
|
import org.springframework.core.io.Resource;
|
||||||
|
|
||||||
import static com.alibaba.cloud.nacos.parser.NacosDataParserHandler.DOT;
|
|
||||||
import static com.alibaba.cloud.nacos.parser.NacosDataParserHandler.VALUE;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author zkz
|
* @author zkz
|
||||||
*/
|
*/
|
||||||
public class NacosJsonPropertySourceLoader extends AbstractPropertySourceLoader {
|
public class NacosJsonPropertySourceLoader extends AbstractPropertySourceLoader {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* constant.
|
||||||
|
*/
|
||||||
|
private static final String VALUE = "value";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the file extensions that the loader supports (excluding the '.').
|
* Returns the file extensions that the loader supports (excluding the '.').
|
||||||
* @return the file extensions
|
* @return the file extensions
|
||||||
|
@@ -127,8 +127,7 @@ public class NacosXmlPropertySourceLoader extends AbstractPropertySourceLoader
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
String key = StringUtils.isEmpty(parentKey) ? name
|
String key = StringUtils.isEmpty(parentKey) ? name : parentKey + DOT + name;
|
||||||
: parentKey + NacosDataParserHandler.DOT + name;
|
|
||||||
NamedNodeMap nodeMap = node.getAttributes();
|
NamedNodeMap nodeMap = node.getAttributes();
|
||||||
parseNodeAttr(nodeMap, map, key);
|
parseNodeAttr(nodeMap, map, key);
|
||||||
if (node.getNodeType() == Node.ELEMENT_NODE && node.hasChildNodes()) {
|
if (node.getNodeType() == Node.ELEMENT_NODE && node.hasChildNodes()) {
|
||||||
@@ -159,8 +158,8 @@ public class NacosXmlPropertySourceLoader extends AbstractPropertySourceLoader
|
|||||||
if (StringUtils.isEmpty(node.getNodeValue())) {
|
if (StringUtils.isEmpty(node.getNodeValue())) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
map.put(String.join(NacosDataParserHandler.DOT, parentKey,
|
map.put(String.join(DOT, parentKey, node.getNodeName()),
|
||||||
node.getNodeName()), node.getNodeValue());
|
node.getNodeValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,71 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2018 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.nacos.utils;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author zkzlx
|
||||||
|
*/
|
||||||
|
public final class NacosConfigUtils {
|
||||||
|
|
||||||
|
private NacosConfigUtils() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert Chinese characters to Unicode.
|
||||||
|
* @param configValue value of config
|
||||||
|
* @return new string
|
||||||
|
*/
|
||||||
|
public static String selectiveConvertUnicode(String configValue) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
char[] chars = configValue.toCharArray();
|
||||||
|
for (char aChar : chars) {
|
||||||
|
if (isBaseLetter(aChar)) {
|
||||||
|
sb.append(aChar);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sb.append(String.format("\\u%04x", (int) aChar));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* char is base latin or whitespace?
|
||||||
|
* @param ch a character
|
||||||
|
* @return true or false
|
||||||
|
*/
|
||||||
|
public static boolean isBaseLetter(char ch) {
|
||||||
|
Character.UnicodeBlock ub = Character.UnicodeBlock.of(ch);
|
||||||
|
return ub == Character.UnicodeBlock.BASIC_LATIN || Character.isWhitespace(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* char is chinese?
|
||||||
|
* @param c a character
|
||||||
|
* @return true or false
|
||||||
|
*/
|
||||||
|
public static boolean isChinese(char c) {
|
||||||
|
Character.UnicodeBlock ub = Character.UnicodeBlock.of(c);
|
||||||
|
return ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS
|
||||||
|
|| ub == Character.UnicodeBlock.CJK_COMPATIBILITY_IDEOGRAPHS
|
||||||
|
|| ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS_EXTENSION_A
|
||||||
|
|| ub == Character.UnicodeBlock.GENERAL_PUNCTUATION
|
||||||
|
|| ub == Character.UnicodeBlock.CJK_SYMBOLS_AND_PUNCTUATION
|
||||||
|
|| ub == Character.UnicodeBlock.HALFWIDTH_AND_FULLWIDTH_FORMS;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -14,6 +14,11 @@
|
|||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.alibaba.cloud</groupId>
|
||||||
|
<artifactId>spring-cloud-alibaba-commons</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-actuator</artifactId>
|
<artifactId>spring-boot-actuator</artifactId>
|
||||||
|
@@ -24,6 +24,7 @@ import java.util.Map;
|
|||||||
import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
|
import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
|
||||||
import com.alibaba.cloud.nacos.NacosServiceManager;
|
import com.alibaba.cloud.nacos.NacosServiceManager;
|
||||||
import com.alibaba.nacos.api.naming.NamingService;
|
import com.alibaba.nacos.api.naming.NamingService;
|
||||||
|
import com.alibaba.nacos.api.naming.pojo.Instance;
|
||||||
import com.alibaba.nacos.api.naming.pojo.ServiceInfo;
|
import com.alibaba.nacos.api.naming.pojo.ServiceInfo;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
@@ -66,6 +67,11 @@ public class NacosDiscoveryEndpoint {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
subscribe = namingService.getSubscribeServices();
|
subscribe = namingService.getSubscribeServices();
|
||||||
|
for (ServiceInfo serviceInfo : subscribe) {
|
||||||
|
List<Instance> instances = namingService.getAllInstances(
|
||||||
|
serviceInfo.getName(), serviceInfo.getGroupName());
|
||||||
|
serviceInfo.setHosts(instances);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e) {
|
catch (Exception e) {
|
||||||
log.error("get subscribe services from nacos fail,", e);
|
log.error("get subscribe services from nacos fail,", e);
|
||||||
|
@@ -20,6 +20,7 @@ import java.util.List;
|
|||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.commons.lang.StringUtils;
|
||||||
import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
|
import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
|
||||||
import com.alibaba.cloud.nacos.NacosServiceManager;
|
import com.alibaba.cloud.nacos.NacosServiceManager;
|
||||||
import com.alibaba.nacos.api.naming.NamingService;
|
import com.alibaba.nacos.api.naming.NamingService;
|
||||||
@@ -28,7 +29,6 @@ import com.netflix.client.config.IClientConfig;
|
|||||||
import com.netflix.loadbalancer.AbstractLoadBalancerRule;
|
import com.netflix.loadbalancer.AbstractLoadBalancerRule;
|
||||||
import com.netflix.loadbalancer.DynamicServerListLoadBalancer;
|
import com.netflix.loadbalancer.DynamicServerListLoadBalancer;
|
||||||
import com.netflix.loadbalancer.Server;
|
import com.netflix.loadbalancer.Server;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
@@ -17,8 +17,8 @@
|
|||||||
package com.alibaba.cloud.seata.feign;
|
package com.alibaba.cloud.seata.feign;
|
||||||
|
|
||||||
import feign.Client;
|
import feign.Client;
|
||||||
import org.apache.commons.logging.Log;
|
import org.slf4j.Logger;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import org.springframework.beans.factory.BeanFactory;
|
import org.springframework.beans.factory.BeanFactory;
|
||||||
import org.springframework.cloud.loadbalancer.blocking.client.BlockingLoadBalancerClient;
|
import org.springframework.cloud.loadbalancer.blocking.client.BlockingLoadBalancerClient;
|
||||||
@@ -32,7 +32,8 @@ import org.springframework.cloud.openfeign.ribbon.LoadBalancerFeignClient;
|
|||||||
*/
|
*/
|
||||||
public class SeataFeignObjectWrapper {
|
public class SeataFeignObjectWrapper {
|
||||||
|
|
||||||
private static final Log LOG = LogFactory.getLog(SeataFeignObjectWrapper.class);
|
private static final Logger LOG = LoggerFactory
|
||||||
|
.getLogger(SeataFeignObjectWrapper.class);
|
||||||
|
|
||||||
private final BeanFactory beanFactory;
|
private final BeanFactory beanFactory;
|
||||||
|
|
||||||
|
@@ -16,10 +16,11 @@
|
|||||||
|
|
||||||
package com.alibaba.cloud.dubbo.actuate;
|
package com.alibaba.cloud.dubbo.actuate;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.dubbo.actuate.endpoint.DubboDiscoveryEndpoint;
|
||||||
|
import com.alibaba.cloud.dubbo.actuate.endpoint.DubboExportedURLsEndpoint;
|
||||||
import com.alibaba.cloud.dubbo.actuate.endpoint.DubboRestMetadataEndpoint;
|
import com.alibaba.cloud.dubbo.actuate.endpoint.DubboRestMetadataEndpoint;
|
||||||
|
|
||||||
import org.springframework.boot.actuate.autoconfigure.endpoint.condition.ConditionalOnAvailableEndpoint;
|
import org.springframework.boot.actuate.autoconfigure.endpoint.condition.ConditionalOnAvailableEndpoint;
|
||||||
import org.springframework.boot.actuate.autoconfigure.web.ManagementContextConfiguration;
|
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
@@ -34,7 +35,7 @@ import org.springframework.context.annotation.PropertySource;
|
|||||||
@ConditionalOnClass(
|
@ConditionalOnClass(
|
||||||
name = "org.springframework.boot.actuate.endpoint.annotation.Endpoint")
|
name = "org.springframework.boot.actuate.endpoint.annotation.Endpoint")
|
||||||
@PropertySource("classpath:/META-INF/dubbo/default/actuator-endpoints.properties")
|
@PropertySource("classpath:/META-INF/dubbo/default/actuator-endpoints.properties")
|
||||||
@ManagementContextConfiguration
|
@Configuration(proxyBeanMethods = false)
|
||||||
public class DubboMetadataEndpointAutoConfiguration {
|
public class DubboMetadataEndpointAutoConfiguration {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@@ -44,4 +45,19 @@ public class DubboMetadataEndpointAutoConfiguration {
|
|||||||
return new DubboRestMetadataEndpoint();
|
return new DubboRestMetadataEndpoint();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnMissingBean
|
||||||
|
@ConditionalOnAvailableEndpoint
|
||||||
|
public DubboDiscoveryEndpoint dubboDiscoveryEndpoint() {
|
||||||
|
|
||||||
|
return new DubboDiscoveryEndpoint();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnMissingBean
|
||||||
|
@ConditionalOnAvailableEndpoint
|
||||||
|
public DubboExportedURLsEndpoint dubboServiceMetadataEndpoint() {
|
||||||
|
return new DubboExportedURLsEndpoint();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,93 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2018 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.dubbo.actuate.endpoint;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.dubbo.registry.DubboCloudRegistry;
|
||||||
|
import com.alibaba.cloud.dubbo.registry.SpringCloudRegistryFactory;
|
||||||
|
import org.apache.dubbo.common.URL;
|
||||||
|
import org.apache.dubbo.registry.NotifyListener;
|
||||||
|
import org.apache.dubbo.registry.integration.RegistryDirectory;
|
||||||
|
import org.apache.dubbo.rpc.Invoker;
|
||||||
|
|
||||||
|
import org.springframework.boot.actuate.endpoint.annotation.Endpoint;
|
||||||
|
import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
|
||||||
|
|
||||||
|
import static org.apache.dubbo.common.constants.CommonConstants.CONSUMER_SIDE;
|
||||||
|
import static org.apache.dubbo.common.constants.CommonConstants.SIDE_KEY;
|
||||||
|
import static org.springframework.http.MediaType.APPLICATION_JSON_VALUE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dubbo Registry Directory Metadata {@link DubboCloudRegistry}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">Theonefx</a>
|
||||||
|
*/
|
||||||
|
@Endpoint(id = "dubboRegistryDirectory")
|
||||||
|
public class DubboDiscoveryEndpoint {
|
||||||
|
|
||||||
|
@ReadOperation(produces = APPLICATION_JSON_VALUE)
|
||||||
|
public Object get() {
|
||||||
|
DubboCloudRegistry registry = (DubboCloudRegistry) SpringCloudRegistryFactory
|
||||||
|
.getRegistries().stream().filter(o -> o instanceof DubboCloudRegistry)
|
||||||
|
.findFirst().orElse(null);
|
||||||
|
|
||||||
|
if (registry == null) {
|
||||||
|
return Collections.emptyMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<URL, Set<NotifyListener>> subscribeMap = registry.getSubscribed();
|
||||||
|
|
||||||
|
Map<String, List<Map<String, Object>>> result = new HashMap<>();
|
||||||
|
subscribeMap.forEach((url, listeners) -> {
|
||||||
|
String side = url.getParameter(SIDE_KEY);
|
||||||
|
if (!CONSUMER_SIDE.equals(side)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Map<String, Object>> pairs = result.computeIfAbsent(url.getServiceKey(),
|
||||||
|
o -> new ArrayList<>());
|
||||||
|
|
||||||
|
Map<String, Object> pair = new HashMap<>();
|
||||||
|
List<String> invokerServices = new ArrayList<>();
|
||||||
|
for (NotifyListener listener : listeners) {
|
||||||
|
if (!(listener instanceof RegistryDirectory)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
RegistryDirectory<?> directory = (RegistryDirectory<?>) listener;
|
||||||
|
List<? extends Invoker<?>> invokers = directory.getAllInvokers();
|
||||||
|
if (invokers == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
invokerServices.addAll(invokers.stream().map(Invoker::getUrl)
|
||||||
|
.map(URL::toServiceString).collect(Collectors.toList()));
|
||||||
|
}
|
||||||
|
pair.put("invokers", invokerServices);
|
||||||
|
pair.put("subscribeUrl", url.toMap());
|
||||||
|
|
||||||
|
pairs.add(pair);
|
||||||
|
});
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2013-2018 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.alibaba.cloud.dubbo.actuate.endpoint;
|
||||||
|
|
||||||
|
import com.alibaba.cloud.dubbo.service.DubboMetadataService;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.boot.actuate.endpoint.annotation.Endpoint;
|
||||||
|
import org.springframework.boot.actuate.endpoint.annotation.ReadOperation;
|
||||||
|
|
||||||
|
import static org.springframework.http.MediaType.APPLICATION_JSON_UTF8_VALUE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dubbo exported URLs.
|
||||||
|
* {@link org.springframework.boot.actuate.endpoint.annotation.Endpoint}.
|
||||||
|
*
|
||||||
|
* @author <a href="mailto:chenxilzx1@gmail.com">Theonefx</a>
|
||||||
|
*/
|
||||||
|
@Endpoint(id = "dubboExportedURLs")
|
||||||
|
public class DubboExportedURLsEndpoint {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private DubboMetadataService dubboMetadataService;
|
||||||
|
|
||||||
|
@ReadOperation(produces = APPLICATION_JSON_UTF8_VALUE)
|
||||||
|
public Object get() {
|
||||||
|
return dubboMetadataService.getAllExportedURLs();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -44,6 +44,8 @@ import org.springframework.cloud.client.ServiceInstance;
|
|||||||
import org.springframework.cloud.client.discovery.DiscoveryClient;
|
import org.springframework.cloud.client.discovery.DiscoveryClient;
|
||||||
import org.springframework.context.ApplicationListener;
|
import org.springframework.context.ApplicationListener;
|
||||||
import org.springframework.context.ConfigurableApplicationContext;
|
import org.springframework.context.ConfigurableApplicationContext;
|
||||||
|
import org.springframework.core.Ordered;
|
||||||
|
import org.springframework.core.annotation.Order;
|
||||||
import org.springframework.util.CollectionUtils;
|
import org.springframework.util.CollectionUtils;
|
||||||
|
|
||||||
import static java.lang.String.format;
|
import static java.lang.String.format;
|
||||||
@@ -184,8 +186,15 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
subscribeURLs(url, getServices(url), listener);
|
subscribeURLs(url, getServices(url), listener);
|
||||||
|
|
||||||
// Async subscription
|
// Async subscription
|
||||||
registerServiceInstancesChangedListener(url, event -> {
|
registerServiceInstancesChangedListener(url,
|
||||||
|
|
||||||
|
new ApplicationListener<ServiceInstancesChangedEvent>() {
|
||||||
|
|
||||||
|
private final URL url2subscribe = url;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Order
|
||||||
|
public void onApplicationEvent(ServiceInstancesChangedEvent event) {
|
||||||
Set<String> serviceNames = getServices(url);
|
Set<String> serviceNames = getServices(url);
|
||||||
|
|
||||||
String serviceName = event.getServiceName();
|
String serviceName = event.getServiceName();
|
||||||
@@ -193,6 +202,13 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
if (serviceNames.contains(serviceName)) {
|
if (serviceNames.contains(serviceName)) {
|
||||||
subscribeURLs(url, serviceNames, listener);
|
subscribeURLs(url, serviceNames, listener);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "ServiceInstancesChangedEventListener:"
|
||||||
|
+ url.getServiceKey();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -375,12 +391,12 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
// Add the EMPTY_PROTOCOL URL
|
// Add the EMPTY_PROTOCOL URL
|
||||||
subscribedURLs.add(emptyURL(url));
|
subscribedURLs.add(emptyURL(url));
|
||||||
|
|
||||||
if (isDubboMetadataServiceURL(url)) {
|
// if (isDubboMetadataServiceURL(url)) {
|
||||||
// if meta service change, and serviceInstances is zero, will clean up
|
// if meta service change, and serviceInstances is zero, will clean up
|
||||||
// information about this client
|
// information about this client
|
||||||
String serviceName = url.getParameter(GROUP_KEY);
|
// String serviceName = url.getParameter(GROUP_KEY);
|
||||||
repository.removeMetadataAndInitializedService(serviceName, url);
|
// repository.removeMetadataAndInitializedService(serviceName, url);
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
@@ -415,7 +431,7 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private String generateId(URL url) {
|
private String generateId(URL url) {
|
||||||
return url.getServiceKey();
|
return url.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
private URL emptyURL(URL url) {
|
private URL emptyURL(URL url) {
|
||||||
@@ -450,8 +466,15 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
|
|
||||||
// Sync subscription
|
// Sync subscription
|
||||||
if (containsProviderCategory(subscribedURL)) {
|
if (containsProviderCategory(subscribedURL)) {
|
||||||
registerServiceInstancesChangedListener(subscribedURL, event -> {
|
registerServiceInstancesChangedListener(subscribedURL,
|
||||||
|
new ApplicationListener<ServiceInstancesChangedEvent>() {
|
||||||
|
|
||||||
|
private final URL url2subscribe = subscribedURL;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Order(Ordered.LOWEST_PRECEDENCE - 1)
|
||||||
|
public void onApplicationEvent(
|
||||||
|
ServiceInstancesChangedEvent event) {
|
||||||
String sourceServiceName = event.getServiceName();
|
String sourceServiceName = event.getServiceName();
|
||||||
String serviceName = getServiceName(subscribedURL);
|
String serviceName = getServiceName(subscribedURL);
|
||||||
|
|
||||||
@@ -459,6 +482,13 @@ public class DubboCloudRegistry extends FailbackRegistry {
|
|||||||
subscribeDubboMetadataServiceURLs(subscribedURL, listener,
|
subscribeDubboMetadataServiceURLs(subscribedURL, listener,
|
||||||
sourceServiceName);
|
sourceServiceName);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "ServiceInstancesChangedEventListener:"
|
||||||
|
+ subscribedURL.getServiceKey();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -84,8 +84,6 @@ public class DubboMetadataServiceProxy implements BeanClassLoaderAware, Disposab
|
|||||||
*/
|
*/
|
||||||
public DubboMetadataService getProxy(List<ServiceInstance> serviceInstances) {
|
public DubboMetadataService getProxy(List<ServiceInstance> serviceInstances) {
|
||||||
|
|
||||||
DubboMetadataService dubboMetadataService = null;
|
|
||||||
|
|
||||||
// attempt to get the proxy of DubboMetadataService in maximum times
|
// attempt to get the proxy of DubboMetadataService in maximum times
|
||||||
int attempts = serviceInstances.size();
|
int attempts = serviceInstances.size();
|
||||||
|
|
||||||
@@ -98,7 +96,8 @@ public class DubboMetadataServiceProxy implements BeanClassLoaderAware, Disposab
|
|||||||
serviceInstance.get());
|
serviceInstance.get());
|
||||||
|
|
||||||
for (URL dubboMetadataServiceURL : dubboMetadataServiceURLs) {
|
for (URL dubboMetadataServiceURL : dubboMetadataServiceURLs) {
|
||||||
dubboMetadataService = createProxyIfAbsent(dubboMetadataServiceURL);
|
DubboMetadataService dubboMetadataService = createProxyIfAbsent(
|
||||||
|
dubboMetadataServiceURL);
|
||||||
if (dubboMetadataService != null) {
|
if (dubboMetadataService != null) {
|
||||||
return dubboMetadataService;
|
return dubboMetadataService;
|
||||||
}
|
}
|
||||||
@@ -106,7 +105,7 @@ public class DubboMetadataServiceProxy implements BeanClassLoaderAware, Disposab
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return dubboMetadataService;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -5,8 +5,7 @@ com.alibaba.cloud.dubbo.autoconfigure.DubboServiceRegistrationAutoConfiguration,
|
|||||||
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceRegistrationNonWebApplicationAutoConfiguration,\
|
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceRegistrationNonWebApplicationAutoConfiguration,\
|
||||||
com.alibaba.cloud.dubbo.autoconfigure.DubboLoadBalancedRestTemplateAutoConfiguration,\
|
com.alibaba.cloud.dubbo.autoconfigure.DubboLoadBalancedRestTemplateAutoConfiguration,\
|
||||||
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceAutoConfiguration,\
|
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceAutoConfiguration,\
|
||||||
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceDiscoveryAutoConfiguration
|
com.alibaba.cloud.dubbo.autoconfigure.DubboServiceDiscoveryAutoConfiguration,\
|
||||||
org.springframework.boot.actuate.autoconfigure.web.ManagementContextConfiguration=\
|
|
||||||
com.alibaba.cloud.dubbo.actuate.DubboMetadataEndpointAutoConfiguration
|
com.alibaba.cloud.dubbo.actuate.DubboMetadataEndpointAutoConfiguration
|
||||||
org.springframework.context.ApplicationContextInitializer=\
|
org.springframework.context.ApplicationContextInitializer=\
|
||||||
com.alibaba.cloud.dubbo.context.DubboServiceRegistrationApplicationContextInitializer
|
com.alibaba.cloud.dubbo.context.DubboServiceRegistrationApplicationContextInitializer
|
||||||
|
Reference in New Issue
Block a user