聊天机器人

前一段时间,突发奇想,开发了一个聊天机器人,本来是想做helpdesk的一个平台。support部门的人先教它,然后有问题的人可以像它请教。

教的越多它越聪明。

但是公司一直没有机器来放,只好自己搞着玩一玩。

作为公司用,只有出现解决方案才有作用。也就是当你问它比较专业的问题时,它会告诉你如何解决,解决方案1,2,3并且要出现图解的内容。所以,最好查找到相应的内容,机器人给你一个网页地址,上面包含你要查找的内容。

好了,不多说了,把开发步骤介绍一下。

(一)分析流程,需求,建立数据库。

SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[sogo_link]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[sogo_link](
 [id] [int] IDENTITY(1,1) NOT NULL,
 [sogo_link_url] [nvarchar](100) NULL,
 [sogo_link_font] [nvarchar](100) NULL,
 [sogo_link_login] [nvarchar](100) NULL,
 [sogo_link_alt] [nvarchar](100) NULL,
 [sogo_link_logo_or_txt] [int] NULL,
 [sogo_link_yes_no] [int] NULL
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[robot]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[robot](
 [ID] [int] IDENTITY(1,1) NOT NULL,
 [title] [nvarchar](50) NULL,
 [robot_content] [nvarchar](4000) NULL,
 [BigClassName] [nvarchar](50) NULL,
 [SmallClassName] [nvarchar](50) NULL,
 [kin_demo] [nvarchar](4000) NULL,
 [picUrl] [nvarchar](255) NULL,
 [imagenum] [int] NULL,
 [firstImageName] [nvarchar](50) NULL,
 [robot_user] [nvarchar](50) NULL,
 [infotime] [datetime] NULL CONSTRAINT [DF_robot_infotime]  DEFAULT (getdate()),
 [hits] [int] NULL,
 [ok] [bit] NULL,
 [tuijian] [bit] NULL,
 [web_yes_no] [bit] NOT NULL,
 CONSTRAINT [PK_robot] PRIMARY KEY CLUSTERED
(
 [ID] ASC
)WITH (IGNORE_DUP_KEY = OFF) ON [PRIMARY]
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[Admin]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[Admin](
 [ID] [int] IDENTITY(1,1) NOT NULL,
 [admin] [nvarchar](50) NULL,
 [password] [nvarchar](50) NULL,
 [aleave] [nvarchar](50) NULL
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[bigClass]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[bigClass](
 [BigClassID] [int] NOT NULL,
 [BigClassName] [nvarchar](50) NULL,
 [Admin] [nvarchar](100) NULL
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[shop_pinglun]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[shop_pinglun](
 [pinglunid] [int] NOT NULL,
 [ID] [int] NULL,
 [pinglunname] [nvarchar](50) NULL,
 [pinglundate] [datetime] NULL,
 [pingluncontent] [nvarchar](max) NULL
) ON [PRIMARY]
END
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = OBJECT_ID(N'[dbo].[SmallClass]’) AND type in (N’U’))
BEGIN
CREATE TABLE [dbo].[SmallClass](
 [SmallClassID] [int] NOT NULL,
 [SmallClassName] [nvarchar](50) NULL,
 [BigClassName] [nvarchar](50) NULL,
 [Admin] [nvarchar](100) NULL
) ON [PRIMARY]
END

(二)建立解决方案网页。

(三)建立web版机器人

(四)申请小I机器人,下载原程序进行修改。

using System;
using System.IO;
using System.Drawing;
using System.Configuration;
using System.Runtime.Serialization;
using System.Collections.Specialized;
using System.Data;
using System.Data.SqlClient;
using System.Xml;
using System.Xml.Serialization;
using System.Text;
using System.Runtime.Serialization.Formatters.Binary;
using System.Threading;

using Incesoft.BotPlatform.SDK;
using Incesoft.BotPlatform.SDK.Interface;

namespace BotPlatformConsoleServer
{
 public class MyRobot
 {
  public string address;

  public int port;

  public string user;

  public string password;

  static void Main(string[] args)
  {
   MyRobot me = new MyRobot();
   me.address = "msnbot.incesoft.com";
   me.port = 6602;
   me.user = "SP092231";
   me.password = "******";
   
   IRobotServer server = RobotServerFactory.Instance.createRobotServer(me.address, me.port);
   server.addRobotHandler(new MyHandler(server));
   server.addConnectionListener(new MyListener());
   try
   {
    server.login(me.user, me.password, 300000);
    server.requestContactList(haigangdeng@126.com);
   }
   catch (RobotException e)  
   {
    Console.Out.WriteLine("Failed to connect :" + e.Message);
   }
  }
 }

 class MyHandler : IRobotHandler
 {
  public MyHandler(IRobotServer server)
  {
   this.server = server;
  }
  private IRobotServer server;
  private Random rdm = new Random();

        public System.String commandList =  "Hi,你好,我是helpdesk小M。r" +
                                                            "如果你有************问题,可以问我。r" +
                                                            "也许我能帮你解答。";

  public System.String commandListEnt = "*****************************************r" +
                  "**  BOTPLATFORM SDK ENTERPRISE ONLY COMMAND LIST  **r" +
                  "**  Only use the following commands after upgraded your sp account  **r" +
                  "
*****************************************r" +
                  " file —— test file transfer.r" +
                  " bg ——– test background sharing.r" +
                  " ink ——- test send ink.r" +
                  " wink —— test send wink.r" +
                  " voice —– test send voiceclip.r" +
                  " webcam —- test send webcamr" +
                  " cs <account> ——– test create session.r" +
                  " pu <account> —— test push offline message.r" +
                  " iv <account> ——– test invite user.r" +
                  " ent ——— print this command list.";
 
  public virtual void  sessionOpened(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: sessionOpened");
   try
   {
    switch(session.OpenMode)
    {
     case SessionOpenMode.OPEN_MODE_CONV_OPEN:
      session.send(commandList);
      break;
     case SessionOpenMode.OPEN_MODE_ROBOT:
      session.send("You have an alert message: blablabla");
      break;
     default:
      break;
    }
   }
   catch (RobotException e)
   {
    util.consoleOut(e.ToString());
   }
  }
  
  
  public virtual void  sessionClosed(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: sessionClosed");
  }
  
  public virtual void  messageReceived(IRobotSession session, IRobotMessage message)
  {
   System.Console.Out.WriteLine("EVENT: messageReceived");
   
   try
   {
    System.String command = message.String;
    
    IRobotMessage msg = session.createMessage();

                if ("+".ToString().Equals(command[0].ToString()))
                {
                    if (command.IndexOf(‘|’) < 0)
                    {
                        session.send("问题和答案要用“|”分开吆");
                    }
                    else
                    {
                        string wenti = command.Substring(1, (command.IndexOf(‘|’) – 1));
                        string daan = command.Substring(command.IndexOf(‘|’) + 1);

                        string connectionstrings = "server=IP*****;UID=robot;PWD=******;DataBase=robot";
                        SqlConnection connection = new SqlConnection(connectionstrings);
                        string sqlxuexi = "insert into robot (title,robot_content,web_yes_no) values (" + "’" + wenti + "’," + "’" + daan + "’," + 0 + ")";
                        SqlCommand commxuexi = new SqlCommand(sqlxuexi,connection);
                        connection.Open();
                        commxuexi.ExecuteNonQuery();
                        connection.Close();

                        session.send("嗯,我记住了,谢谢你。");
                    }
                }

                else if ("help".ToUpper().Equals(command.ToUpper()) || "?".ToUpper().Equals(command.ToUpper()))
                {
                    session.send(commandList);
                }
                else if ("nudge".ToUpper().Equals(command.ToUpper()))
                {
                    session.sendNudge();
                }
    else if ("bye".ToUpper().Equals(command.ToUpper()))
    {
     session.cl
ose();
    }    
    else
                {
                    string connectionstrings = "server=IP******;UID=robot;PWD=******;DataBase=robot";
                    SqlConnection connection = new SqlConnection(connectionstrings);
                    string sqlcontent = "select * from robot where title like" + "’%" + command.ToString() + "%’";
                    SqlDataAdapter myad = new SqlDataAdapter(sqlcontent, connection);
                    connection.Open();

                    DataSet myds = new DataSet();
                    myad.Fill(myds,"robot");

                    if (myds.Tables["robot"].Rows.Count > 0)
                    {
                        int i = 0;
                        while (i < myds.Tables["robot"].Rows.Count)
                        {
                            int f = i + 1;
                            if (myds.Tables["robot"].Rows[i]["web_yes_no"].GetHashCode() == 0)
                            {
                                session.send(myds.Tables["robot"].Rows[i]["robot_content"].ToString());
                            }
                            else
                            {
                                session.send("解决方案(" + f +")n" + "http://www.xxxxx.com/robot/onews.asp?id=" + myds.Tables["robot"].Rows[i]["ID"].ToString());
                            }
                            i = i + 1;
                        }
                    }
                    else
                    {
                        session.send("对不起,我还不知道这个问题的答案,如果你知道你能告诉我吗?用“+问题|答案”的方式告诉我,我就能记住了。");
                    }
                    connection.Close();
                }
   }
   catch (RobotException e)
   {
    util.consoleOut(e.ToString());
   }   
  }
  
  public virtual void  nudgeReceived(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: nudgeReceived");
  }
  
  
  public virtual void  activityAccepted(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: activityAccepted");
  }
  
  
  public virtual void  activityRejected(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: activityRejected");
  }

  public virtual void userAdd(String robot, String user)
  {
   System.Console.Out.WriteLine("EVENT: userAdd");
  }
  
  public virtual void userRemove(String robot, String user)
  {
   System.Console.Out.WriteLine("EVENT: userRemove");
  }
  
  public virtual void  exceptionCaught(IRobotSession session, System.Exception cause)
  {
   System.Console.Out.WriteLine("SERVER ERROR: " + cause.Message);
  }

  public virtual void  activityClosed(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: activityClosed");
  }
  public virtual void  fileAccepted(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: fileAccepted");
  }
  
  public virtual void  fileRejected(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: fileRejected");
  }
  
  public virtual void  fileTransferEnded(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: fileTransferEnded");
  }
  
  public virtu
al void  backgroundAccepted(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: backgroundAccepted");
  }
  
  public virtual void  backgroundRejected(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: backgroundRejected");
  }
  
  public virtual void  backgroundTransferEnded(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: backgroundTransferEnded");
  }
  
  public virtual void  webcamAccepted(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: webcamAccepted");
  }
  
  public virtual void  webcamRejected(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: webcamRejected");
  }
    
  public virtual void  activityLoaded(IRobotSession session)
  {
   System.Console.Out.WriteLine("EVENT: activityLoaded");
  }

  public virtual void  activityReceived(IRobotSession session, System.String data)
  {
   System.Console.Out.WriteLine("EVENT: activityReceived:" + data);
  }

  public virtual void  userJoined(IRobotSession session, IRobotUser user)
  {
   System.Console.Out.WriteLine("EVENT: userJoined : " + user);
  }

  public virtual void  userLeft(IRobotSession session, IRobotUser user)
  {
   System.Console.Out.WriteLine("EVENT: userLeft :" + user);
  }
  
  public virtual void  userUpdated(IRobotUser user)
  {
   System.Console.Out.WriteLine("EVENT: userUpdated (" + user.ID +","+user.Status+","+user.FriendlyName+ ")");
  }
  
  public virtual void  personalMessageUpdated(System.String robot, System.String user, System.String personalMessage)
  {
   System.Console.Out.WriteLine("EVENT: personalMessageUpdated (" + robot + ", " + user + ", " + personalMessage + ")");
  }   
  public virtual void contactListReceived(System.String robot, System.Collections.ArrayList contactList)
  {
   System.Console.Out.WriteLine("EVENT: contactListRecieved ");
   for (int i=0; i<contactList.Count; i++)
   {
    IRobotUser user = (IRobotUser)contactList[i];
    System.Console.Out.WriteLine("User "+i+": "+user.ID);
   }
  } 
  
 }
 class MyListener : IRobotConnectionListener
 {
  public void serverConnected(IRobotServer server)
  {
   Console.Out.WriteLine("Server connected.");
  }

  public void serverReconnected(IRobotServer server)
  {
   Console.Out.WriteLine("Server reconnected.");
  }

  public void serverDisconnected(IRobotServer server)
  {
   Console.Out.WriteLine("Server disconnected.");
  }

  public void serverLoggedIn(IRobotServer sever)
  {
   Console.Out.WriteLine("Server logged in.");
  }
 }
}

机器人

第一,添加引用

项目->添加引用

 

第二,工具箱添加

工具->选择工具箱项

 Microsoft Agent是微软公司于1997年9月发布的一项代理软件开发技术,而后被集成到Internet Explorer 4.0 Plus及更高版本中,目前版本为2.0,支持简体中文。作为一种软件代理工具,Agent以其强大的交互功能、鲜明的人性特点、优美的操作界面和简便统一的编程方法,对由菜单、按钮、提示框等组成的传统人机交互方式产生了很大影响,在多媒体创作、Web应用、教育软件、软件帮助系统和辅助工具制作等方面,具有广阔的应用前景。

一、 Microsoft Agent技术简介

—- Agent一词的中文意思是“代理”,故名思意,这类代理软件的主要作用是提供一种易于理解和使用的操作界面,接受用户的指令、代替用户完成某些复杂繁琐的工作、或为用户提供帮助。科学研究表明:从人机工程的角度考虑,赋予电脑或程序更多人性化色彩,如支持语音合成输出信息、语音识别输入指令、智能提示、动画等,能够充分提高人机交互的有效性和易用性。

—- 我们知道,在Office 97帮助系统中有一种叫作Office助手的代理软件,其中幽默风趣的动画人物,如“大眼夹”,能够根据用户的操作步骤“智能化”地提出一些建议或帮助信息。与Office助手相比,Agent功能更加丰富强大,表现在:

Office助手沿用传统的不透明矩形窗口,窗口后面的背景被完全挡住;Agent动画人物本身是一个“AlwaysOnTop”窗口,只有人物身体部分挡住背景,其余部分透明。

Office助手只允许Office 97的各个组件调用,Agent动画人物可由任何Windows程序调用;

Agent支持文字气球和输入提示条,在输出语音的同时把文字输出至一个卡通式文字气球中。如果电脑系统中安装有Agent语音识别引擎,当用户按住输入键,如Scroll Lock键,在Agent人物下方出现一个输入命令窗口,表示它正在“聆听”用户的语音命令。

Agent提供Genie、Merlin、Bobby和Peedy等动画人物,不仅具有优美诙谐的动作,而且可以通过声卡、麦克风与用户交谈。
二、 Microsoft Agent安装方法
—- Agent是一种ActiveX控件,其服务器程序(C:WINDOWSMSAGENTAgentsvr.exe)运行在Win95/98/NT操作平台上,为其它调用Agent的客户程序提供服务。客户程序可以是利用VB、C++、Delphi、VFP、PB等开发的应用软件,也可以是HTML文件中的Visual Script、Java Script脚本语言代码。

安装Agent核心组件Msagent.exe。

安装Lernout & Hauspie TruVoice Text-to-Speech Engine(完成从文字合成语音的工作,目前版本只支持英语)Cgram.exe。

安装Microsoft Command and Control Speech Engine(提供语音识别功能,目前版本只支持英语)Actcnc.exe。

Agent动画人物人物数据文件。Agent支持单结构角色文件(.acs,角色数据与动画数据存于同一个文件),也支持分离结构角色文件(.acf,角色数据存于.acf中,动画数据存于.aca中)。基于本地硬盘和网络的应用程序可采用.acs文件,基于Web页面的HTML文件在浏览器中访问时会自动调用,习惯上多使用.acs文件。
—- 上述文件可从 http://www.microsoft.com//workshop/imedia/agent/default.asp下载。
三、 Microsoft Agent编程要点

—- 1、 在Delphi中添加Agent控件

—- 在Delphi IDE中选择菜单Component,Import ActiveX Control——在Import ActiveX下的列表框中选择Microsoft Agent Control 2.0(Version 2.0),点击按钮Install——在Install对话框中点击按钮OK——在Confirm对话框中点击按钮Yes——在对话框中点击按钮OK。至此,Agent控件被添加到Delphi组件板ActiveX页上。

—- 2、 在Delphi中调用Agent控件

—- ⑴、在Delphi IDE 中新建Project1(包含Form1),在Form1上放置Agent1(Name属性为MyAgent)、Button1。

—- ⑵、定义变量。

Var
Peedy: IagentCtlCharacterEx;
Request1,Request2: IagentCtlRequest;

—- ⑶Agent主要属性、方法、事件。

—- ①Characters属性与Load、Unload方法:初始化Agent,加载动画数据,卸载动画人物

—- Request1 := MyAgent.Characters.Load( ‘Peedy’, ‘C:WindowsSystemPeedy.acs’ );
—- Peedy := MyAgent.Characters.Character( ‘Peedy’ );
—- MyAgent.Characters.Unload(‘Peedy’);

—- ②Show与Hide方法:显示、隐藏Agent动画人物。参数Fast表示动画人物显示、隐藏的速度。Fast为0表示Agent飞入、飞出,Fast为1表示Agent直接出现、消失,没有动画过程。

—- Peedy.Show(0);
—- Peedy.Hide(1);

—- ③StopAll、Stop方法:停止所有动作。
—- Peedy.StopAll(”);
—- Peedy.Stop(Request);

—- ④Connected属性:Agent是否与Microsoft Agent服务器连接。
—- MyAgent.Connected := True;

—- ⑤Get_Height、Get_Width、Set_Height、Set_Width方法:获得、设置Agent动画人物的高度、宽度。

AgentHight := Peedy.Get_Height;
AgentWidth := Peedy.Get_Width;
Peedy.Set_Height( round(AgentHeight / 2 ));
Peedy.Set_Width( round( AgentWidth / 2 ));

—- ⑥MoveTo方法:参数x,y表示Agent动画人物移动到的坐标,参数Speed表示移动的速度。

—- Peedy.MoveTo( (Screen.Width-Peedy.Get_Width) Div 2, (Screen.Height-Peedy.Get_Height) Div 2, 2 );

—- ⑦Interrupt方法:中断指定角色的动画。在几十种动画动作中,有一类后缀名为“ing” 的动作比较特殊,如“processing”、“searching”、“reading”、“writing”,一旦执行就必须通过调用另外一个动画人物的Interrupt方法才能中断 。角色不能中断自身的动作,否则会导致异常。

Request1 := Robby.Play(‘reading’);
Peedy.Interrupt(Request1);

—- ⑧Wait方法:在多个不同动画人物之间协调动画动作,如等待sRobby的当前动作执行完毕后,Peedy做另外一个动作。

Request1 := Robby.Play(‘read’);
Peedy.Wait(Request1);
Peedy.Play(‘search’);

—- ⑨Play方法:执行指定的动画动作,参数Animation为一常量,表示动画类型。Agent提供了几十种预定义动作,开发软件时只需直接调用这些常量即可。下面简要介绍其中常用的一些常量仅供编程参考。

—- 动作类型 中文含意 动作类型 中文含意 动作类型 中文含意

Acknowledge 承认 LookDown 向下看 Sad 悲伤
Alert 警告 LookDownBlink 向下看眨眼 Search 寻找
Announce 声明 LookUp 向上看 StartListening 开始聆听
Blink 眨眼 LookUpBlink 向下看眨眼 StopListening 停止聆听
Confused 迷惑 LookLeft 向左看 Suggest 建议
Congratulate 祝贺 LookLeftBlink 向左看眨眼 Surprised 吃惊
Decline 拒绝 LookRight 向右看 Think 思考
DontRecognize 不承认 LookRightBlink 向右看眨眼 Wave 挥动
Explain 解释 MoveDown 向下移动 Write 书写
GestureDown 向下姿势 MoveUp 向上移动 Processing 计算……
GestureUp 向上姿势 Mo
veRight 向右移动 Reading 阅读……
GestureLeft 向左姿势 MoveLeft 向左移动 Searching 寻找……
GestureRight 向右姿势 Pleased 高兴 Writing 书写……
GetAttention 获得注意 Read 阅读 Greet
问候 RestPose 恢复初始状态
Request := peedy.Play(‘search’);

—- ⑩Speak方法:Agent通过调用语音合成引擎,支持英语的语音合成输出功能,只需输入英文字符串,Agent就能通过声卡、音箱“说话”。Agent的当前版本尚不具备支持中文语音合成功能,所以只能以.wav文件的形式播放中文。使用时可以将想要说的中文句子预先录好,以.wav形式存储,在程序中调用即可。参数Text表示文字气球中显示的文字,参数url表示.wav文件的路径。

—- Peedy.speak(‘嗨,朋友们,你们好,现在由我为您提供帮助!’,’C:新建文件夹Help.wav’);

—- Peedy.Speak( ‘Do you like my help? If so, Please send me an email! Thank you!’, ” );

—- ⑾Commands.Add方法:在Button1Click事件中为Agent添加自定义语音识别命令,参数const Name表示语音命令的标识,参数Caption表示该命令在输入命令窗口中的显示标题,参数Voice表示该命令对应的语句。当用户通过麦克风向动画人物发出语音命令时,触发Agent的Command事件,因此我们可以另外编写一个过程完成语音识别功能。

procedure TForm1.Button1Click(Sender: TObject);
begin
Peedy.Commands.Add(‘Delphi’,’Run
Delphi’,’Delphi’,True,True);
Peedy.Commands.Add(‘Visual Basic’,’
Run VB’,’VB’,True,True);
Peedy.Show(0);
Peedy.Speak(‘What can I do for you, sir?’,”);
end;