XMPP: Openfire + Spark 實(shí)現(xiàn)即時(shí)通訊

資源下載: https://pan.baidu.com/s/1ge3cajh 密碼: 6ytk

安裝XMPP集成環(huán)境

  • 將下載好的xampp-osx根據(jù)提示進(jìn)行安裝并打開。
    F4648516-D77F-4DF4-9031-F7BED0F8C13F.png
  • 點(diǎn)擊Manage Servers,啟動(dòng)所有服務(wù)邻吭,如果MySQL Database無法啟動(dòng)吕粗,可打開終端執(zhí)行如下命令: sudo /Applications/XAMPP/xamppfiles/bin/mysql.server start
    1B424F11-55A2-4F2C-B188-89059822D4CD.png
  • 點(diǎn)擊Go To Application,配置數(shù)據(jù)庫
    A3FE18F4-5CE0-4E8A-8BB6-A7AF97F216D2.png
  • 數(shù)據(jù)庫新建成功后垢乙,需要導(dǎo)入數(shù)據(jù)庫格式⊙雕桑可使用下載好的openfire_mysql.sq文件袄简,也可用另外途徑:前往/usr/local武福,找到文件夾openfire,但此時(shí)的openfire文件夾無法打開痘番。選中后右鍵查看簡介,將下面屬性修改為讀與寫平痰。這時(shí)可以打開openfire文件夾汞舱,查找路徑/usr/local/openfire/resources/database下,找到openfire_mysql.sq文件宗雇,將其拷貝到桌面昂芜,然后打開數(shù)據(jù)庫配置頁面。
    C51E4113-4627-4076-A8AB-0C5A4A092378.png

Openfire服務(wù)器搭建

可以使用它輕易的構(gòu)建高效率的技師通訊服務(wù)器赔蒲,Openfire的安裝和使用也是非常的簡單泌神,并利用Web進(jìn)行后臺(tái)管理良漱。單臺(tái)服務(wù)器可支持上萬并發(fā)用戶。由于是采用開放的XMPP協(xié)議欢际,我們可以使用各種支持XMPP協(xié)議的IM客戶端軟件登陸服務(wù)(在這里我就使用了Spark)母市。

  • 下載Openfire,并根據(jù)提示進(jìn)行安裝损趋。
  • 安裝完成后患久,打開系統(tǒng)設(shè)置。
    2337F0EA-21E2-4864-8322-976E3D084138.png
    3C705089-8D87-44E8-BDCA-AB87D0F18484.png
  • 點(diǎn)擊Open Admin Console,到網(wǎng)站中配置Openfire浑槽。
    7FDF4D15-E4E8-4AA9-AC5C-2D448EACB2AB.png
    6477BDE5-63A9-483A-A333-AFED5A64C13C.png
    D31A34D9-945C-40C6-BF02-0AE5ACCBC123.png
    833FB7BA-32F5-403C-9F78-B21389322536.png
    0602DDA6-5F09-4254-82CF-9E8E3404D753.png
    7B7802D1-FFB2-4D77-A5ED-6462D71E1577.png
  • 到這一步,基本上openfire就已經(jīng)配置完畢了。

安裝Spark客戶端

  • 按照提示安裝客戶端泛源,完成后打開


    E24D0A1D-BC30-4901-8845-D84B40F25EBA.png
  • 高級(jí)配置
    5D1E4204-7E6A-47EB-9219-65DFDA664C7A.png
  • 然后以管理員身份登錄冲秽,登錄成功后刷新openfire管理界面,可以看到頭像變亮了镊靴,說明環(huán)境配置成功铣卡。接下來就可以擼碼了。

代碼實(shí)現(xiàn)

  • 主要實(shí)現(xiàn)這些功能:
    • 注冊邑闲、登錄算行、退出登錄;
    • 添加好友苫耸、好友請求
    • 發(fā)送消息州邢、接收消息
    • 消息記錄
  • 新建一個(gè)管理類XMPPManager,創(chuàng)建以下對象:
import UIKit
import XMPPFramework

// 枚舉:連接服務(wù)器的目的
enum ConnectServerPurpose : Int{
  case connectServerToLogin     // 登錄
  case connectServerToRegister  // 注冊
}

class XMPPManager: NSObject {
  
  deinit {
      NotificationCenter.default.removeObserver(self)
  }
  
  fileprivate var password : String?
  fileprivate var userName : String?
  fileprivate var connectServerPurpose : ConnectServerPurpose = .connectServerToLogin
  
  // 通信通道對象
  var xmppStream : XMPPStream?
  // JID
  var xmppJID : XMPPJID?
  // 好友花名冊管理對象
  var xmppRoster : XMPPRoster?
  // 花名冊數(shù)據(jù)存儲(chǔ)對象
  var xmppRosterCoreDataStorage : XMPPRosterCoreDataStorage?
  // 信息歸檔對象
  var xmppMessageArchiving : XMPPMessageArchiving?
  // 信息存儲(chǔ)對象
  var xmppMessageArchivingCoreDataStorage : XMPPMessageArchivingCoreDataStorage?
  
  var friendsListResultController : NSFetchedResultsController<NSFetchRequestResult>?
  var chatRecordsResultController : NSFetchedResultsController<NSFetchRequestResult>?
  // 好友請求
  var xmppPresence : XMPPPresence?
  
  // 單例
  static let manager : XMPPManager = {
      let manager = XMPPManager.init()
      // 創(chuàng)建通信通道對象
      manager.xmppStream = XMPPStream.init()
      // 設(shè)置服務(wù)器IP地址
      manager.xmppStream?.hostName = kHostName
      // 設(shè)置服務(wù)器端口
      manager.xmppStream?.hostPort = kHostPort
      // 添加代理
      manager.xmppStream?.addDelegate(manager, delegateQueue: DispatchQueue.main)
      
      // 花名冊數(shù)據(jù)存儲(chǔ)對象
      manager.xmppRosterCoreDataStorage = XMPPRosterCoreDataStorage.sharedInstance()
      manager.xmppRoster = XMPPRoster.init(rosterStorage: manager.xmppRosterCoreDataStorage)
      manager.xmppRoster?.activate(manager.xmppStream)
      manager.xmppRoster?.addDelegate(manager, delegateQueue: DispatchQueue.main)
      
      // 信息存儲(chǔ)對象
      manager.xmppMessageArchivingCoreDataStorage = XMPPMessageArchivingCoreDataStorage.sharedInstance()
      manager.xmppMessageArchiving = XMPPMessageArchiving.init(messageArchivingStorage: manager.xmppMessageArchivingCoreDataStorage, dispatchQueue: DispatchQueue.main)
      // 激活通信通道對象
      manager.xmppMessageArchiving?.activate(manager.xmppStream)

      return manager
  }()
  
  // 連接服務(wù)器
  func connectToServer(withUserName userName: String) {
      // 創(chuàng)建XMPPJID對象
      self.xmppJID = XMPPJID.init(user: userName, domain: kDomin, resource: kResource)
      // 設(shè)置通信通道對象的JID
      self.xmppStream?.myJID = self.xmppJID
      // 發(fā)送請求
      if self.xmppStream?.isConnected() == true || self.xmppStream?.isConnecting() == true {
          // 先退出登錄狀態(tài)
         self.exitLogin()
      }
      // 連接服務(wù)器
      do {
          try self.xmppStream?.connect(withTimeout: -1)
      }catch let error as NSError{
          print("連接服務(wù)器失敗: " + error.description)
      }
  }
}

// MARK:- 登錄方法
extension XMPPManager {
  // 登錄方法
  func login(widthUserName userName: String?, andPassword password: String?) {
      guard let userName = userName, let password = password else {
          print("用戶名和密碼不能為空")
          return
      }
      // 記錄連接服務(wù)器的目的是登錄
      connectServerPurpose = .connectServerToLogin
      // 記錄登錄信息
      self.password = password
      self.userName = userName
      connectToServer(withUserName: userName)
  }
  
  // 退出登錄
  func exitLogin() {
      // 先發(fā)送下線狀態(tài)
      let presence = XMPPPresence.init(type: "unavailable")
      self.xmppStream?.send(presence)
      // 斷開連接
      self.xmppStream?.disconnect()
  }
}

// MARK:- 注冊方法
extension XMPPManager {
  func register(widthUserName userName: String?, andPassword password: String?) {
      guard let userName = userName, let password = password else {
          print("用戶名和密碼不能為空")
          return
      }
      // 記錄連接服務(wù)器的目的是注冊
      connectServerPurpose = .connectServerToRegister
      // 記錄密碼
      self.password = password
      connectToServer(withUserName: userName)
  }
}

// MARK:- XMPPStreamDelegate
extension XMPPManager : XMPPStreamDelegate {
  // 連接成功
  func xmppStreamDidConnect(_ sender: XMPPStream!) {
      print("連接成功")
      switch connectServerPurpose {
          case .connectServerToLogin:
              // 驗(yàn)證密碼
              do {
                  try self.xmppStream?.authenticate(withPassword: self.password)
              } catch let error as NSError {
                  print("登錄時(shí)驗(yàn)證密碼失斖首印: " + error.description)
              }
              break
          case .connectServerToRegister:
              do {
                  try self.xmppStream?.register(withPassword: self.password)
              } catch let error as NSError {
                  print("注冊時(shí)驗(yàn)證密碼失斄刻省: " + error.description)
              }
              break
      }
  }
  
  // 連接超時(shí)
  func xmppStreamConnectDidTimeout(_ sender: XMPPStream!) {
      print("連接超時(shí)")
  }
  
  // 登錄成功
  func xmppStreamDidAuthenticate(_ sender: XMPPStream!) {
      print("登錄成功  ", #line, #function)
      // 發(fā)送上線狀態(tài)
      let presence = XMPPPresence.init(type: "available")
      XMPPManager.manager.xmppStream?.send(presence)
  }
  
  // 已經(jīng)斷開連接
  func xmppStreamDidDisconnect(_ sender: XMPPStream!, withError error: Error!) {
      print("++++++++++")
  }
}

// MARK:- 好友
extension XMPPManager {
  // 添加好友
  func addNewFriends(userName: String?) {
      guard let name = userName else { return }
      let friendJID = XMPPJID.init(string: "\(name)@\(kDomin)")
      self.xmppRoster?.subscribePresence(toUser: friendJID)
  }
  
  // 獲取好友列表
  func getFriendsList() -> Array<Any>? {
      guard let context = self.xmppRosterCoreDataStorage?.mainThreadManagedObjectContext else {
          return nil
      }
      let request = NSFetchRequest<NSFetchRequestResult>.init(entityName: "XMPPUserCoreDataStorageObject")
      let userInfo = String.init(format: "%@@%@", self.userName ?? "", kDomin)
      // 謂詞
      let predicate = NSPredicate.init(format: "streamBareJidStr = %@", userInfo)
      request.predicate = predicate
      
      // 排序
      let sort = NSSortDescriptor.init(key: "displayName", ascending: true)
      request.sortDescriptors = [sort]
      
      friendsListResultController = NSFetchedResultsController.init(fetchRequest: request, managedObjectContext: context, sectionNameKeyPath: nil, cacheName: nil)
      friendsListResultController?.delegate = self
      do {
          try friendsListResultController?.performFetch()
      } catch let error as NSError {
          print("獲取好友聊天記錄失敗: " + error.description)
      }
      return friendsListResultController?.fetchedObjects
  }
  
  // 獲取與某個(gè)好友的聊天記錄
  func getChatRecords(withFriendName friendName: String) ->Array<Any>?{
      guard let context = self.xmppMessageArchivingCoreDataStorage?.mainThreadManagedObjectContext else { return nil}
      let request = NSFetchRequest<NSFetchRequestResult>.init(entityName: "XMPPMessageArchiving_Message_CoreDataObject")
      let userInfo = String.init(format: "%@@%@", self.userName ?? "", kDomin)
      let friendsInfo = String.init(format: "%@@%@", friendName, kDomin)
      // 謂詞
      let predicate = NSPredicate.init(format: "streamBareJidStr=%@ AND bareJidStr=%@", userInfo, friendsInfo)
      request.predicate = predicate
      
      // 排序
      let sort = NSSortDescriptor.init(key: "timestamp", ascending: true)
      request.sortDescriptors = [sort]
      
      chatRecordsResultController = NSFetchedResultsController.init(fetchRequest: request, managedObjectContext: context, sectionNameKeyPath: nil, cacheName: nil)
      chatRecordsResultController?.delegate = self
      do {
          try chatRecordsResultController?.performFetch()
      } catch let error as NSError {
          print("獲取好友聊天記錄失斚油省: " + error.description)
      }
      guard let fetchedObjects = chatRecordsResultController?.fetchedObjects else { return nil }
      var messageArray = Array<Any>()
      for obj in fetchedObjects {
          if let _ = (obj as? XMPPMessageArchiving_Message_CoreDataObject)?.body {
              messageArray.append(obj)
          }
      }
      return messageArray
  }
}

// MARK:- XMPPRosterDelegate
extension XMPPManager : XMPPRosterDelegate {
  func xmppRoster(_ sender: XMPPRoster!,
                  didReceiveRosterItem item: DDXMLElement!) {
      print(#line, #function)
  }
  func xmppRosterDidBeginPopulating(_ sender: XMPPRoster!,
                                    withVersion version: String!) {
      print(#line, #function)
  }
  func xmppRosterDidEndPopulating(_ sender: XMPPRoster!) {
      print(#line, #function)
  }
  
  // 收到好友請求
  func xmppRoster(_ sender: XMPPRoster!,
                  didReceivePresenceSubscriptionRequest presence: XMPPPresence!) {
      self.xmppPresence = presence
      // 彈框
      let alert = UIAlertView.init(title: "好友請求", message: "\(presence.from().user)請求添加你為好友", delegate: self, cancelButtonTitle:
          "拒絕", otherButtonTitles: "同意")
      alert.show()
      
  }
}

// MARK:- 好友請求彈框
extension XMPPManager : UIAlertViewDelegate {
  func alertView(_ alertView: UIAlertView,
                 clickedButtonAt buttonIndex: Int) {
      switch buttonIndex {
      case 0:
          // 拒絕
          let jid = XMPPJID.init(string: self.xmppPresence?.from().user)
          self.xmppRoster?.rejectPresenceSubscriptionRequest(from: jid)
          break
      case 1:
          // 同意
          let jid = XMPPJID.init(string: self.xmppPresence?.from().user)
          self.xmppRoster?.acceptPresenceSubscriptionRequest(from: jid, andAddToRoster: true)
          break
      default: break
      }
  }
}

// MARK:- NSFetchedResultsControllerDelegate
extension XMPPManager : NSFetchedResultsControllerDelegate {
  func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>,
                  didChange anObject: Any,
                  at indexPath: IndexPath?,
                  for type: NSFetchedResultsChangeType,
                  newIndexPath: IndexPath?) {
      if anObject is XMPPUserCoreDataStorageObject {
          // 好友列表數(shù)據(jù)庫發(fā)生變化
          NotificationCenter.default.post(name: NSNotification.Name(rawValue: notificationName_friendsListDidChange), object: nil)
      }else if anObject is XMPPMessageArchiving_Message_CoreDataObject{
          // 聊天記錄數(shù)據(jù)庫發(fā)生變化
          NotificationCenter.default.post(name: NSNotification.Name(rawValue: notificationName_chatRecordsDidChange), object: nil)
      }
  }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末呀枢,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子笼痛,更是在濱河造成了極大的恐慌裙秋,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,744評論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件缨伊,死亡現(xiàn)場離奇詭異摘刑,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)刻坊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,505評論 3 392
  • 文/潘曉璐 我一進(jìn)店門枷恕,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人谭胚,你說我怎么就攤上這事徐块∥床#” “怎么了?”我有些...
    開封第一講書人閱讀 163,105評論 0 353
  • 文/不壞的土叔 我叫張陵胡控,是天一觀的道長扳剿。 經(jīng)常有香客問我,道長铜犬,這世上最難降的妖魔是什么舞终? 我笑而不...
    開封第一講書人閱讀 58,242評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮癣猾,結(jié)果婚禮上敛劝,老公的妹妹穿的比我還像新娘。我一直安慰自己纷宇,他們只是感情好夸盟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,269評論 6 389
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著像捶,像睡著了一般上陕。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上拓春,一...
    開封第一講書人閱讀 51,215評論 1 299
  • 那天释簿,我揣著相機(jī)與錄音,去河邊找鬼硼莽。 笑死庶溶,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的懂鸵。 我是一名探鬼主播偏螺,決...
    沈念sama閱讀 40,096評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼匆光!你這毒婦竟也來了套像?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,939評論 0 274
  • 序言:老撾萬榮一對情侶失蹤终息,失蹤者是張志新(化名)和其女友劉穎夺巩,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體周崭,經(jīng)...
    沈念sama閱讀 45,354評論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡柳譬,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,573評論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了休傍。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,745評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡蹲姐,死狀恐怖磨取,靈堂內(nèi)的尸體忽然破棺而出人柿,到底是詐尸還是另有隱情,我是刑警寧澤忙厌,帶...
    沈念sama閱讀 35,448評論 5 344
  • 正文 年R本政府宣布凫岖,位于F島的核電站,受9級(jí)特大地震影響逢净,放射性物質(zhì)發(fā)生泄漏哥放。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,048評論 3 327
  • 文/蒙蒙 一爹土、第九天 我趴在偏房一處隱蔽的房頂上張望甥雕。 院中可真熱鬧,春花似錦胀茵、人聲如沸社露。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,683評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽峭弟。三九已至,卻和暖如春脱拼,著一層夾襖步出監(jiān)牢的瞬間瞒瘸,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,838評論 1 269
  • 我被黑心中介騙來泰國打工熄浓, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留情臭,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,776評論 2 369
  • 正文 我出身青樓玉组,卻偏偏與公主長得像谎柄,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子惯雳,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,652評論 2 354

推薦閱讀更多精彩內(nèi)容