* Add sentinel-quarkus-adapter module, which provides sentinel-annotation-quarkus-adapter and sentinel-jax-rs-quarkus-adapter to adapt sentinel-annotation-cdi-interceptor and sentinel-jax-rs-adapter for Quarkus. It also provides sentinel-native-image-quarkus-adapter to support running Sentinel with Quarkus in native image mode.master
@@ -29,6 +29,7 @@ | |||||
<module>sentinel-zuul2-adapter</module> | <module>sentinel-zuul2-adapter</module> | ||||
<module>sentinel-okhttp-adapter</module> | <module>sentinel-okhttp-adapter</module> | ||||
<module>sentinel-jax-rs-adapter</module> | <module>sentinel-jax-rs-adapter</module> | ||||
<module>sentinel-quarkus-adapter</module> | |||||
</modules> | </modules> | ||||
<dependencyManagement> | <dependencyManagement> | ||||
@@ -0,0 +1,86 @@ | |||||
# sentinel quarkus adapter | |||||
sentinel quarkus adapter provides `sentinel-annotation-quarkus-adapter` and `sentinel-jax-rs-quarkus-adapter` to adapt `sentinel-annotation-cdi-interceptor` and `sentinel-jax-rs-adapter` for quarkus | |||||
sentinel quarkus adapter also provides `sentinel-native-image-quarkus-adapter` to support running sentinel with quarkus in native image mode. | |||||
To use sentinel-jax-rs-quarkus-adapter, you can simply add the following dependency to your `pom.xml`: | |||||
```xml | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-jax-rs-quarkus-adapter</artifactId> | |||||
<version>x.y.z</version> | |||||
</dependency> | |||||
``` | |||||
To use sentinel-annotation-quarkus-adapter, you can simply add the following dependency to your `pom.xml`: | |||||
```xml | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-annotation-quarkus-adapter</artifactId> | |||||
<version>x.y.z</version> | |||||
</dependency> | |||||
``` | |||||
if your quarkus application want to use both `sentinel-annotation-quarkus-adapter` and `sentinel-jax-rs-quarkus-adapter` , then add these two dependency together to your `pom.xml`: | |||||
```xml | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-jax-rs-quarkus-adapter</artifactId> | |||||
<version>x.y.z</version> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-annotation-quarkus-adapter</artifactId> | |||||
<version>x.y.z</version> | |||||
</dependency> | |||||
``` | |||||
when quarkus application started, you can see the enabled feature like: | |||||
``` | |||||
INFO [io.quarkus] (Quarkus Main Thread) Installed features: [cdi, resteasy, sentinel-annotation, sentinel-jax-rs] | |||||
``` | |||||
## for quarkus native image | |||||
if you want to make sentinel with quarkus running in native image mode, you should add the following dependency to your `pom.xml`: | |||||
```xml | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-native-image-quarkus-adapter</artifactId> | |||||
<version>x.y.z</version> | |||||
</dependency> | |||||
``` | |||||
and then add `--allow-incomplete-classpath` to `quarkus.native.additional-build-args`. | |||||
if you use `sentinel-jax-rs-quarkus-adapter` you should set `quarkus.native.auto-service-loader-registration` to true. | |||||
you can refer to `sentinel-demo-quarkus`'s `pom.xml` for more details. | |||||
when quarkus application started, you can see the enabled feature like: | |||||
``` | |||||
INFO [io.quarkus] (main) Installed features: [cdi, resteasy, sentinel-annotation, sentinel-jax-rs, sentinel-native-image] | |||||
``` | |||||
### notes for limitations | |||||
`sentinel-native-image-quarkus-adapter` currently rely on `sentinel-logging-slf4j` to make sentinel run in native image mode easily, because `quarkus-core` provides `Target_org_slf4j_LoggerFactory` to substitue `getLogger` method. | |||||
currently `sentinel-transport-simple-http` can work in native image mode, while `sentinel-transport-netty-http` cannot work in native image mode without extra config or substitutions. | |||||
## references for build native image or AOT | |||||
- [Quarkus - Tips for writing native applications](https://quarkus.io/guides/writing-native-applications-tips) | |||||
- [Quarkus - Class Loading Reference](https://quarkus.io/guides/class-loading-reference) | |||||
- [substratevm LIMITATIONS](https://github.com/oracle/graal/blob/master/substratevm/LIMITATIONS.md) | |||||
- [Accessing resources in Substrate VM images](https://github.com/oracle/graal/blob/master/substratevm/RESOURCES.md) |
@@ -0,0 +1,55 @@ | |||||
<?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"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-adapter</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
</parent> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<packaging>pom</packaging> | |||||
<properties> | |||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> | |||||
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> | |||||
<maven.compiler.source>1.8</maven.compiler.source> | |||||
<maven.compiler.target>1.8</maven.compiler.target> | |||||
<maven.compiler.parameters>true</maven.compiler.parameters> | |||||
<quarkus.version>1.4.1.Final</quarkus.version> | |||||
<compiler-plugin.version>3.8.1</compiler-plugin.version> | |||||
</properties> | |||||
<modules> | |||||
<module>sentinel-annotation-quarkus-adapter-deployment</module> | |||||
<module>sentinel-annotation-quarkus-adapter-runtime</module> | |||||
<module>sentinel-jax-rs-quarkus-adapter-deployment</module> | |||||
<module>sentinel-jax-rs-quarkus-adapter-runtime</module> | |||||
<module>sentinel-native-image-quarkus-adapter-deployment</module> | |||||
<module>sentinel-native-image-quarkus-adapter-runtime</module> | |||||
</modules> | |||||
<dependencyManagement> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-bom-deployment</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
<type>pom</type> | |||||
<scope>import</scope> | |||||
</dependency> | |||||
</dependencies> | |||||
</dependencyManagement> | |||||
<build> | |||||
<pluginManagement> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<version>${compiler-plugin.version}</version> | |||||
</plugin> | |||||
</plugins> | |||||
</pluginManagement> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,72 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-annotation-quarkus-adapter-deployment</artifactId> | |||||
<name>sentinel-annotation-quarkus-adapter-deployment</name> | |||||
<properties> | |||||
<java.source.version>1.8</java.source.version> | |||||
<java.target.version>1.8</java.target.version> | |||||
</properties> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-arc-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-annotation-quarkus-adapter</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
<!-- Test dependencies --> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-junit5-internal</artifactId> | |||||
<scope>test</scope> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>org.assertj</groupId> | |||||
<artifactId>assertj-core</artifactId> | |||||
<scope>test</scope> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-arc-deployment</artifactId> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,46 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.annotation.quarkus.adapter.deployment; | |||||
import com.alibaba.csp.sentinel.annotation.cdi.interceptor.SentinelResourceInterceptor; | |||||
import io.quarkus.arc.deployment.AdditionalBeanBuildItem; | |||||
import io.quarkus.deployment.annotations.BuildProducer; | |||||
import io.quarkus.deployment.annotations.BuildStep; | |||||
import io.quarkus.deployment.builditem.FeatureBuildItem; | |||||
import java.util.Arrays; | |||||
import java.util.List; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
class SentinelAnnotationQuarkusAdapterProcessor { | |||||
private static final String FEATURE_ANNOTATION = "sentinel-annotation"; | |||||
@BuildStep | |||||
void feature(BuildProducer<FeatureBuildItem> featureProducer) { | |||||
featureProducer.produce(new FeatureBuildItem(FEATURE_ANNOTATION)); | |||||
} | |||||
@BuildStep | |||||
List<AdditionalBeanBuildItem> additionalBeans() { | |||||
return Arrays.asList( | |||||
new AdditionalBeanBuildItem(SentinelResourceInterceptor.class) | |||||
); | |||||
} | |||||
} |
@@ -0,0 +1,84 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* Licensed under the Apache License, Version 2.0 (the "License"); | |||||
* you may not use this file except in compliance with the License. | |||||
* You may obtain a copy of the License at | |||||
* | |||||
* http://www.apache.org/licenses/LICENSE-2.0 | |||||
* | |||||
* Unless required by applicable law or agreed to in writing, software | |||||
* distributed under the License is distributed on an "AS IS" BASIS, | |||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||||
* See the License for the specific language governing permissions and | |||||
* limitations under the License. | |||||
*/ | |||||
package com.alibaba.csp.sentinel.annotation.quarkus.adapter.deployment; | |||||
import com.alibaba.csp.sentinel.annotation.cdi.interceptor.SentinelResourceBinding; | |||||
import com.alibaba.csp.sentinel.slots.block.BlockException; | |||||
import javax.enterprise.context.ApplicationScoped; | |||||
import java.util.concurrent.ThreadLocalRandom; | |||||
/** | |||||
* @author Eric Zhao | |||||
* @author sea | |||||
*/ | |||||
@ApplicationScoped | |||||
public class FooService { | |||||
@SentinelResourceBinding(value = "apiFoo", blockHandler = "fooBlockHandler", | |||||
exceptionsToTrace = {IllegalArgumentException.class}) | |||||
public String foo(int i) throws Exception { | |||||
if (i == 5758) { | |||||
throw new IllegalAccessException(); | |||||
} | |||||
if (i == 5763) { | |||||
throw new IllegalArgumentException(); | |||||
} | |||||
return "Hello for " + i; | |||||
} | |||||
@SentinelResourceBinding(value = "apiFooWithFallback", blockHandler = "fooBlockHandler", fallback = "fooFallbackFunc", | |||||
exceptionsToTrace = {IllegalArgumentException.class}) | |||||
public String fooWithFallback(int i) throws Exception { | |||||
if (i == 5758) { | |||||
throw new IllegalAccessException(); | |||||
} | |||||
if (i == 5763) { | |||||
throw new IllegalArgumentException(); | |||||
} | |||||
return "Hello for " + i; | |||||
} | |||||
@SentinelResourceBinding(value = "apiAnotherFooWithDefaultFallback", defaultFallback = "globalDefaultFallback", | |||||
fallbackClass = {FooUtil.class}) | |||||
public String anotherFoo(int i) { | |||||
if (i == 5758) { | |||||
throw new IllegalArgumentException("oops"); | |||||
} | |||||
return "Hello for " + i; | |||||
} | |||||
@SentinelResourceBinding(blockHandler = "globalBlockHandler", blockHandlerClass = FooUtil.class) | |||||
public int random() { | |||||
return ThreadLocalRandom.current().nextInt(0, 30000); | |||||
} | |||||
@SentinelResourceBinding(value = "apiBaz", blockHandler = "bazBlockHandler", | |||||
exceptionsToIgnore = {IllegalMonitorStateException.class}) | |||||
public String baz(String name) { | |||||
if (name.equals("fail")) { | |||||
throw new IllegalMonitorStateException("boom!"); | |||||
} | |||||
return "cheers, " + name; | |||||
} | |||||
public String fooBlockHandler(int i, BlockException ex) { | |||||
return "Oops, " + i; | |||||
} | |||||
public String fooFallbackFunc(int i) { | |||||
return "eee..."; | |||||
} | |||||
} |
@@ -0,0 +1,37 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* Licensed under the Apache License, Version 2.0 (the "License"); | |||||
* you may not use this file except in compliance with the License. | |||||
* You may obtain a copy of the License at | |||||
* | |||||
* http://www.apache.org/licenses/LICENSE-2.0 | |||||
* | |||||
* Unless required by applicable law or agreed to in writing, software | |||||
* distributed under the License is distributed on an "AS IS" BASIS, | |||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |||||
* See the License for the specific language governing permissions and | |||||
* limitations under the License. | |||||
*/ | |||||
package com.alibaba.csp.sentinel.annotation.quarkus.adapter.deployment; | |||||
import com.alibaba.csp.sentinel.slots.block.BlockException; | |||||
/** | |||||
* @author Eric Zhao | |||||
*/ | |||||
public class FooUtil { | |||||
public static final int BLOCK_FLAG = 88888; | |||||
public static final String FALLBACK_DEFAULT_RESULT = "fallback"; | |||||
public static int globalBlockHandler(BlockException ex) { | |||||
System.out.println("Oops: " + ex.getClass().getSimpleName()); | |||||
return BLOCK_FLAG; | |||||
} | |||||
public static String globalDefaultFallback(Throwable t) { | |||||
System.out.println("Fallback caught: " + t.getClass().getSimpleName()); | |||||
return FALLBACK_DEFAULT_RESULT; | |||||
} | |||||
} |
@@ -0,0 +1,197 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.annotation.quarkus.adapter.deployment; | |||||
import com.alibaba.csp.sentinel.node.ClusterNode; | |||||
import com.alibaba.csp.sentinel.slots.block.flow.FlowRule; | |||||
import com.alibaba.csp.sentinel.slots.block.flow.FlowRuleManager; | |||||
import com.alibaba.csp.sentinel.slots.clusterbuilder.ClusterBuilderSlot; | |||||
import com.alibaba.csp.sentinel.util.MethodUtil; | |||||
import io.quarkus.arc.ArcUndeclaredThrowableException; | |||||
import io.quarkus.test.QuarkusUnitTest; | |||||
import org.jboss.shrinkwrap.api.ShrinkWrap; | |||||
import org.jboss.shrinkwrap.api.spec.JavaArchive; | |||||
import org.junit.jupiter.api.AfterEach; | |||||
import org.junit.jupiter.api.Assertions; | |||||
import org.junit.jupiter.api.BeforeEach; | |||||
import org.junit.jupiter.api.Test; | |||||
import org.junit.jupiter.api.extension.RegisterExtension; | |||||
import javax.inject.Inject; | |||||
import java.util.ArrayList; | |||||
import java.util.Collections; | |||||
import static org.assertj.core.api.Assertions.assertThat; | |||||
import static org.assertj.core.api.Assertions.fail; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
public class SentinelAnnotationQuarkusAdapterTest { | |||||
@RegisterExtension | |||||
static final QuarkusUnitTest TEST = new QuarkusUnitTest() | |||||
.setArchiveProducer(() -> ShrinkWrap | |||||
.create(JavaArchive.class) | |||||
.addClasses(FooService.class, FooUtil.class) | |||||
.addPackage("com.alibaba.csp.sentinel.annotation.cdi.interceptor") | |||||
); | |||||
@Inject | |||||
FooService fooService; | |||||
@BeforeEach | |||||
public void setUp() throws Exception { | |||||
FlowRuleManager.loadRules(new ArrayList<FlowRule>()); | |||||
ClusterBuilderSlot.resetClusterNodes(); | |||||
} | |||||
@AfterEach | |||||
public void tearDown() throws Exception { | |||||
FlowRuleManager.loadRules(new ArrayList<FlowRule>()); | |||||
ClusterBuilderSlot.resetClusterNodes(); | |||||
} | |||||
@Test | |||||
public void testForeignBlockHandlerClass() throws Exception { | |||||
assertThat(fooService.random()).isNotEqualTo(FooUtil.BLOCK_FLAG); | |||||
String resourceName = MethodUtil.resolveMethodName(FooService.class.getDeclaredMethod("random")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
FlowRuleManager.loadRules(Collections.singletonList( | |||||
new FlowRule(resourceName).setCount(0) | |||||
)); | |||||
assertThat(fooService.random()).isEqualTo(FooUtil.BLOCK_FLAG); | |||||
assertThat(cn.blockQps()).isPositive(); | |||||
} | |||||
@Test | |||||
public void testBlockHandlerNotFound() { | |||||
assertThat(fooService.baz("Sentinel")).isEqualTo("cheers, Sentinel"); | |||||
String resourceName = "apiBaz"; | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
FlowRuleManager.loadRules(Collections.singletonList( | |||||
new FlowRule(resourceName).setCount(0) | |||||
)); | |||||
Assertions.assertThrows(ArcUndeclaredThrowableException.class, () -> { | |||||
fooService.baz("Sentinel"); | |||||
}); | |||||
} | |||||
@Test | |||||
public void testAnnotationExceptionsToIgnore() { | |||||
assertThat(fooService.baz("Sentinel")).isEqualTo("cheers, Sentinel"); | |||||
String resourceName = "apiBaz"; | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
try { | |||||
fooService.baz("fail"); | |||||
fail("should not reach here"); | |||||
} catch (IllegalMonitorStateException ex) { | |||||
assertThat(cn.exceptionQps()).isZero(); | |||||
} | |||||
} | |||||
@Test | |||||
public void testFallbackWithNoParams() throws Exception { | |||||
assertThat(fooService.fooWithFallback(1)).isEqualTo("Hello for 1"); | |||||
String resourceName = "apiFooWithFallback"; | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
// Fallback should be ignored for this. | |||||
try { | |||||
fooService.fooWithFallback(5758); | |||||
fail("should not reach here"); | |||||
} catch (IllegalAccessException e) { | |||||
assertThat(cn.exceptionQps()).isZero(); | |||||
} | |||||
// Fallback should take effect. | |||||
assertThat(fooService.fooWithFallback(5763)).isEqualTo("eee..."); | |||||
assertThat(cn.exceptionQps()).isPositive(); | |||||
assertThat(cn.blockQps()).isZero(); | |||||
FlowRuleManager.loadRules(Collections.singletonList( | |||||
new FlowRule(resourceName).setCount(0) | |||||
)); | |||||
// Fallback should not take effect for BlockException, as blockHandler is configured. | |||||
assertThat(fooService.fooWithFallback(2221)).isEqualTo("Oops, 2221"); | |||||
assertThat(cn.blockQps()).isPositive(); | |||||
} | |||||
@Test | |||||
public void testDefaultFallbackWithSingleParam() { | |||||
assertThat(fooService.anotherFoo(1)).isEqualTo("Hello for 1"); | |||||
String resourceName = "apiAnotherFooWithDefaultFallback"; | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
// Default fallback should take effect. | |||||
assertThat(fooService.anotherFoo(5758)).isEqualTo(FooUtil.FALLBACK_DEFAULT_RESULT); | |||||
assertThat(cn.exceptionQps()).isPositive(); | |||||
assertThat(cn.blockQps()).isZero(); | |||||
FlowRuleManager.loadRules(Collections.singletonList( | |||||
new FlowRule(resourceName).setCount(0) | |||||
)); | |||||
// Default fallback should also take effect for BlockException. | |||||
assertThat(fooService.anotherFoo(5758)).isEqualTo(FooUtil.FALLBACK_DEFAULT_RESULT); | |||||
assertThat(cn.blockQps()).isPositive(); | |||||
} | |||||
@Test | |||||
public void testNormalBlockHandlerAndFallback() throws Exception { | |||||
assertThat(fooService.foo(1)).isEqualTo("Hello for 1"); | |||||
String resourceName = "apiFoo"; | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertThat(cn).isNotNull(); | |||||
assertThat(cn.passQps()).isPositive(); | |||||
// Test for biz exception. | |||||
try { | |||||
fooService.foo(5758); | |||||
fail("should not reach here"); | |||||
} catch (Exception ex) { | |||||
// Should not be traced. | |||||
assertThat(cn.exceptionQps()).isZero(); | |||||
} | |||||
try { | |||||
fooService.foo(5763); | |||||
fail("should not reach here"); | |||||
} catch (Exception ex) { | |||||
assertThat(cn.exceptionQps()).isPositive(); | |||||
} | |||||
// Test for blockHandler | |||||
FlowRuleManager.loadRules(Collections.singletonList( | |||||
new FlowRule(resourceName).setCount(0) | |||||
)); | |||||
assertThat(fooService.foo(1121)).isEqualTo("Oops, 1121"); | |||||
assertThat(cn.blockQps()).isPositive(); | |||||
} | |||||
} |
@@ -0,0 +1,63 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-annotation-quarkus-adapter</artifactId> | |||||
<name>sentinel-annotation-quarkus-adapter</name> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-annotation-cdi-interceptor</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-bootstrap-maven-plugin</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
<executions> | |||||
<execution> | |||||
<goals> | |||||
<goal>extension-descriptor</goal> | |||||
</goals> | |||||
<phase>compile</phase> | |||||
<configuration> | |||||
<deployment>${project.groupId}:${project.artifactId}-deployment:${project.version} | |||||
</deployment> | |||||
</configuration> | |||||
</execution> | |||||
</executions> | |||||
</plugin> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,11 @@ | |||||
--- | |||||
name: "sentinel annotation extension" | |||||
metadata: | |||||
keywords: | |||||
- "sentinel" | |||||
- "rate limit" | |||||
- "circuit breaker" | |||||
categories: | |||||
- "rate limit" | |||||
- "circuit breaker" | |||||
status: "preview" |
@@ -0,0 +1,77 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-jax-rs-quarkus-adapter-deployment</artifactId> | |||||
<name>sentinel-jax-rs-quarkus-adapter-deployment</name> | |||||
<properties> | |||||
<java.source.version>1.8</java.source.version> | |||||
<java.target.version>1.8</java.target.version> | |||||
</properties> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-arc-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-resteasy-server-common-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-jax-rs-quarkus-adapter</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
<!-- Test dependencies --> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-junit5-internal</artifactId> | |||||
<scope>test</scope> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-resteasy-deployment</artifactId> | |||||
<scope>test</scope> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.rest-assured</groupId> | |||||
<artifactId>rest-assured</artifactId> | |||||
<scope>test</scope> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,37 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.jaxrs.quarkus.adapter.deployment; | |||||
import io.quarkus.deployment.annotations.BuildProducer; | |||||
import io.quarkus.deployment.annotations.BuildStep; | |||||
import io.quarkus.deployment.builditem.FeatureBuildItem; | |||||
import org.jboss.logging.Logger; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
class SentinelJaxRsQuarkusAdapterProcessor { | |||||
private static final Logger logger = Logger.getLogger(SentinelJaxRsQuarkusAdapterProcessor.class); | |||||
private static final String FEATURE_JAX_RS = "sentinel-jax-rs"; | |||||
@BuildStep | |||||
void feature(BuildProducer<FeatureBuildItem> featureProducer) { | |||||
featureProducer.produce(new FeatureBuildItem(FEATURE_JAX_RS)); | |||||
} | |||||
} |
@@ -0,0 +1,242 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.jaxrs.quarkus.adapter.deployment; | |||||
import com.alibaba.csp.sentinel.Constants; | |||||
import com.alibaba.csp.sentinel.adapter.jaxrs.config.SentinelJaxRsConfig; | |||||
import com.alibaba.csp.sentinel.adapter.jaxrs.fallback.SentinelJaxRsFallback; | |||||
import com.alibaba.csp.sentinel.adapter.jaxrs.request.RequestOriginParser; | |||||
import com.alibaba.csp.sentinel.node.ClusterNode; | |||||
import com.alibaba.csp.sentinel.node.EntranceNode; | |||||
import com.alibaba.csp.sentinel.node.Node; | |||||
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.FlowRuleManager; | |||||
import com.alibaba.csp.sentinel.slots.clusterbuilder.ClusterBuilderSlot; | |||||
import com.alibaba.csp.sentinel.util.StringUtil; | |||||
import io.quarkus.test.QuarkusUnitTest; | |||||
import io.restassured.response.Response; | |||||
import org.jboss.shrinkwrap.api.ShrinkWrap; | |||||
import org.jboss.shrinkwrap.api.spec.JavaArchive; | |||||
import org.junit.jupiter.api.AfterEach; | |||||
import org.junit.jupiter.api.Test; | |||||
import org.junit.jupiter.api.extension.RegisterExtension; | |||||
import javax.ws.rs.container.ContainerRequestContext; | |||||
import javax.ws.rs.core.MediaType; | |||||
import java.util.Collections; | |||||
import java.util.concurrent.Callable; | |||||
import java.util.concurrent.Future; | |||||
import java.util.concurrent.FutureTask; | |||||
import static io.restassured.RestAssured.given; | |||||
import static org.hamcrest.CoreMatchers.equalTo; | |||||
import static org.junit.jupiter.api.Assertions.*; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
public class SentinelJaxRsQuarkusAdapterTest { | |||||
private static final String HELLO_STR = "Hello!"; | |||||
@RegisterExtension | |||||
static final QuarkusUnitTest TEST = new QuarkusUnitTest() | |||||
.setArchiveProducer(() -> ShrinkWrap | |||||
.create(JavaArchive.class) | |||||
.addClasses(TestResource.class)); | |||||
@AfterEach | |||||
public void cleanUp() { | |||||
FlowRuleManager.loadRules(null); | |||||
ClusterBuilderSlot.resetClusterNodes(); | |||||
} | |||||
@Test | |||||
public void testGetHello() { | |||||
String url = "/test/hello"; | |||||
String resourceName = "GET:" + url; | |||||
Response response = given().get(url); | |||||
response.then().statusCode(200).body(equalTo(HELLO_STR)); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(1, cn.passQps(), 0.01); | |||||
String context = ""; | |||||
for (Node n : Constants.ROOT.getChildList()) { | |||||
if (n instanceof EntranceNode) { | |||||
String id = ((EntranceNode) n).getId().getName(); | |||||
if (url.equals(id)) { | |||||
context = ((EntranceNode) n).getId().getName(); | |||||
} | |||||
} | |||||
} | |||||
assertEquals("", context); | |||||
} | |||||
@Test | |||||
public void testAsyncGetHello() { | |||||
String url = "/test/async-hello"; | |||||
String resourceName = "GET:" + url; | |||||
Response response = given().get(url); | |||||
response.then().statusCode(200).body(equalTo(HELLO_STR)); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(1, cn.passQps(), 0.01); | |||||
String context = ""; | |||||
for (Node n : Constants.ROOT.getChildList()) { | |||||
if (n instanceof EntranceNode) { | |||||
String id = ((EntranceNode) n).getId().getName(); | |||||
if (url.equals(id)) { | |||||
context = ((EntranceNode) n).getId().getName(); | |||||
} | |||||
} | |||||
} | |||||
assertEquals("", context); | |||||
} | |||||
@Test | |||||
public void testUrlPathParam() { | |||||
String url = "/test/hello/{name}"; | |||||
String resourceName = "GET:" + url; | |||||
String url1 = "/test/hello/abc"; | |||||
Response response1 = given().get(url1); | |||||
response1.then().statusCode(200).body(equalTo("Hello abc !")); | |||||
String url2 = "/test/hello/def"; | |||||
Response response2 = given().get(url2); | |||||
response2.then().statusCode(200).body(equalTo("Hello def !")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(2, cn.passQps(), 0.01); | |||||
assertNull(ClusterBuilderSlot.getClusterNode("GET:" + url1)); | |||||
assertNull(ClusterBuilderSlot.getClusterNode("GET:" + url2)); | |||||
} | |||||
@Test | |||||
public void testDefaultFallback() { | |||||
String url = "/test/hello"; | |||||
String resourceName = "GET:" + url; | |||||
configureRulesFor(resourceName, 0); | |||||
Response response = given().get(url); | |||||
response.then().statusCode(javax.ws.rs.core.Response.Status.TOO_MANY_REQUESTS.getStatusCode()) | |||||
.body(equalTo("Blocked by Sentinel (flow limiting)")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(0, cn.passQps(), 0.01); | |||||
} | |||||
@Test | |||||
public void testCustomFallback() { | |||||
String url = "/test/hello"; | |||||
String resourceName = "GET:" + url; | |||||
SentinelJaxRsConfig.setJaxRsFallback(new SentinelJaxRsFallback() { | |||||
@Override | |||||
public javax.ws.rs.core.Response fallbackResponse(String route, Throwable cause) { | |||||
return javax.ws.rs.core.Response.status(javax.ws.rs.core.Response.Status.OK) | |||||
.entity("Blocked by Sentinel (flow limiting)") | |||||
.type(MediaType.APPLICATION_JSON_TYPE) | |||||
.build(); | |||||
} | |||||
@Override | |||||
public Future<javax.ws.rs.core.Response> fallbackFutureResponse(final String route, final Throwable cause) { | |||||
return new FutureTask<>(new Callable<javax.ws.rs.core.Response>() { | |||||
@Override | |||||
public javax.ws.rs.core.Response call() throws Exception { | |||||
return fallbackResponse(route, cause); | |||||
} | |||||
}); | |||||
} | |||||
}); | |||||
configureRulesFor(resourceName, 0); | |||||
Response response = given().get(url); | |||||
response.then().statusCode(javax.ws.rs.core.Response.Status.OK.getStatusCode()) | |||||
.body(equalTo("Blocked by Sentinel (flow limiting)")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(0, cn.passQps(), 0.01); | |||||
} | |||||
@Test | |||||
public void testCustomRequestOriginParser() { | |||||
String url = "/test/hello"; | |||||
String resourceName = "GET:" + url; | |||||
String limitOrigin = "appB"; | |||||
final String headerName = "X-APP"; | |||||
configureRulesFor(resourceName, 0, limitOrigin); | |||||
SentinelJaxRsConfig.setRequestOriginParser(new RequestOriginParser() { | |||||
@Override | |||||
public String parseOrigin(ContainerRequestContext request) { | |||||
String origin = request.getHeaderString(headerName); | |||||
return origin != null ? origin : ""; | |||||
} | |||||
}); | |||||
Response response = given() | |||||
.header(headerName, "appA").get(url); | |||||
response.then().statusCode(200).body(equalTo(HELLO_STR)); | |||||
Response blockedResp = given() | |||||
.header(headerName, "appB") | |||||
.get(url); | |||||
blockedResp.then().statusCode(javax.ws.rs.core.Response.Status.TOO_MANY_REQUESTS.getStatusCode()) | |||||
.body(equalTo("Blocked by Sentinel (flow limiting)")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
assertEquals(1, cn.passQps(), 0.01); | |||||
assertEquals(1, cn.blockQps(), 0.01); | |||||
} | |||||
@Test | |||||
public void testExceptionMapper() { | |||||
String url = "/test/ex"; | |||||
String resourceName = "GET:" + url; | |||||
Response response = given().get(url); | |||||
response.then().statusCode(javax.ws.rs.core.Response.Status.INTERNAL_SERVER_ERROR.getStatusCode()).body(equalTo("test exception mapper")); | |||||
ClusterNode cn = ClusterBuilderSlot.getClusterNode(resourceName); | |||||
assertNotNull(cn); | |||||
} | |||||
private void configureRulesFor(String resource, int count) { | |||||
configureRulesFor(resource, count, "default"); | |||||
} | |||||
private void configureRulesFor(String resource, int count, String limitApp) { | |||||
FlowRule rule = new FlowRule() | |||||
.setCount(count) | |||||
.setGrade(RuleConstant.FLOW_GRADE_QPS); | |||||
rule.setResource(resource); | |||||
if (StringUtil.isNotBlank(limitApp)) { | |||||
rule.setLimitApp(limitApp); | |||||
} | |||||
FlowRuleManager.loadRules(Collections.singletonList(rule)); | |||||
} | |||||
} |
@@ -0,0 +1,90 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.jaxrs.quarkus.adapter.deployment; | |||||
import javax.ws.rs.*; | |||||
import javax.ws.rs.container.AsyncResponse; | |||||
import javax.ws.rs.container.Suspended; | |||||
import javax.ws.rs.core.MediaType; | |||||
import javax.ws.rs.core.Response; | |||||
import java.util.concurrent.ExecutorService; | |||||
import java.util.concurrent.Executors; | |||||
import java.util.concurrent.TimeUnit; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
@Path("/test") | |||||
public class TestResource { | |||||
ExecutorService executor = Executors.newFixedThreadPool(5); | |||||
@Path("/hello") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public String sayHello() { | |||||
return "Hello!"; | |||||
} | |||||
@Path("/async-hello") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public void asyncSayHello(@Suspended final AsyncResponse asyncResponse) { | |||||
executor.submit(new Runnable() { | |||||
@Override | |||||
public void run() { | |||||
try { | |||||
TimeUnit.MILLISECONDS.sleep(100); | |||||
} catch (InterruptedException e) { | |||||
e.printStackTrace(); | |||||
} | |||||
asyncResponse.resume("Hello!"); | |||||
} | |||||
}); | |||||
} | |||||
@Path("/hello/{name}") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public String sayHelloWithName(@PathParam(value = "name") String name) { | |||||
return "Hello " + name + " !"; | |||||
} | |||||
@Path("/ex") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public String exception() { | |||||
throw new RuntimeException("test exception mapper"); | |||||
} | |||||
@Path("/400") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public String badRequest() { | |||||
throw new WebApplicationException(Response.status(Response.Status.BAD_REQUEST) | |||||
.entity("test return 400") | |||||
.build()); | |||||
} | |||||
@Path("/delay/{seconds}") | |||||
@GET | |||||
@Produces({ MediaType.APPLICATION_JSON }) | |||||
public String delay(@PathParam(value = "seconds") long seconds) throws InterruptedException { | |||||
TimeUnit.SECONDS.sleep(seconds); | |||||
return "finish"; | |||||
} | |||||
} |
@@ -0,0 +1,63 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-jax-rs-quarkus-adapter</artifactId> | |||||
<name>sentinel-jax-rs-quarkus-adapter</name> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-jax-rs-adapter</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-bootstrap-maven-plugin</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
<executions> | |||||
<execution> | |||||
<goals> | |||||
<goal>extension-descriptor</goal> | |||||
</goals> | |||||
<phase>compile</phase> | |||||
<configuration> | |||||
<deployment>${project.groupId}:${project.artifactId}-deployment:${project.version} | |||||
</deployment> | |||||
</configuration> | |||||
</execution> | |||||
</executions> | |||||
</plugin> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,11 @@ | |||||
--- | |||||
name: "sentinel jax rs extension" | |||||
metadata: | |||||
keywords: | |||||
- "resteasy" | |||||
- "jaxrs" | |||||
- "sentinel" | |||||
categories: | |||||
- "web" | |||||
- "circuit breaker" | |||||
status: "preview" |
@@ -0,0 +1,2 @@ | |||||
com.alibaba.csp.sentinel.adapter.jaxrs.SentinelJaxRsProviderFilter | |||||
com.alibaba.csp.sentinel.adapter.jaxrs.exception.DefaultExceptionMapper |
@@ -0,0 +1,60 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-native-image-quarkus-adapter-deployment</artifactId> | |||||
<name>sentinel-native-image-quarkus-adapter-deployment</name> | |||||
<properties> | |||||
<java.source.version>1.8</java.source.version> | |||||
<java.target.version>1.8</java.target.version> | |||||
</properties> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-arc-deployment</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>org.graalvm.nativeimage</groupId> | |||||
<artifactId>svm</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-native-image-quarkus-adapter</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,73 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.nativeimage; | |||||
import com.alibaba.csp.sentinel.slots.DefaultSlotChainBuilder; | |||||
import io.quarkus.deployment.annotations.BuildProducer; | |||||
import io.quarkus.deployment.annotations.BuildStep; | |||||
import io.quarkus.deployment.annotations.ExecutionTime; | |||||
import io.quarkus.deployment.annotations.Record; | |||||
import io.quarkus.deployment.builditem.FeatureBuildItem; | |||||
import io.quarkus.deployment.builditem.nativeimage.ReflectiveClassBuildItem; | |||||
import io.quarkus.deployment.builditem.nativeimage.RuntimeInitializedClassBuildItem; | |||||
import io.quarkus.deployment.pkg.steps.NativeBuild; | |||||
import java.util.Arrays; | |||||
import java.util.List; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
class SentinelNativeImageProcessor { | |||||
private static final String FEATURE_NATIVE_IMAGE = "sentinel-native-image"; | |||||
@BuildStep | |||||
void feature(BuildProducer<FeatureBuildItem> featureProducer) { | |||||
featureProducer.produce(new FeatureBuildItem(FEATURE_NATIVE_IMAGE)); | |||||
} | |||||
@BuildStep(onlyIf = NativeBuild.class) | |||||
List<RuntimeInitializedClassBuildItem> runtimeInitializedClasses() { | |||||
return Arrays.asList( | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.fastjson.serializer.JodaCodec"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.fastjson.serializer.GuavaCodec"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.fastjson.support.moneta.MonetaCodec"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.Env"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.init.InitExecutor"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.cluster.ClusterStateManager"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.slots.block.flow.FlowRuleManager"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.slots.block.degrade.DegradeRuleManager"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.node.metric.MetricTimerListener"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.node.metric.MetricWriter"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.util.TimeUtil"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.eagleeye.StatLogController"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.slots.logger.EagleEyeLogUtil"), | |||||
new RuntimeInitializedClassBuildItem("com.alibaba.csp.sentinel.eagleeye.EagleEye")); | |||||
} | |||||
@BuildStep(onlyIf = NativeBuild.class) | |||||
ReflectiveClassBuildItem setupSentinelReflectiveClasses() { | |||||
return new ReflectiveClassBuildItem(true, true, true, | |||||
DefaultSlotChainBuilder.class.getName()); | |||||
} | |||||
@BuildStep(onlyIf = NativeBuild.class) | |||||
@Record(ExecutionTime.STATIC_INIT) | |||||
void record(SentinelRecorder recorder) { | |||||
recorder.init(); | |||||
} | |||||
} |
@@ -0,0 +1,75 @@ | |||||
<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd"> | |||||
<modelVersion>4.0.0</modelVersion> | |||||
<parent> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-quarkus-adapter-parent</artifactId> | |||||
<version>1.8.0-SNAPSHOT</version> | |||||
<relativePath>../pom.xml</relativePath> | |||||
</parent> | |||||
<artifactId>sentinel-native-image-quarkus-adapter</artifactId> | |||||
<name>sentinel-native-image-quarkus-adapter</name> | |||||
<dependencies> | |||||
<dependency> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-core</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>org.graalvm.nativeimage</groupId> | |||||
<artifactId>svm</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-transport-simple-http</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-parameter-flow-control</artifactId> | |||||
</dependency> | |||||
<dependency> | |||||
<groupId>com.alibaba.csp</groupId> | |||||
<artifactId>sentinel-logging-slf4j</artifactId> | |||||
<version>${project.version}</version> | |||||
</dependency> | |||||
</dependencies> | |||||
<build> | |||||
<plugins> | |||||
<plugin> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-bootstrap-maven-plugin</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
<executions> | |||||
<execution> | |||||
<goals> | |||||
<goal>extension-descriptor</goal> | |||||
</goals> | |||||
<phase>compile</phase> | |||||
<configuration> | |||||
<deployment>${project.groupId}:${project.artifactId}-deployment:${project.version} | |||||
</deployment> | |||||
</configuration> | |||||
</execution> | |||||
</executions> | |||||
</plugin> | |||||
<plugin> | |||||
<groupId>org.apache.maven.plugins</groupId> | |||||
<artifactId>maven-compiler-plugin</artifactId> | |||||
<configuration> | |||||
<annotationProcessorPaths> | |||||
<path> | |||||
<groupId>io.quarkus</groupId> | |||||
<artifactId>quarkus-extension-processor</artifactId> | |||||
<version>${quarkus.version}</version> | |||||
</path> | |||||
</annotationProcessorPaths> | |||||
</configuration> | |||||
</plugin> | |||||
</plugins> | |||||
</build> | |||||
</project> |
@@ -0,0 +1,52 @@ | |||||
/* | |||||
* Copyright 1999-2020 Alibaba Group Holding Ltd. | |||||
* | |||||
* 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.csp.sentinel.nativeimage; | |||||
import com.alibaba.csp.sentinel.command.vo.NodeVo; | |||||
import com.alibaba.csp.sentinel.slots.block.authority.AuthorityRule; | |||||
import com.alibaba.csp.sentinel.slots.block.degrade.DegradeRule; | |||||
import com.alibaba.csp.sentinel.slots.block.flow.FlowRule; | |||||
import com.alibaba.csp.sentinel.slots.block.flow.param.ParamFlowRule; | |||||
import com.alibaba.csp.sentinel.slots.system.SystemRule; | |||||
import com.alibaba.fastjson.parser.ParserConfig; | |||||
import com.alibaba.fastjson.serializer.SerializeConfig; | |||||
import io.quarkus.runtime.annotations.Recorder; | |||||
/** | |||||
* @author sea | |||||
*/ | |||||
@Recorder | |||||
public class SentinelRecorder { | |||||
/** | |||||
* register fastjson serializer deserializer class info | |||||
*/ | |||||
public void init() { | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(NodeVo.class); | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(FlowRule.class); | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(SystemRule.class); | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(DegradeRule.class); | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(AuthorityRule.class); | |||||
SerializeConfig.getGlobalInstance().getObjectWriter(ParamFlowRule.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(NodeVo.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(FlowRule.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(SystemRule.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(DegradeRule.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(AuthorityRule.class); | |||||
ParserConfig.getGlobalInstance().getDeserializer(ParamFlowRule.class); | |||||
} | |||||
} |
@@ -0,0 +1,12 @@ | |||||
--- | |||||
name: "sentinel native image extension" | |||||
metadata: | |||||
keywords: | |||||
- "sentinel" | |||||
- "rate limit" | |||||
- "circuit breaker" | |||||
- "native image" | |||||
categories: | |||||
- "rate limit" | |||||
- "circuit breaker" | |||||
status: "preview" |