const JSONRPCWebSocket = require('../util/jsonrpc-web-socket'); const log = require('../util/log'); const ScratchLinkWebSocket = 'ws://localhost:20110/scratch/bt'; class BTSession extends JSONRPCWebSocket { /** * A BT device session object. It handles connecting, over web sockets, to * BT devices, and reading and writing data to them. * @param {Runtime} runtime - the Runtime for sending/receiving GUI update events. * @param {object} deviceOptions - the list of options for device discovery. * @param {object} connectCallback - a callback for connection. * @param {object} messageCallback - a callback for message sending. */ constructor (runtime, deviceOptions, connectCallback, messageCallback) { const ws = new WebSocket(ScratchLinkWebSocket); super(ws); this._ws = ws; this._ws.onopen = this.requestDevice.bind(this); // only call request device after socket opens this._ws.onerror = this._sendError.bind(this, 'ws onerror'); this._ws.onclose = this._sendError.bind(this, 'ws onclose'); this._availablePeripherals = {}; this._connectCallback = connectCallback; this._characteristicDidChangeCallback = null; this._deviceOptions = deviceOptions; this._messageCallback = messageCallback; this._runtime = runtime; this._connected = false; } /** * Request connection to the device. * If the web socket is not yet open, request when the socket promise resolves. */ requestDevice () { if (this._ws.readyState === 1) { // is this needed since it's only called on ws.onopen? // TODO: start a 'discover' timeout this.sendRemoteRequest('discover', this._deviceOptions) .catch(e => this._sendError(e)); // never reached? } // TODO: else? } /** * Try connecting to the input peripheral id, and then call the connect * callback if connection is successful. * @param {number} id - the id of the peripheral to connect to */ connectDevice (id) { this.sendRemoteRequest('connect', {peripheralId: id}) .then(() => { log.info('should have connected'); this._runtime.emit(this._runtime.constructor.PERIPHERAL_CONNECTED); this._connected = true; this._connectCallback(); }) .catch(e => { this._sendError(e); }); } /** * Close the websocket. */ disconnectSession () { this._ws.close(); this._connected = false; } /** * @return {bool} whether the peripheral is connected. */ getPeripheralIsConnected () { return this._connected; } sendMessage (options) { return this.sendRemoteRequest('send', options); } /** * Handle a received call from the socket. * @param {string} method - a received method label. * @param {object} params - a received list of parameters. * @return {object} - optional return value. */ didReceiveCall (method, params) { // TODO: Add peripheral 'undiscover' handling switch (method) { case 'didDiscoverPeripheral': this._availablePeripherals[params.peripheralId] = params; this._runtime.emit( this._runtime.constructor.PERIPHERAL_LIST_UPDATE, this._availablePeripherals ); // TODO: cancel a discover timeout if one is active break; case 'didReceiveMessage': this._messageCallback(params); // TODO: refine? break; default: return 'nah'; } } _sendError (e) { log.error(`BLESession error: ${e}`); this._runtime.emit(this._runtime.constructor.PERIPHERAL_ERROR); } } module.exports = BTSession;