mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
4b9c99dff0
Summary: Prior to this patch the websocket protocol was not being set when a connection was opened, which could cause client libraries and apps to not work properly. According to the [whatwg] spec the protocol must be set once the connection is estabilished. [whatwg]: https://html.spec.whatwg.org/multipage/web-sockets.html#feedback-from-the-protocol ## Changelog [Javascript] [Fixed] - Properly set the this.protocol on WebSocket open [Android] [Fixed] - Send the server chosen protocol to the WebSocket object [iOS] [Fixed] - Send the server chosen protocol to the WebSocket object Pull Request resolved: https://github.com/facebook/react-native/pull/25273 Test Plan: In order to reproduce the issue you **need to install wampy@6.2.1**. Since **wampy@6.2.2** and newer contains a workaround for this react-native bug. https://www.npmjs.com/package/wampy ```javascript /** * Sample React Native App * https://github.com/facebook/react-native * * format * flow */ import React, { Component } from 'react'; import { Platform, StyleSheet, Text, View } from 'react-native'; import Wampy from "wampy"; const instructions = Platform.select({ ios: 'Press Cmd+R to reload,\n' + 'Cmd+D or shake for dev menu', android: 'Double tap R on your keyboard to reload,\n' + 'Shake or press menu button for dev menu', }); type Props = {}; export default class App extends Component<Props> { state = {conState: 'Initializing...'}; componentDidMount() { const url = "wss://demo.crossbar.io/ws"; const ws = new Wampy(url, { realm: "crossbardemo", ws: WebSocket, debug: true, onConnect: () => { console.log("WAMP onConnect"); this.setState({conState: 'Connected'}); }, onClose: () => { console.log("WAMP onClose"); this.setState({conState: 'Connection closed'}); }, onError: () => { console.log("WAMP onError"); this.setState({conState: 'Connection Error'}); } }); } render() { return ( <View style={styles.container}> <Text style={styles.message}>{this.state.conState}</Text> </View> ); } } const styles = StyleSheet.create({ container: { flex: 1, justifyContent: 'center', alignItems: 'center', backgroundColor: '#F5FCFF', }, message: { fontSize: 20, color: 'black' }, }); ``` Using the code above one must see the message **WAMP onConnect** on Console and **Connected** in the middle of the screen Closes https://github.com/facebook/react-native/issues/24796 Differential Revision: D15938870 Pulled By: cpojer fbshipit-source-id: 10a0a9b40c2a69e484ead37149abc2b1158a4ffc
280 lines
7.8 KiB
JavaScript
280 lines
7.8 KiB
JavaScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @format
|
|
* @flow
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
const Blob = require('../Blob/Blob');
|
|
const EventTarget = require('event-target-shim');
|
|
const NativeEventEmitter = require('../EventEmitter/NativeEventEmitter');
|
|
const BlobManager = require('../Blob/BlobManager');
|
|
const Platform = require('../Utilities/Platform');
|
|
const WebSocketEvent = require('./WebSocketEvent');
|
|
|
|
const base64 = require('base64-js');
|
|
const binaryToBase64 = require('../Utilities/binaryToBase64');
|
|
const invariant = require('invariant');
|
|
|
|
import NativeWebSocketModule from './NativeWebSocketModule';
|
|
|
|
import type EventSubscription from '../vendor/emitter/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;
|
|
|
|
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
|
|
/* $FlowFixMe(>=0.68.0 site=react_native_fb) This comment suppresses an
|
|
* error found when Flow v0.68 was deployed. To see the error delete this
|
|
* comment and run Flow. */
|
|
if (unrecognized && typeof unrecognized.origin === 'string') {
|
|
console.warn(
|
|
'Specifying `origin` as a WebSocket connection option is deprecated. Include it under `headers` instead.',
|
|
);
|
|
/* $FlowFixMe(>=0.54.0 site=react_native_fb,react_native_oss) This
|
|
* comment suppresses an error found when Flow v0.54 was deployed. To see
|
|
* the error delete this comment and run Flow. */
|
|
headers.origin = unrecognized.origin;
|
|
/* $FlowFixMe(>=0.54.0 site=react_native_fb,react_native_oss) This
|
|
* comment suppresses an error found when Flow v0.54 was deployed. To see
|
|
* the error delete this comment and run Flow. */
|
|
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;
|
|
}
|
|
|
|
this._eventEmitter = new NativeEventEmitter(NativeWebSocketModule);
|
|
this._socketId = nextWebSocketId++;
|
|
this._registerEvents();
|
|
NativeWebSocketModule.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') {
|
|
invariant(
|
|
BlobManager.isAvailable,
|
|
'Native module BlobModule is required for blob support',
|
|
);
|
|
if (binaryType === 'blob') {
|
|
BlobManager.addWebSocketHandler(this._socketId);
|
|
} else {
|
|
BlobManager.removeWebSocketHandler(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) {
|
|
invariant(
|
|
BlobManager.isAvailable,
|
|
'Native module BlobModule is required for blob support',
|
|
);
|
|
BlobManager.sendOverSocket(data, this._socketId);
|
|
return;
|
|
}
|
|
|
|
if (typeof data === 'string') {
|
|
NativeWebSocketModule.send(data, this._socketId);
|
|
return;
|
|
}
|
|
|
|
if (data instanceof ArrayBuffer || ArrayBuffer.isView(data)) {
|
|
NativeWebSocketModule.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');
|
|
}
|
|
|
|
NativeWebSocketModule.ping(this._socketId);
|
|
}
|
|
|
|
_close(code?: number, reason?: string): void {
|
|
// 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 : '';
|
|
NativeWebSocketModule.close(statusCode, closeReason, this._socketId);
|
|
|
|
if (BlobManager.isAvailable && this._binaryType === 'blob') {
|
|
BlobManager.removeWebSocketHandler(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 = BlobManager.createFromOptions(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.protocol = ev.protocol;
|
|
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;
|