mirror of https://github.com/longtai-cn/hippo4j
Merge pull request #494 from mabaiwan/develop
Add thread pool automatic registration functionpull/499/head
commit
92ede063d9
@ -0,0 +1,77 @@
|
|||||||
|
package cn.hippo4j.common.model.register;
|
||||||
|
|
||||||
|
import lombok.AllArgsConstructor;
|
||||||
|
import lombok.Builder;
|
||||||
|
import lombok.Data;
|
||||||
|
import lombok.NoArgsConstructor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool register parameter.
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
@Builder
|
||||||
|
@NoArgsConstructor
|
||||||
|
@AllArgsConstructor
|
||||||
|
public class DynamicThreadPoolRegisterParameter {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Thread-pool id
|
||||||
|
* Empty or empty strings are not allowed, and `+` signs are not allowed
|
||||||
|
*/
|
||||||
|
private String threadPoolId;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Content
|
||||||
|
*/
|
||||||
|
private String content;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Core pool size
|
||||||
|
*/
|
||||||
|
private Integer corePoolSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maximum pool size
|
||||||
|
*/
|
||||||
|
private Integer maximumPoolSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Queue type
|
||||||
|
*/
|
||||||
|
private Integer queueType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Capacity
|
||||||
|
*/
|
||||||
|
private Integer capacity;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Keep alive time
|
||||||
|
*/
|
||||||
|
private Integer keepAliveTime;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rejected type
|
||||||
|
*/
|
||||||
|
private Integer rejectedType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is alarm
|
||||||
|
*/
|
||||||
|
private Integer isAlarm;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Capacity alarm
|
||||||
|
*/
|
||||||
|
private Integer capacityAlarm;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Liveness alarm
|
||||||
|
*/
|
||||||
|
private Integer livenessAlarm;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allow core thread timeout
|
||||||
|
*/
|
||||||
|
private Integer allowCoreThreadTimeOut;
|
||||||
|
}
|
@ -0,0 +1,45 @@
|
|||||||
|
package cn.hippo4j.common.model.register;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonIgnore;
|
||||||
|
import lombok.AllArgsConstructor;
|
||||||
|
import lombok.Builder;
|
||||||
|
import lombok.Data;
|
||||||
|
import lombok.NoArgsConstructor;
|
||||||
|
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool register wrapper.
|
||||||
|
*/
|
||||||
|
@Data
|
||||||
|
@Builder
|
||||||
|
@NoArgsConstructor
|
||||||
|
@AllArgsConstructor
|
||||||
|
public class DynamicThreadPoolRegisterWrapper {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tenant id
|
||||||
|
*/
|
||||||
|
private String tenantId;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Item id
|
||||||
|
*/
|
||||||
|
private String itemId;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update if exists
|
||||||
|
*/
|
||||||
|
private Boolean updateIfExists = Boolean.TRUE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool executor
|
||||||
|
*/
|
||||||
|
@JsonIgnore
|
||||||
|
private ThreadPoolExecutor dynamicThreadPoolExecutor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool register parameter
|
||||||
|
*/
|
||||||
|
private DynamicThreadPoolRegisterParameter dynamicThreadPoolRegisterParameter;
|
||||||
|
}
|
@ -0,0 +1,16 @@
|
|||||||
|
package cn.hippo4j.core.executor.support;
|
||||||
|
|
||||||
|
import cn.hippo4j.common.model.register.DynamicThreadPoolRegisterWrapper;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool service.
|
||||||
|
*/
|
||||||
|
public interface DynamicThreadPoolService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registering dynamic thread pools at runtime.
|
||||||
|
*
|
||||||
|
* @param registerWrapper
|
||||||
|
*/
|
||||||
|
void registerDynamicThreadPool(DynamicThreadPoolRegisterWrapper registerWrapper);
|
||||||
|
}
|
@ -0,0 +1,49 @@
|
|||||||
|
package cn.hippo4j.example.server;
|
||||||
|
|
||||||
|
import cn.hippo4j.core.executor.support.ThreadPoolBuilder;
|
||||||
|
import cn.hippo4j.common.model.register.DynamicThreadPoolRegisterParameter;
|
||||||
|
import cn.hippo4j.common.model.register.DynamicThreadPoolRegisterWrapper;
|
||||||
|
import cn.hippo4j.core.executor.support.DynamicThreadPoolService;
|
||||||
|
import lombok.AllArgsConstructor;
|
||||||
|
import org.springframework.boot.ApplicationArguments;
|
||||||
|
import org.springframework.boot.ApplicationRunner;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register dynamic thread-pool test
|
||||||
|
*/
|
||||||
|
@Component
|
||||||
|
@AllArgsConstructor
|
||||||
|
public class RegisterDynamicThreadPoolTest implements ApplicationRunner {
|
||||||
|
|
||||||
|
private final DynamicThreadPoolService dynamicThreadPoolService;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run(ApplicationArguments args) throws Exception {
|
||||||
|
String threadPoolId = "register-dynamic-thread-pool";
|
||||||
|
DynamicThreadPoolRegisterParameter parameterInfo = new DynamicThreadPoolRegisterParameter();
|
||||||
|
parameterInfo.setThreadPoolId(threadPoolId);
|
||||||
|
parameterInfo.setCorePoolSize(2);
|
||||||
|
parameterInfo.setMaximumPoolSize(14);
|
||||||
|
parameterInfo.setQueueType(9);
|
||||||
|
parameterInfo.setCapacity(110);
|
||||||
|
parameterInfo.setKeepAliveTime(110);
|
||||||
|
parameterInfo.setRejectedType(2);
|
||||||
|
parameterInfo.setIsAlarm(0);
|
||||||
|
parameterInfo.setCapacityAlarm(90);
|
||||||
|
parameterInfo.setLivenessAlarm(90);
|
||||||
|
parameterInfo.setAllowCoreThreadTimeOut(0);
|
||||||
|
ThreadPoolExecutor threadPoolExecutor = ThreadPoolBuilder.builder()
|
||||||
|
.threadPoolId(threadPoolId)
|
||||||
|
.threadFactory(threadPoolId)
|
||||||
|
.dynamicPool()
|
||||||
|
.build();
|
||||||
|
DynamicThreadPoolRegisterWrapper registerWrapper = DynamicThreadPoolRegisterWrapper.builder()
|
||||||
|
.dynamicThreadPoolRegisterParameter(parameterInfo)
|
||||||
|
.dynamicThreadPoolExecutor(threadPoolExecutor)
|
||||||
|
.build();
|
||||||
|
dynamicThreadPoolService.registerDynamicThreadPool(registerWrapper);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,91 @@
|
|||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You 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 cn.hippo4j.springboot.starter.core;
|
||||||
|
|
||||||
|
import cn.hippo4j.common.model.ThreadPoolParameterInfo;
|
||||||
|
import cn.hippo4j.common.toolkit.Assert;
|
||||||
|
import cn.hippo4j.common.toolkit.JSONUtil;
|
||||||
|
import cn.hippo4j.common.web.base.Result;
|
||||||
|
import cn.hippo4j.core.executor.DynamicThreadPoolWrapper;
|
||||||
|
import cn.hippo4j.core.executor.manage.GlobalThreadPoolManage;
|
||||||
|
import cn.hippo4j.common.model.register.DynamicThreadPoolRegisterParameter;
|
||||||
|
import cn.hippo4j.common.model.register.DynamicThreadPoolRegisterWrapper;
|
||||||
|
import cn.hippo4j.core.executor.support.DynamicThreadPoolService;
|
||||||
|
import cn.hippo4j.springboot.starter.config.BootstrapProperties;
|
||||||
|
import cn.hippo4j.springboot.starter.event.ApplicationCompleteEvent;
|
||||||
|
import cn.hippo4j.springboot.starter.remote.HttpAgent;
|
||||||
|
import lombok.RequiredArgsConstructor;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.context.ApplicationListener;
|
||||||
|
|
||||||
|
import static cn.hippo4j.common.constant.Constants.REGISTER_DYNAMIC_THREAD_POOL_PATH;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool config service.
|
||||||
|
*/
|
||||||
|
@Slf4j
|
||||||
|
@RequiredArgsConstructor
|
||||||
|
public class DynamicThreadPoolConfigService implements DynamicThreadPoolService, ApplicationListener<ApplicationCompleteEvent> {
|
||||||
|
|
||||||
|
private final HttpAgent httpAgent;
|
||||||
|
|
||||||
|
private final ClientWorker clientWorker;
|
||||||
|
|
||||||
|
private final BootstrapProperties properties;
|
||||||
|
|
||||||
|
private final DynamicThreadPoolSubscribeConfig dynamicThreadPoolSubscribeConfig;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onApplicationEvent(ApplicationCompleteEvent event) {
|
||||||
|
clientWorker.notifyApplicationComplete();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void registerDynamicThreadPool(DynamicThreadPoolRegisterWrapper registerWrapper) {
|
||||||
|
DynamicThreadPoolRegisterParameter registerParameter = registerWrapper.getDynamicThreadPoolRegisterParameter();
|
||||||
|
checkThreadPoolParameter(registerParameter);
|
||||||
|
ThreadPoolParameterInfo parameter = JSONUtil.parseObject(JSONUtil.toJSONString(registerParameter), ThreadPoolParameterInfo.class);
|
||||||
|
String threadPoolId = registerParameter.getThreadPoolId();
|
||||||
|
Result registerResult;
|
||||||
|
try {
|
||||||
|
failDynamicThreadPoolRegisterWrapper(registerWrapper);
|
||||||
|
registerResult = httpAgent.httpPost(REGISTER_DYNAMIC_THREAD_POOL_PATH, registerWrapper);
|
||||||
|
} catch (Throwable ex) {
|
||||||
|
log.error("Failed to dynamically register thread pool: {}", threadPoolId, ex);
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
if (registerResult.isSuccess()) {
|
||||||
|
DynamicThreadPoolWrapper dynamicThreadPoolWrapper = DynamicThreadPoolWrapper.builder()
|
||||||
|
.threadPoolId(threadPoolId)
|
||||||
|
.executor(registerWrapper.getDynamicThreadPoolExecutor())
|
||||||
|
.build();
|
||||||
|
GlobalThreadPoolManage.register(threadPoolId, parameter, dynamicThreadPoolWrapper);
|
||||||
|
dynamicThreadPoolSubscribeConfig.subscribeConfig(threadPoolId);
|
||||||
|
clientWorker.addCacheDataIfAbsent(properties.getNamespace(), properties.getItemId(), threadPoolId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkThreadPoolParameter(DynamicThreadPoolRegisterParameter registerParameter) {
|
||||||
|
Assert.isTrue(!registerParameter.getThreadPoolId().contains("+"), "The thread pool contains sensitive characters.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void failDynamicThreadPoolRegisterWrapper(DynamicThreadPoolRegisterWrapper registerWrapper) {
|
||||||
|
registerWrapper.setTenantId(properties.getNamespace());
|
||||||
|
registerWrapper.setItemId(properties.getItemId());
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,57 @@
|
|||||||
|
package cn.hippo4j.springboot.starter.core;
|
||||||
|
|
||||||
|
import cn.hippo4j.common.api.ThreadPoolDynamicRefresh;
|
||||||
|
import cn.hippo4j.core.executor.support.QueueTypeEnum;
|
||||||
|
import cn.hippo4j.core.executor.support.ThreadPoolBuilder;
|
||||||
|
import cn.hippo4j.springboot.starter.config.BootstrapProperties;
|
||||||
|
import lombok.RequiredArgsConstructor;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.concurrent.Executor;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dynamic thread-pool subscribe config.
|
||||||
|
*/
|
||||||
|
@RequiredArgsConstructor
|
||||||
|
public class DynamicThreadPoolSubscribeConfig {
|
||||||
|
|
||||||
|
private final ThreadPoolDynamicRefresh threadPoolDynamicRefresh;
|
||||||
|
|
||||||
|
private final ClientWorker clientWorker;
|
||||||
|
|
||||||
|
private final BootstrapProperties properties;
|
||||||
|
|
||||||
|
private final ExecutorService configRefreshExecutorService = ThreadPoolBuilder.builder()
|
||||||
|
.corePoolSize(1)
|
||||||
|
.maxPoolNum(2)
|
||||||
|
.keepAliveTime(2000)
|
||||||
|
.timeUnit(TimeUnit.MILLISECONDS)
|
||||||
|
.workQueue(QueueTypeEnum.SYNCHRONOUS_QUEUE)
|
||||||
|
.allowCoreThreadTimeOut(true)
|
||||||
|
.threadFactory("client.dynamic.threadPool.change.config")
|
||||||
|
.rejected(new ThreadPoolExecutor.AbortPolicy())
|
||||||
|
.build();
|
||||||
|
|
||||||
|
public void subscribeConfig(String threadPoolId) {
|
||||||
|
subscribeConfig(threadPoolId, config -> threadPoolDynamicRefresh.dynamicRefresh(config));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void subscribeConfig(String threadPoolId, ThreadPoolSubscribeCallback threadPoolSubscribeCallback) {
|
||||||
|
Listener configListener = new Listener() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void receiveConfigInfo(String config) {
|
||||||
|
threadPoolSubscribeCallback.callback(config);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Executor getExecutor() {
|
||||||
|
return configRefreshExecutorService;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
clientWorker.addTenantListeners(properties.getNamespace(), properties.getItemId(), threadPoolId, Arrays.asList(configListener));
|
||||||
|
}
|
||||||
|
}
|
@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
||||||
* contributor license agreements. See the NOTICE file distributed with
|
|
||||||
* this work for additional information regarding copyright ownership.
|
|
||||||
* The ASF licenses this file to You 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 cn.hippo4j.springboot.starter.core;
|
|
||||||
|
|
||||||
import cn.hippo4j.springboot.starter.event.ApplicationCompleteEvent;
|
|
||||||
import cn.hippo4j.springboot.starter.remote.HttpAgent;
|
|
||||||
import cn.hippo4j.springboot.starter.remote.ServerHealthCheck;
|
|
||||||
import org.springframework.context.ApplicationListener;
|
|
||||||
|
|
||||||
import java.util.Arrays;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Thread-pool config service.
|
|
||||||
*/
|
|
||||||
public class ThreadPoolConfigService implements ConfigService, ApplicationListener<ApplicationCompleteEvent> {
|
|
||||||
|
|
||||||
private final ClientWorker clientWorker;
|
|
||||||
|
|
||||||
private final ServerHealthCheck serverHealthCheck;
|
|
||||||
|
|
||||||
public ThreadPoolConfigService(HttpAgent httpAgent, String identify, ServerHealthCheck serverHealthCheck) {
|
|
||||||
this.serverHealthCheck = serverHealthCheck;
|
|
||||||
this.clientWorker = new ClientWorker(httpAgent, identify, serverHealthCheck);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void addListener(String tenantId, String itemId, String threadPoolId, Listener listener) {
|
|
||||||
clientWorker.addTenantListeners(tenantId, itemId, threadPoolId, Arrays.asList(listener));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getServerStatus() {
|
|
||||||
if (serverHealthCheck.isHealthStatus()) {
|
|
||||||
return "UP";
|
|
||||||
} else {
|
|
||||||
return "DOWN";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onApplicationEvent(ApplicationCompleteEvent event) {
|
|
||||||
clientWorker.notifyApplicationComplete();
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,54 +0,0 @@
|
|||||||
/*
|
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
||||||
* contributor license agreements. See the NOTICE file distributed with
|
|
||||||
* this work for additional information regarding copyright ownership.
|
|
||||||
* The ASF licenses this file to You 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 cn.hippo4j.springboot.starter.core;
|
|
||||||
|
|
||||||
import cn.hippo4j.springboot.starter.config.BootstrapProperties;
|
|
||||||
|
|
||||||
import java.util.concurrent.Executor;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Thread-pool operation.
|
|
||||||
*/
|
|
||||||
public class ThreadPoolOperation {
|
|
||||||
|
|
||||||
private final ConfigService configService;
|
|
||||||
|
|
||||||
private final BootstrapProperties properties;
|
|
||||||
|
|
||||||
public ThreadPoolOperation(BootstrapProperties properties, ConfigService configService) {
|
|
||||||
this.properties = properties;
|
|
||||||
this.configService = configService;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Listener subscribeConfig(String threadPoolId, Executor executor, ThreadPoolSubscribeCallback threadPoolSubscribeCallback) {
|
|
||||||
Listener configListener = new Listener() {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void receiveConfigInfo(String config) {
|
|
||||||
threadPoolSubscribeCallback.callback(config);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Executor getExecutor() {
|
|
||||||
return executor;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
configService.addListener(properties.getNamespace(), properties.getItemId(), threadPoolId, configListener);
|
|
||||||
return configListener;
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in new issue