封装Angularjs Websocket并支持Promise

最近完成了一个ionic项目,服务端是用的websocket,之前还没应用过ws来做前后端的交互;经过这个项目的实践,将这段代码进行封装,并做简要的说明。

1.添加依赖[$websocket]

这个不多说,这个模块是ng1中使用websocket的不二之选,官方文档也很详细:传送门

官方实例是这么写的:

   angular.module('YOUR_APP', [
     'ngWebSocket' // you may also use 'angular-websocket' if you prefer
   ])
   //                          WebSocket works as well
   .factory('MyData', function($websocket) {
     // Open a WebSocket connection
     var dataStream = $websocket('ws://website.com/data');

     var collection = [];

     dataStream.onMessage(function(message) {
       collection.push(JSON.parse(message.data));
     });

     var methods = {
       collection: collection,
       get: function() {
         dataStream.send(JSON.stringify({ action: 'get' }));
       }
     };

     return methods;
   })
   .controller('SomeController', function ($scope, MyData) {
     $scope.MyData = MyData;
   });

如果你的项目中没有复杂的交互,仅仅是在一个页面中存取数据,这样已经足够了。

但往往项目稍微复杂的应用都无法通过这个来满足。

以下是我个人的简单实现;也仅仅是一种思路的实现,仅供参考 :D

2.将callback转换成promise的实现

思路核心:

  1. push callback to an array => 将callback添加到一个数组中得到一个id
  2. defer with $q.promise => 通过$q来挂起callback
  3. send request with a callbackid => 在发送请求的时候带着这个id
  4. resolve the promise after websocket.onMessage => websocket响应之后通知promise
  5. get the callback from the array with the callbackid => 通过返回的callbackid 从数组中取到这个callback
  6. do the callback code things. 执行想要的代码

3.封装过的代码调用起来是这样的

// 引入模块
angular.module('YourApp', [])
  .controller('AppCtrl', ['$scope', '$rootScope', '$state', '$stateParams', 'wsApi',
    function ($scope, $rootScope, $state, $stateParams, wsApi) {
      //初始化链接信息
      wsApi.init({
    		//your ws server
          endpoint: 'ws://localhost:8090/ws/test/chat',
      });
      //绑定全局的错误回调
      wsApi.bind('error', function () {
        $scope.$emit('WS_ERROR');
      });
		 //绑定全局的关闭回调
      wsApi.bind('close', function () {
        $scope.$emit('WS_CLOSE');
      });
      //建立链接
      wsApi.connect();
      //主动发送数据到服务端
      wsApi.Request({message:'hi there'}).send()
        .then(function(data){
          console.log(data);
        })
        
      //订阅频道
      wsApi.bind('foo', function(data, channel, id){
        console.log(data);
      })
      
  })

y+ws.js文件的代码

"use strict";

(function() {

   angular.module('io.y+.ws', ['ngWebSocket'])
    .factory('yws', ['$websocket', '$q', function ($websocket, $q) {
      //---- options of the lib
      var _options = {
        endpoint: 'ws://localhost:8089',
        channels: []
      };

      //---- websocket instance
      var _dataStream;

      //---- callback arrays
      var _callbacks = {};

      //---- listener arrays
      var _bindListeners = {};

      //---- sequence id for callback identify
      var _promiseId = 0;
      //
      var _Request = function(args){
        this._args = args;
        this._id = ++_promiseId;
        this._response = undefined;
      };

      _Request.prototype.response = function() {
        return this._response;
      };

      _Request.prototype.send = function(){
        var deferred = $q.defer();
        _callbacks[this._id] = deferred;
        this._args.id = this._id;
        _dataStream.send(JSON.stringify(this._args));
        var me = this;
        return deferred.promise.then(function(response) {
          me._response = response;
          return response;
        }).catch(function(err){
          me._error = err;
          return err;
        });
      };

      var funcInit = function(){
        if(_dataStream !== undefined){
          _dataStream.onClose(function(){
            console.log('close');
            var listener = _bindListeners['close'];
            if(listener !== undefined){
              listener(arguments);
            }
          });
          _dataStream.onError(function(){
            console.log('error');
            var listener = _bindListeners['error'];
            if(listener !== undefined){
              listener(arguments);
            }
          });
          _dataStream.onMessage(function(message) {
            var data = message.data;
            //TODO: check json type
            data = JSON.parse(data);
            var id = data.id;
            if(id !== undefined){
              //event callback
              var callback = _callbacks[id];

              if(callback !== undefined){
                delete _callbacks[id];
                callback.resolve(data);
              }else{
                //TODO: no callback handle;
              }
            }else{
              //mybe boardcast callback
              var channel = data.channel;
              if(channel){
                var listener = _bindListeners[channel];
                if(listener !== undefined){
                  listener(data);
                }
              }else{
                //TODO: unknown message;
              }
            }

          });
        }else{
          console.log('Fetal~ DataStream Not Defined!');
        }
      };

      return {
        init: function(options){
          options = options || {};
          for(var k in options){
              _options[k] = options[k];
          }
          _dataStream = $websocket(_options.endpoint, _options.channels);
          funcInit();
        },
        reconnect: function(){
          if(_dataStream){
            _dataStream.reconnect();
          }
        },
        bind: function(channel, func){
          _bindListeners[channel] = func;
        },
        unbind: function(channel){
          delete _bindListeners[channel];
        },
        Request: _Request
      }

   }]);

})();

将其再进行一次封装 api.js

  angular.factory('wsApi', ['yws', function(ws){
    var TAG = 'wsApi.js';
    var _service = {};
    _service.init = function(options){
      ws.init(options);
    };
    _service.bind = function (channel, func) {
      ws.bind(channel, func);
    };
    _service.unbind = function (channel) {
      ws.unbind(channel);
    };
    _service.reconnect = function(){
      ws.reconnect();
    };
    _service.connect = function(){
      ws.connect();
    };
    return _service;
}])