mirror of
https://github.com/status-im/react-native.git
synced 2025-01-28 02:04:55 +00:00
cd9d6e34fd
Summary: This is a simple groundwork PR to allow options to be passed to the `WebSocket` constructor. It represents a minor change to an undocumented part of the API, moving `headers` to within `options`. This will be a BC for anyone manually specifying headers other than `origin` but a) that's not a common use case with WebSockets and b) it's not documented even in code and wouldn't currently pass a flow check. NB: The third argument to the WebSocket constructor isn't part of the W3C spec, so I think this is a good place for RN-specific named parameters, better than adding a fourth argument. `protocols` needs to stay where it is, in line with the spec. If this goes through I'd like to build on it by adding an additional connection option for SSL certificate pinning, as already supported by the underlying `okhttp` and `RCTSRWebSocket`. It could later be expanded for various other uses. Currently, there's no way for a `WebSocket` user to specify any connection options other than url, protocol and headers. The fact that `WebSocket` connects in its constructor means any options have to go in there. Connect to a websocket server using iOS and Android, observe the connection headers: 1. Without specifying `origin`, the default header should be set 2. Specifying it in the old way `new WebSocket(url, protocols, { origin: 'customorigin.com' })` 3. Specifying it in the new way `new WebSocket(url, protocols, { headers: { origin: 'customorigin.com' }})`. I've tested myself using the test app with iOS and Android. Closes https://github.com/facebook/react-native/pull/15334 Differential Revision: D5601675 Pulled By: javache fbshipit-source-id: 5959d03a3e1d269b2c6775f3e0cf071ff08617bf
268 lines
7.5 KiB
JavaScript
268 lines
7.5 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule WebSocket
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
const Blob = require('Blob');
|
|
const EventTarget = require('event-target-shim');
|
|
const NativeEventEmitter = require('NativeEventEmitter');
|
|
const NativeModules = require('NativeModules');
|
|
const Platform = require('Platform');
|
|
const WebSocketEvent = require('WebSocketEvent');
|
|
|
|
const base64 = require('base64-js');
|
|
const binaryToBase64 = require('binaryToBase64');
|
|
const invariant = require('fbjs/lib/invariant');
|
|
|
|
const {WebSocketModule} = NativeModules;
|
|
|
|
import type EventSubscription from 'EventSubscription';
|
|
|
|
type ArrayBufferView =
|
|
| Int8Array
|
|
| Uint8Array
|
|
| Uint8ClampedArray
|
|
| Int16Array
|
|
| Uint16Array
|
|
| Int32Array
|
|
| Uint32Array
|
|
| Float32Array
|
|
| Float64Array
|
|
| DataView
|
|
|
|
type BinaryType = 'blob' | 'arraybuffer'
|
|
|
|
const CONNECTING = 0;
|
|
const OPEN = 1;
|
|
const CLOSING = 2;
|
|
const CLOSED = 3;
|
|
|
|
const CLOSE_NORMAL = 1000;
|
|
|
|
const WEBSOCKET_EVENTS = [
|
|
'close',
|
|
'error',
|
|
'message',
|
|
'open',
|
|
];
|
|
|
|
let nextWebSocketId = 0;
|
|
|
|
/**
|
|
* Browser-compatible WebSockets implementation.
|
|
*
|
|
* See https://developer.mozilla.org/en-US/docs/Web/API/WebSocket
|
|
* See https://github.com/websockets/ws
|
|
*/
|
|
class WebSocket extends EventTarget(...WEBSOCKET_EVENTS) {
|
|
static CONNECTING = CONNECTING;
|
|
static OPEN = OPEN;
|
|
static CLOSING = CLOSING;
|
|
static CLOSED = CLOSED;
|
|
|
|
CONNECTING: number = CONNECTING;
|
|
OPEN: number = OPEN;
|
|
CLOSING: number = CLOSING;
|
|
CLOSED: number = CLOSED;
|
|
|
|
_socketId: number;
|
|
_eventEmitter: NativeEventEmitter;
|
|
_subscriptions: Array<EventSubscription>;
|
|
_binaryType: ?BinaryType;
|
|
|
|
onclose: ?Function;
|
|
onerror: ?Function;
|
|
onmessage: ?Function;
|
|
onopen: ?Function;
|
|
|
|
bufferedAmount: number;
|
|
extension: ?string;
|
|
protocol: ?string;
|
|
readyState: number = CONNECTING;
|
|
url: ?string;
|
|
|
|
// This module depends on the native `WebSocketModule` module. If you don't include it,
|
|
// `WebSocket.isAvailable` will return `false`, and WebSocket constructor will throw an error
|
|
static isAvailable: boolean = !!WebSocketModule;
|
|
|
|
constructor(url: string, protocols: ?string | ?Array<string>, options: ?{headers?: {origin?: string}}) {
|
|
super();
|
|
if (typeof protocols === 'string') {
|
|
protocols = [protocols];
|
|
}
|
|
|
|
const {headers = {}, ...unrecognized} = options || {};
|
|
|
|
// Preserve deprecated backwards compatibility for the 'origin' option
|
|
if (unrecognized && typeof unrecognized.origin === 'string') {
|
|
console.warn('Specifying `origin` as a WebSocket connection option is deprecated. Include it under `headers` instead.');
|
|
headers.origin = unrecognized.origin;
|
|
delete unrecognized.origin;
|
|
}
|
|
|
|
// Warn about and discard anything else
|
|
if (Object.keys(unrecognized).length > 0) {
|
|
console.warn('Unrecognized WebSocket connection option(s) `' + Object.keys(unrecognized).join('`, `') + '`. '
|
|
+ 'Did you mean to put these under `headers`?');
|
|
}
|
|
|
|
if (!Array.isArray(protocols)) {
|
|
protocols = null;
|
|
}
|
|
|
|
if (!WebSocket.isAvailable) {
|
|
throw new Error('Cannot initialize WebSocket module. ' +
|
|
'Native module WebSocketModule is missing.');
|
|
}
|
|
|
|
this._eventEmitter = new NativeEventEmitter(WebSocketModule);
|
|
this._socketId = nextWebSocketId++;
|
|
this._registerEvents();
|
|
WebSocketModule.connect(url, protocols, { headers }, this._socketId);
|
|
}
|
|
|
|
get binaryType(): ?BinaryType {
|
|
return this._binaryType;
|
|
}
|
|
|
|
set binaryType(binaryType: BinaryType): void {
|
|
if (binaryType !== 'blob' && binaryType !== 'arraybuffer') {
|
|
throw new Error('binaryType must be either \'blob\' or \'arraybuffer\'');
|
|
}
|
|
if (this._binaryType === 'blob' || binaryType === 'blob') {
|
|
const BlobModule = NativeModules.BlobModule;
|
|
invariant(BlobModule, 'Native module BlobModule is required for blob support');
|
|
if (BlobModule) {
|
|
if (binaryType === 'blob') {
|
|
BlobModule.enableBlobSupport(this._socketId);
|
|
} else {
|
|
BlobModule.disableBlobSupport(this._socketId);
|
|
}
|
|
}
|
|
}
|
|
this._binaryType = binaryType;
|
|
}
|
|
|
|
close(code?: number, reason?: string): void {
|
|
if (this.readyState === this.CLOSING ||
|
|
this.readyState === this.CLOSED) {
|
|
return;
|
|
}
|
|
|
|
this.readyState = this.CLOSING;
|
|
this._close(code, reason);
|
|
}
|
|
|
|
send(data: string | ArrayBuffer | ArrayBufferView | Blob): void {
|
|
if (this.readyState === this.CONNECTING) {
|
|
throw new Error('INVALID_STATE_ERR');
|
|
}
|
|
|
|
if (data instanceof Blob) {
|
|
const BlobModule = NativeModules.BlobModule;
|
|
invariant(BlobModule, 'Native module BlobModule is required for blob support');
|
|
BlobModule.sendBlob(data, this._socketId);
|
|
return;
|
|
}
|
|
|
|
if (typeof data === 'string') {
|
|
WebSocketModule.send(data, this._socketId);
|
|
return;
|
|
}
|
|
|
|
if (data instanceof ArrayBuffer || ArrayBuffer.isView(data)) {
|
|
WebSocketModule.sendBinary(binaryToBase64(data), this._socketId);
|
|
return;
|
|
}
|
|
|
|
throw new Error('Unsupported data type');
|
|
}
|
|
|
|
ping(): void {
|
|
if (this.readyState === this.CONNECTING) {
|
|
throw new Error('INVALID_STATE_ERR');
|
|
}
|
|
|
|
WebSocketModule.ping(this._socketId);
|
|
}
|
|
|
|
_close(code?: number, reason?: string): void {
|
|
if (Platform.OS === 'android') {
|
|
// See https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent
|
|
const statusCode = typeof code === 'number' ? code : CLOSE_NORMAL;
|
|
const closeReason = typeof reason === 'string' ? reason : '';
|
|
WebSocketModule.close(statusCode, closeReason, this._socketId);
|
|
} else {
|
|
WebSocketModule.close(this._socketId);
|
|
}
|
|
}
|
|
|
|
_unregisterEvents(): void {
|
|
this._subscriptions.forEach(e => e.remove());
|
|
this._subscriptions = [];
|
|
}
|
|
|
|
_registerEvents(): void {
|
|
this._subscriptions = [
|
|
this._eventEmitter.addListener('websocketMessage', ev => {
|
|
if (ev.id !== this._socketId) {
|
|
return;
|
|
}
|
|
let data = ev.data;
|
|
switch (ev.type) {
|
|
case 'binary':
|
|
data = base64.toByteArray(ev.data).buffer;
|
|
break;
|
|
case 'blob':
|
|
data = Blob.create(ev.data);
|
|
break;
|
|
}
|
|
this.dispatchEvent(new WebSocketEvent('message', { data }));
|
|
}),
|
|
this._eventEmitter.addListener('websocketOpen', ev => {
|
|
if (ev.id !== this._socketId) {
|
|
return;
|
|
}
|
|
this.readyState = this.OPEN;
|
|
this.dispatchEvent(new WebSocketEvent('open'));
|
|
}),
|
|
this._eventEmitter.addListener('websocketClosed', ev => {
|
|
if (ev.id !== this._socketId) {
|
|
return;
|
|
}
|
|
this.readyState = this.CLOSED;
|
|
this.dispatchEvent(new WebSocketEvent('close', {
|
|
code: ev.code,
|
|
reason: ev.reason,
|
|
}));
|
|
this._unregisterEvents();
|
|
this.close();
|
|
}),
|
|
this._eventEmitter.addListener('websocketFailed', ev => {
|
|
if (ev.id !== this._socketId) {
|
|
return;
|
|
}
|
|
this.readyState = this.CLOSED;
|
|
this.dispatchEvent(new WebSocketEvent('error', {
|
|
message: ev.message,
|
|
}));
|
|
this.dispatchEvent(new WebSocketEvent('close', {
|
|
message: ev.message,
|
|
}));
|
|
this._unregisterEvents();
|
|
this.close();
|
|
})
|
|
];
|
|
}
|
|
}
|
|
|
|
module.exports = WebSocket;
|