redis插件

This commit is contained in:
mxd 2019-08-27 17:34:02 +08:00
parent 56606342f5
commit 1aa685ff5f
11 changed files with 402 additions and 17 deletions

View File

@ -43,7 +43,7 @@ spider-flow
#### 插件列表
- [x] Selenium插件
- [ ] Redis插件
- [x] Redis插件
- [ ] Mongodb插件
- [ ] Hbase插件
- [x] IP代理池插件

View File

@ -35,5 +35,6 @@
<module>spider-flow-web</module>
<module>spider-flow-selenium</module>
<module>spider-flow-proxypool</module>
<module>spider-flow-redis</module>
</modules>
</project>

31
spider-flow-redis/pom.xml Normal file
View File

@ -0,0 +1,31 @@
<?xml version="1.0"?>
<project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.spiderflow</groupId>
<artifactId>spider-flow</artifactId>
<version>0.0.1</version>
</parent>
<artifactId>spider-flow-redis</artifactId>
<name>spider-flow-selenium</name>
<url>https://gitee.com/jmxd/spider-flow/tree/master/spider-flow-redis</url>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>org.spiderflow</groupId>
<artifactId>spider-flow-api</artifactId>
<version>0.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
</dependency>
</dependencies>
</project>

View File

@ -0,0 +1,95 @@
package org.spiderflow.redis.executor.shape;
import java.util.Map;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.spiderflow.ExpressionEngine;
import org.spiderflow.context.SpiderContext;
import org.spiderflow.executor.ShapeExecutor;
import org.spiderflow.model.Shape;
import org.spiderflow.model.SpiderNode;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
@Component
public class RedisCommandExecutor implements ShapeExecutor{
private static Logger logger = LoggerFactory.getLogger(RedisCommandExecutor.class);
public static final String DATASOURCE_ID = "datasourceId";
public static final String REDIS_COMMAND = "command";
public static final String REDIS_OPERATION_TYPE = "operationType";
public static final String REDIS_COMMAND_VAR = "___redis";
@Autowired
private ExpressionEngine engine;
@Override
public String supportShape() {
return "rediscommand";
}
@Override
public Shape shape() {
Shape shape = new Shape();
shape.setImage("");
shape.setLabel("Redis命令");
shape.setName("rediscommand");
shape.setTitle("Redis命令");
return shape;
}
@Override
public void execute(SpiderNode node, SpiderContext context, Map<String, Object> variables) {
String datasourceId = node.getStringJsonValue(DATASOURCE_ID);
String command = node.getStringJsonValue(REDIS_COMMAND);
String operationType = node.getStringJsonValue(REDIS_OPERATION_TYPE);
if(!StringUtils.isNotBlank(datasourceId)){
context.log("Redis数据源ID为空");
if(logger.isDebugEnabled()){
logger.debug("Redis数据源ID为空");
}
}else if(!StringUtils.isNotBlank(command)){
context.log("redis命令为空");
if(logger.isDebugEnabled()){
logger.debug("redis命令为空");
}
}else if(!StringUtils.isNotBlank(operationType)){
context.log("redis命令类型为空");
if(logger.isDebugEnabled()){
logger.debug("redis命令类型为空");
}
}else{
StringRedisTemplate redisTemplate = (StringRedisTemplate) context.get(RedisExecutor.REDIS_CONTEXT_KEY + datasourceId);
Object operation = getOperations(redisTemplate, operationType);
variables.put(REDIS_COMMAND_VAR, operation);
String expression = String.format("${%s.%s}", REDIS_COMMAND_VAR,command);
Object result = engine.execute(expression, variables);
variables.put("rs", result);
variables.remove(REDIS_COMMAND_VAR);
}
}
private Object getOperations(StringRedisTemplate redisTemplate,String operationType){
switch(operationType){
case "list" : return redisTemplate.opsForList();
case "value" : return redisTemplate.opsForValue();
case "set" : return redisTemplate.opsForSet();
case "zset" : return redisTemplate.opsForZSet();
case "geo" : return redisTemplate.opsForGeo();
case "hash" : return redisTemplate.opsForHash();
case "hyperLogLog" : return redisTemplate.opsForHyperLogLog();
}
return redisTemplate;
}
}

View File

@ -0,0 +1,61 @@
package org.spiderflow.redis.executor.shape;
import java.util.Map;
import org.apache.commons.lang3.math.NumberUtils;
import org.spiderflow.context.SpiderContext;
import org.spiderflow.executor.ShapeExecutor;
import org.spiderflow.model.Shape;
import org.spiderflow.model.SpiderNode;
import org.spiderflow.redis.utils.RedisUtils;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
@Component
public class RedisExecutor implements ShapeExecutor{
public static final String DATABASE_INDEX = "database";
public static final String REDIS_HOST = "host";
public static final String REDIS_PORT = "port";
public static final String REDIS_PASSWORD = "password";
public static final String REDIS_POOL_MAX_ACTIVE = "poolMaxActive";
public static final String REDIS_POOL_MAX_IDLE = "poolMaxIdle";
public static final String REDIS_POOL_MIN_IDLE = "poolMinIdle";
public static final String REDIS_CONTEXT_KEY = "$redis_";
@Override
public String supportShape() {
return "redis";
}
@Override
public Shape shape() {
Shape shape = new Shape();
shape.setImage("");
shape.setLabel("Redis");
shape.setName("redis");
shape.setTitle("Redis");
return shape;
}
@Override
public void execute(SpiderNode node, SpiderContext context, Map<String, Object> variables) {
int database = NumberUtils.toInt(node.getStringJsonValue(DATABASE_INDEX),0);
String host = node.getStringJsonValue(REDIS_HOST);
String password = node.getStringJsonValue(REDIS_PASSWORD);
int port = NumberUtils.toInt(node.getStringJsonValue(REDIS_PORT),6379);
int poolMaxActive = NumberUtils.toInt(node.getStringJsonValue(REDIS_POOL_MAX_ACTIVE),8);
int poolMaxIdle = NumberUtils.toInt(node.getStringJsonValue(REDIS_POOL_MAX_IDLE),8);
int poolMinIdle = NumberUtils.toInt(node.getStringJsonValue(REDIS_POOL_MIN_IDLE),0);
StringRedisTemplate redisTemplate = RedisUtils.createRedisTemplate(host, port, password, database, poolMaxActive, poolMaxIdle, poolMinIdle);
context.put(REDIS_CONTEXT_KEY + node.getNodeId(), redisTemplate);
}
}

View File

@ -0,0 +1,28 @@
package org.spiderflow.redis.utils;
import org.springframework.data.redis.connection.RedisPassword;
import org.springframework.data.redis.connection.RedisStandaloneConfiguration;
import org.springframework.data.redis.connection.jedis.JedisClientConfiguration;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.data.redis.core.StringRedisTemplate;
import redis.clients.jedis.JedisPoolConfig;
public class RedisUtils {
public static StringRedisTemplate createRedisTemplate(String host,int port,String password,int database,int poolMaxActive,int poolMaxIdle,int poolMinIdle){
JedisPoolConfig poolConfig = new JedisPoolConfig();
poolConfig.setMaxTotal(poolMaxActive);
poolConfig.setMaxIdle(poolMaxIdle);
poolConfig.setMinIdle(poolMinIdle);
JedisClientConfiguration jedisConfigConfiguration = JedisClientConfiguration.builder().usePooling().poolConfig(poolConfig).build();
RedisStandaloneConfiguration standaloneConfiguration = new RedisStandaloneConfiguration();
standaloneConfiguration.setDatabase(database);
standaloneConfiguration.setHostName(host);
standaloneConfiguration.setPassword(RedisPassword.of(password));
standaloneConfiguration.setPort(port);
StringRedisTemplate redisTemplate = new StringRedisTemplate(new JedisConnectionFactory(standaloneConfiguration,jedisConfigConfiguration));
return redisTemplate;
}
}

View File

@ -0,0 +1,32 @@
package org.spiderflow.redis.web;
import org.spiderflow.model.JsonBean;
import org.spiderflow.redis.utils.RedisUtils;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.ValueOperations;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/redis")
public class RedisController {
@RequestMapping("/test")
public JsonBean<String> test(String host,Integer port,String password,Integer database){
try {
StringRedisTemplate redisTemplate = RedisUtils.createRedisTemplate(host, port, password, database, 1, 1, 0);
ValueOperations<String, String> operation = redisTemplate.opsForValue();
String testKey = "____spider_flow_redis_test";
operation.set(testKey, "1");
if (redisTemplate.hasKey(testKey)) {
redisTemplate.delete(testKey);
return new JsonBean<String>(1,"测试成功");
}
return new JsonBean<String>(0,"测试失败");
} catch (Exception e) {
e.printStackTrace();
return new JsonBean<String>(-1,e.getMessage());
}
}
}

View File

@ -0,0 +1,96 @@
<div class="layui-tab layui-tab-fixed layui-tab-brief">
<ul class="layui-tab-title">
<li class="layui-this">基本配置</li>
</ul>
<div class="layui-tab-content editor-form-node">
<div class="layui-tab-item layui-show">
<form class="layui-form">
<div class="layui-form-item">
<label class="layui-form-label">节点名称</label>
<div class="layui-input-block">
<input type="text" name="value" placeholder="请输入节点名称" value="{{=d.value}}" autocomplete="off" class="layui-input">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">host</label>
<div class="layui-input-block">
<input type="text" name="host" placeholder="请输入redis host" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.host}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">port</label>
<div class="layui-input-block">
<input type="text" name="port" placeholder="请输入redis port" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.port}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">密码</label>
<div class="layui-input-block">
<input type="password" name="password" placeholder="请输入redis password,没有则留空" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.password}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">数据库索引</label>
<div class="layui-input-block">
<input type="text" name="database" placeholder="请输入redis 数据库索引,默认为0" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.database}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">最大连接数</label>
<div class="layui-input-block">
<input type="text" name="poolMaxActive" placeholder="请输入最大连接数,默认为8" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.poolMaxActive}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">最大空闲连接</label>
<div class="layui-input-block">
<input type="text" name="poolMaxIdle" placeholder="请输入最大空闲连接,默认为8" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.poolMaxIdle}}">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">最小空闲连接</label>
<div class="layui-input-block">
<input type="text" name="poolMinIdle" placeholder="请输入最小空闲连接,默认为0" autocomplete="off" class="layui-input input-default" value="{{=d.data.object.poolMinIdle}}">
</div>
</div>
<div class="layui-form-item">
<div class="layui-input-block">
<button class="layui-btn btn-redis-datasource-test" type="button">测试连接</button>
</div>
</div>
</form>
</div>
</div>
</div>
<script>
$('.layui-form').on('click','.btn-redis-datasource-test',function(){
var $form = $('.layui-form');
var host = $form.find('input[name=host]').val();
var port = $form.find('input[name=port]').val();
var password = $form.find('input[name=password]').val();
var database = $form.find('input[name=database]').val();
$.ajax({
url : 'redis/test',
data : {
host : host || '127.0.0.1',
port : port || 6379,
password : password,
database : database || 0
},
type : 'post',
dataType : 'json',
success : function(json){
if(json.code == 1){
layui.layer.msg('测试成功');
}else{
layui.layer.alert('测试失败,' + json.message || '',{
icon : 2
})
}
},
error : function(){
layui.layer.msg('测试失败');
}
})
});
</script>

View File

@ -0,0 +1,50 @@
<div class="layui-tab layui-tab-fixed layui-tab-brief">
<ul class="layui-tab-title">
<li class="layui-this">基本配置</li>
</ul>
<div class="layui-tab-content editor-form-node">
<div class="layui-tab-item layui-show">
<form class="layui-form">
<div class="layui-form-item">
<label class="layui-form-label">节点名称</label>
<div class="layui-input-block">
<input type="text" name="value" placeholder="请输入节点名称" value="{{=d.value}}" autocomplete="off" class="layui-input">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">数据源</label>
<div class="layui-input-block">
<select name="datasourceId">
{{# for(var datasourceIndex in d.model.cells){ }}
{{# var cell = d.model.cells[datasourceIndex] }}
{{# if(cell.data&&cell.data.get('shape') == 'redis'){ }}
<option value="{{=datasourceIndex}}" {{datasourceIndex == d.data.object.datasourceId ? 'selected': ''}}>{{cell.value}}</option>
{{# } }}
{{# } }}
</select>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">命令类型</label>
<div class="layui-input-block">
<select name="operationType">
<option value="value" {{d.data.object.operationType == 'value' ? 'selected': ''}}>value</option>
<option value="keys" {{d.data.object.operationType == 'keys' ? 'selected': ''}}>keys</option>
<option value="list" {{d.data.object.operationType == 'list' ? 'selected': ''}}>list</option>
<option value="hash" {{d.data.object.operationType == 'hash' ? 'selected': ''}}>hash</option>
<option value="set" {{d.data.object.operationType == 'set' ? 'selected': ''}}>set</option>
<option value="zset" {{d.data.object.operationType == 'zset' ? 'selected': ''}}>zset</option>
<option value="hyperLogLog" {{d.data.object.operationType == 'hyperLogLog' ? 'selected': ''}}>hyperLogLog</option>
</select>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">Redis命令</label>
<div class="layui-input-block">
<textarea class="layui-input" style="height:200px" name="command">{{=d.data.object.command}}</textarea>
</div>
</div>
</form>
</div>
</div>
</div>

View File

@ -32,5 +32,10 @@
<artifactId>spider-flow-proxypool</artifactId>
<version>0.0.1</version>
</dependency>
<dependency>
<groupId>org.spiderflow</groupId>
<artifactId>spider-flow-redis</artifactId>
<version>0.0.1</version>
</dependency>
</dependencies>
</project>

View File

@ -98,7 +98,8 @@ $(function(){
data : cell.data,
value : cell.value,
datasources : datasources,
flows : flows || []
flows : flows || [],
model : model
},function(html){
$(".properties-container").html(html);
layui.form.render();
@ -291,21 +292,6 @@ $(function(){
name : 'process',
image : '',
title : '子流程'
},{
name : 'redis',
image : '',
title : 'redis',
disabled : true
},{
name : 'mongodb',
image : '',
title : 'mongodb',
disabled : true
},{
name : 'oss',
image : '',
title : 'oss',
disabled : true
}]
var addShape = function(shape){
var image = new Image();