如果一个搜索引擎仅仅是网页搜索,那么将会是非常枯燥的,也不能根据业务需求扩展,还好Iveely在设计之初,就考虑了扩展性,预留插件功能,在不关闭服务或者停用服务的情况下,可以随时启用新插件或者禁用。

首先先介绍下Iveely加载插件的流程,再举例一步一步写插件。

原理

在Iveely.Service下面,存在一个plugin.json文件,Iveely.Service将会每六个小时,更新配置信息,如果plugin.json有更新,将会更新到系统中。Iveey.Service只是一个服务中转站,它将用户的请求,根据请求的命令或者匹配的规则,将请求数据转发给对应插件进行处理,最终将插件处理后的结果返回给用户。plugin.json详细信息介绍如下:

plugin.json文件示例

{
"emailServer":"smtp.126.com",
"emailPort":"25",
"emailName":"Iveely后台服务",
"email": "2@126.com",
"password": "2",
"notify": "liufanping@iveely.com,founder@iveely.com",
"plugins": [
{
"name": "天气预报",
"enable": "1",
"pattern": ".*天气.*",
"command": "weather",
"executeType":"1",
"owner": "1@qq.com",
"ip": "1.iveely.com,5001",
"backup": "1.iveely.com,5001"
},
{
"name": "计算器",
"enable": "1",
"pattern": ".*等于.*",
"command": "calculate",
"executeType":"2",
"owner": "2@qq.com",
"ip": "2.iveely.com,5002",
"backup": "2.iveely.com,5002"
}]
}

下面将以搜索引擎的计算功能为例,为Iveely新增计算插件,首先效果图如下:

     示例

第一步:新建纯Java应用程序工程。

第二步:添加相应的库。

由于插件是以网络方式存在的,因此添加必要的网络库是必然的,其次是基于Iveely.Framework存在。

这些jar文件,均可以在Iveely.Framework工程中找到。

第三步:编写计算器计算类:Calculate类。

/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package com.iveely.plugins.calculator; import java.util.Collections;
import java.util.Stack; /**
*
* @author X1 Carbon
*/
public class Calculate { /**
* Algorithm helper.
*
* @author liufanping@iveely.com
* @date 2014-11-16 10:38:07
*/
public static class ArithHelper { /**
* The default precision division.
*/
private static final int DEF_DIV_SCALE = 16; private ArithHelper() {
} /**
* Provide accurate addition.
*
* @param v1
* @param v2
* @return
*/
public static double add(double v1, double v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(Double.toString(v1));
java.math.BigDecimal b2 = new java.math.BigDecimal(Double.toString(v2));
return b1.add(b2).doubleValue();
} /**
* Provide accurate addition.
*
* @param v1
* @param v2
* @return
*/
public static double add(String v1, String v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(v1);
java.math.BigDecimal b2 = new java.math.BigDecimal(v2);
return b1.add(b2).doubleValue();
} /**
* Provide accurate subtraction.
*
* @param v1
* @param v2
* @return
*/
public static double sub(double v1, double v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(Double.toString(v1));
java.math.BigDecimal b2 = new java.math.BigDecimal(Double.toString(v2));
return b1.subtract(b2).doubleValue();
} /**
* Provide accurate subtraction.
*
* @param v1
* @param v2
* @return
*/
public static double sub(String v1, String v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(v1);
java.math.BigDecimal b2 = new java.math.BigDecimal(v2);
return b1.subtract(b2).doubleValue();
} /**
* Provides accurate multiplication.
*
* @param v1
* @param v2
* @return
*/
public static double mul(double v1, double v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(Double.toString(v1));
java.math.BigDecimal b2 = new java.math.BigDecimal(Double.toString(v2));
return b1.multiply(b2).doubleValue();
} /**
* Provides accurate multiplication.
*
* @param v1
* @param v2
* @return
*/
public static double mul(String v1, String v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(v1);
java.math.BigDecimal b2 = new java.math.BigDecimal(v2);
return b1.multiply(b2).doubleValue();
} /**
* Provide (relatively) precise division, except when the situation
* occurs when the endless, accurate to 10 decimal point, after the
* figures are rounded.
*
* @param v1
* @param v2
* @return
*/
public static double div(double v1, double v2) {
return div(v1, v2, DEF_DIV_SCALE);
} /**
* Provide (relatively) precise division, except when the situation
* occurs when the endless, accurate to 10 decimal point, after the
* figures are rounded.
*
* @param v1
* @param v2
* @return
*/
public static double div(String v1, String v2) {
java.math.BigDecimal b1 = new java.math.BigDecimal(v1);
java.math.BigDecimal b2 = new java.math.BigDecimal(v2);
return b1.divide(b2, DEF_DIV_SCALE, java.math.BigDecimal.ROUND_HALF_UP).doubleValue();
} /**
* Providing (relatively) accurate division. When occurrence except
* endless, specify the scale parameter accuracy, after rounding
* numbers.
*
* @param v1
* @param v2
* @param scale。
* @return
*/
public static double div(double v1, double v2, int scale) {
if (scale < 0) {
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
java.math.BigDecimal b1 = new java.math.BigDecimal(Double.toString(v1));
java.math.BigDecimal b2 = new java.math.BigDecimal(Double.toString(v2));
return b1.divide(b2, scale, java.math.BigDecimal.ROUND_HALF_UP).doubleValue();
} /**
* Provides precise decimals rounded handle.
*
* @param v
* @param scale
* @return
*/
public static double round(double v, int scale) {
if (scale < 0) {
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
java.math.BigDecimal b = new java.math.BigDecimal(Double.toString(v));
java.math.BigDecimal one = new java.math.BigDecimal("1");
return b.divide(one, scale, java.math.BigDecimal.ROUND_HALF_UP).doubleValue();
} /**
* Provides precise decimals rounded handle.
*
* @param v
* @param scale
* @return
*/
public static double round(String v, int scale) {
if (scale < 0) {
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
java.math.BigDecimal b = new java.math.BigDecimal(v);
java.math.BigDecimal one = new java.math.BigDecimal("1");
return b.divide(one, scale, java.math.BigDecimal.ROUND_HALF_UP).doubleValue();
}
} /**
* Postfix stack.
*/
private final Stack<String> postfixStack; /**
* Operator Stack.
*/
private final Stack<Character> opStack; /**
* Operators use the ASCII -40 indexing of operator precedence.
*/
private final int[] operatPriority; public Calculate() {
postfixStack = new Stack<>();
opStack = new Stack<>();
operatPriority = new int[]{0, 3, 2, 1, -1, 1, 0, 2};
} /**
* According to the given expression evaluates.
*
* @param expression
* @return
*/
public String calculate(String expression) {
try {
Stack<String> resultStack = new Stack<>();
prepare(expression);
Collections.reverse(postfixStack);
String firstValue, secondValue, currentValue;
while (!postfixStack.isEmpty()) {
currentValue = postfixStack.pop();
if (!isOperator(currentValue.charAt(0))) {
resultStack.push(currentValue);
} else {
secondValue = resultStack.pop();
firstValue = resultStack.pop();
String tempResult = calculate(firstValue, secondValue, currentValue.charAt(0));
resultStack.push(tempResult);
}
}
return expression + "=" + Double.valueOf(resultStack.pop());
} catch (NumberFormatException e) {
}
return "";
} /**
* Be converted into postfix expression stack.
*
* @param expression
*/
private void prepare(String expression) {
opStack.push(',');
char[] arr = expression.toCharArray();
int currentIndex = 0;
int count = 0;
char currentOp, peekOp;
for (int i = 0; i < arr.length; i++) {
currentOp = arr[i];
if (isOperator(currentOp)) {
if (count > 0) {
postfixStack.push(new String(arr, currentIndex, count));
}
peekOp = opStack.peek();
if (currentOp == ')') {
while (opStack.peek() != '(') {
postfixStack.push(String.valueOf(opStack.pop()));
}
opStack.pop();
} else {
while (currentOp != '(' && peekOp != ',' && compare(currentOp, peekOp)) {
postfixStack.push(String.valueOf(opStack.pop()));
peekOp = opStack.peek();
}
opStack.push(currentOp);
}
count = 0;
currentIndex = i + 1;
} else {
count++;
}
}
if (count > 1 || (count == 1 && !isOperator(arr[currentIndex]))) {
postfixStack.push(new String(arr, currentIndex, count));
}
while (opStack.peek() != ',') {
postfixStack.push(String.valueOf(opStack.pop()));
}
} /**
* Determine whether the arithmetic sign.
*
* @param c
* @return
*/
private boolean isOperator(char c) {
return c == '+' || c == '-' || c == '*' || c == '/' || c == '(' || c == ')';
} /**
* Use ASCII code -40 subscript to do arithmetic signs priority.
*
* @param cur
* @param peek
* @return
*/
public boolean compare(char cur, char peek) {
boolean result = false;
if (operatPriority[(peek) - 40] >= operatPriority[(cur) - 40]) {
result = true;
}
return result;
} /**
* According to the given arithmetic operators to do the calculation.
*
* @param firstValue
* @param secondValue
* @param currentOp
* @return
*/
private String calculate(String firstValue, String secondValue, char currentOp) {
String result = "";
switch (currentOp) {
case '+':
result = String.valueOf(ArithHelper.add(firstValue, secondValue));
break;
case '-':
result = String.valueOf(ArithHelper.sub(firstValue, secondValue));
break;
case '*':
result = String.valueOf(ArithHelper.mul(firstValue, secondValue));
break;
case '/':
result = String.valueOf(ArithHelper.div(firstValue, secondValue));
break;
default:
result = "error format.";
}
return result;
}
}

Calculate类是用于服务的,那么它是怎么提供对外服务呢?

第四步:新建EventHandler类:用于消息处理。

/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package com.iveely.plugins.calculator; import com.iveely.framework.net.ICallback;
import com.iveely.framework.net.InternetPacket;
import java.io.UnsupportedEncodingException; /**
*
* @author 凡平
*/
public class EventHandler implements ICallback { /**
* Weather forecast.
*/
private Calculate calculate; public EventHandler() {
this.calculate = new Calculate();
} @Override
public InternetPacket invoke(InternetPacket packet) {
InternetPacket respPacket = new InternetPacket();
respPacket.setMimeType(0);
respPacket.setExecutType(packet.getExecutType() * -1);
if (packet.getExecutType() == 2) {
try {
String query = getString(packet.getData());
System.out.println("计算表达式:" + query);
String result = this.calculate.calculate(query);
if (result.isEmpty()) {
respPacket.setExecutType(Integer.MIN_VALUE);
respPacket.setData(getBytes("Expression error."));
}else{
respPacket.setData(getBytes(result));
} } catch (Exception e) {
respPacket.setExecutType(Integer.MIN_VALUE);
respPacket.setData(getBytes(e.getMessage()));
}
return respPacket;
} else {
return InternetPacket.getUnknowPacket();
}
} /**
* Convert string to byte[].
*
* @param content
* @return
*/
private byte[] getBytes(String content) {
byte[] bytes;
try {
bytes = content.getBytes("UTF-8");
} catch (UnsupportedEncodingException ex) {
bytes = content.getBytes();
}
return bytes;
} /**
* Convert byte[] to string.
*
* @param bytes
* @return
*/
private String getString(byte[] bytes) {
try {
return new String(bytes, "UTF-8").trim();
} catch (UnsupportedEncodingException ex) {
return new String(bytes).trim();
}
}
}

这里面有几个注意的事项:

1. 一定要继承 ICallback。

2. packet.getExecutType() == 2 表示它的执行类型,需要在plugin.json 中配置。

第五步:启动插件服务。

/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package com.iveely.plugins.calculator; import com.iveely.framework.net.Server; /**
*
* @author 凡平
*/
public class Program { /**
* @param args the command line arguments
*/
public static void main(String[] args) {
int port = 5002;
System.out.println("Server started, port = " + port);
EventHandler handler = new EventHandler();
Server server = new Server(handler, port);
server.start();
} }

按照上面,五个步骤,你就完成了一个最基本的插件编写,生成jar包,找一台机器,将它运行起来,要让搜索引擎能够正常使用,还需要让搜索引擎知道这个插件,那么需要配置plugin.json文件。

如何编写Iveely搜索引擎插件的更多相关文章

  1. 自己编写Android Studio插件 别停留在用的程度了(转载)

    转自:自己编写Android Studio插件 别停留在用的程度了 1概述 相信大家在使用Android Studio的时候,或多或少的会使用一些插件,适当的配合插件可以帮助我们提升一定的开发效率,更 ...

  2. 用jquery编写的分页插件

    用jquery编写的分页插件 源码 function _pager_go(total_page) { var page_str = $("#_pager_textbox").val ...

  3. 用jquery编写的tab插件

    用jquery编写的tab插件 源码 $.fn.ss_tab = function (options) { var box = $(this); var btns = $(this).find(&qu ...

  4. 如何自己编写一个easyui插件续

    接着如何自己编写一个easyui插件继续分享一下如何从上一节写的“hello”插件继承出一个“hello2”. 参考了combobox的源码中继承combo,当然我这个简单很多了.都是根据自己的理解来 ...

  5. 用Javascript编写Chrome浏览器插件

    原文:http://homepage.yesky.com/62/11206062.shtml 用Javascript编写Chrome浏览器插件 2010-04-12 07:30 来源:天极网软件频道 ...

  6. 如何编写一个gulp插件

    很久以前,我们在"细说gulp"随笔中,以压缩JavaScript为例,详细地讲解了如何利用gulp来完成前端自动化. 再来短暂回顾下,当时除了借助gulp之外,我们还利用了第三方 ...

  7. 使用canvas编写时间轴插件

    使用canvas编写时间轴插件 背景 项目中有一个视频广场的功能,需要一个时间轴类似视频播放中进度条功能一样显示录像情况,并且可以点击.拖动.放大缩小展示时间轴,获取到时间轴的某个时间.原来的时间轴是 ...

  8. JQuery编写自己的插件(七)

    一:jQuery插件的编写基础1.插件的种类编写插件的目的是给一系列已经方法或函数做一个封装,以便在其他地方重复使用,方便后期维护和提高开发效率.常见的种类有以下三种:封装对象方法的插件

  9. 从零开始编写一个vue插件

    title: 从零开始编写一个vue插件 toc: true date: 2018-12-17 10:54:29 categories: Web tags: vue mathjax 写毕设的时候需要一 ...

随机推荐

  1. 问题解决——VS2010 将生成的文件复制到指定位置

    我是从VC6直接过渡到VS2010的,VS2008没怎么用过.用VS2010的时候,每次生成dll后,手工把dll.lib..h文件复制到指定文件夹太麻烦了,所以着手写了这个. =========== ...

  2. ASP.NET中iframe框架点击左边页面链接,右边显示链接页面内容

    首先是主页面main.aspx <body style="background-color: #AFEEEE"> <form id="form1&quo ...

  3. Android adb 无线调试

    转自:使用WIFI连接android进行调试和adb操作 1. 手机端开启adb tcp连接端口,下载android终端app(终端模拟器) :/$su:/$setprop service.adb.t ...

  4. [分享]4412开发板Android教程——Android开发环境搭建

    分享视频链接:http://pan.baidu.com/s/1o6tgEeQ 本文转载:www.topeetboard.com 下载和安装Android应用的开发环境 新建Android虚拟环境 An ...

  5. ETL from hadoop to vertica

    根据项目需要,我做了一个POC(proof of concept),XML TXT的数据从HADOOP 引入到VERTICA. 我采用的方案是pig,具体信息可以参加vertica官方的文档. Acc ...

  6. [转]GridView排序——微软提供Sort

    本文转自:http://www.cnblogs.com/eva_2010/articles/1995646.html 在GridView中,根据其中的某列进行排序. 1. 页面:AllowSortin ...

  7. [转]EXCEL如何使用动态公式

    本文转自:http://tech.cncms.com/ruanjian/office/excel/95440.html 也许大家可能还不知道Excel中的动态公式是什么,所谓的动态公式,不是普通的公式 ...

  8. Golang tips ----- 函数

    1.在函数调用时,Golang没有默认参数值 2.一个函数声明如果没有函数体,表面该函数不是由Golang实现的,这样的声明定义了函数标识符 3.拥有函数名的函数只能在包级语法块中被声明 4.函数值( ...

  9. ckplayer.js视频播放插件

    网页中常见的功能就是播放视频,下面介绍的这个ckplayer.js既可以在pc端播放,也可以在手机网页上播放. 可调用flash也可以调用html5播放器: <div id="a1&q ...

  10. 使用Unity3D的50个技巧:Unity3D最佳实践

    翻译故事 原文:http://devmag.org.za/2012/07/12/50-tips-for-working-with-unity-best-practices/ 这篇技巧,我自己也在翻译, ...