Closed kovkev closed 1 month ago
Note, if I use https://github.com/JonnyBGod/react-native-eventsource , I am able to have .onmessage
calls when an Event is sent from the server, but I am unable to see what was sent:
[Tue May 05 2020 19:31:21.748] LOG {"data": undefined, "type": "message"}
Also, my ec2 instance runs nginx and routes the requests to Express. If that makes any difference. You can see the request headers above^ . The response headers should be the same I believe?
Do you still have a problem with this?
Yes, it's still not working on Android
Note, if I use https://github.com/JonnyBGod/react-native-eventsource , I am able to have
.onmessage
calls when an Event is sent from the server, but I am unable to see what was sent:[Tue May 05 2020 19:31:21.748] LOG {"data": undefined, "type": "message"}
Im just facing same problem - "react-native-eventsource": "0.0.2", I open node_modules/react-native-eventsource/EventSource.js Change
var event = new EventSourceEvent(ev.type, {
data:ev.message
});
To:
var event = new EventSourceEvent(ev.type, {
data:ev.message || ev.data
});
then it worked like a charm Hope this helps
Note, if I use https://github.com/JonnyBGod/react-native-eventsource , I am able to have
.onmessage
calls when an Event is sent from the server, but I am unable to see what was sent:[Tue May 05 2020 19:31:21.748] LOG {"data": undefined, "type": "message"}
Im just facing same problem - "react-native-eventsource": "0.0.2", I open node_modules/react-native-eventsource/EventSource.js Change
var event = new EventSourceEvent(ev.type, { data:ev.message });
To:
var event = new EventSourceEvent(ev.type, { data:ev.message || ev.data });
then it worked like a charm Hope this helps
How you've managed to run this package? Did you link it manually? I face the same problem. ES events works just fine on ios and on Android simulator. But it does not work on real devices
unfortunately i had to write a native module for android based on OkSse and register it on react-native to get SSE working
...
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.Headers;
import okhttp3.FormBody;
import com.here.oksse.OkSse;
import com.here.oksse.ServerSentEvent;
import java.util.Map;
import java.util.HashMap;
import com.facebook.react.modules.core.DeviceEventManagerModule;
// barebones Android EventSource java implementation
public class EventSource extends ReactContextBaseJavaModule {
private static ReactApplicationContext reactContext;
private OkSse okSse;
private HashMap<String, ServerSentEvent> sseManager;
private int sseCount;
public EventSource(ReactApplicationContext context) {
super(context);
reactContext = context;
this.okSse = new OkSse();
this.sseManager = new HashMap(100);
}
public String getName() {
return "EventSource";
}
// returns an SSE id which you need to refer back to for closing the connection
// assumes that you're making a GET request, but you can also modify this to accept POST/PATCH requests as well
@ReactMethod
public void initRequest(
String path,
ReadableMap headersMap,
Promise promise
) {
Request.Builder requestBuilder = new Request.Builder().url(path);
HashMap<String, Object> headersHashMap = headersMap.toHashMap();
for (Map.Entry<String, Object> entry : headersHashMap.entrySet()) {
String headerKey = entry.getKey();
String headerValue = (String) entry.getValue(); // assumes that the key-value JSON passed into headers are <String, String>
requestBuilder.addHeader(headerKey, headerValue);
}
Request request = requestBuilder.build();
String sseRefID = String.valueOf(this.sseCount) + " - " + path;
ServerSentEvent sse = this.okSse.newServerSentEvent(request, new ServerSentEvent.Listener() {
@Override
public void onOpen(ServerSentEvent sse, Response response) {
}
@Override
public void onMessage(ServerSentEvent sse, String id, String event, String message) {
// When a message is received
WritableMap map = new WritableNativeMap();
map.putString("id", id);
map.putString("event", event);
map.putString("message", message);
reactContext
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class)
.emit(sseRefID, map);
}
@WorkerThread
@Override
public void onComment(ServerSentEvent sse, String comment) {
WritableMap map = new WritableNativeMap();
map.putString("comment", comment);
reactContext
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class)
.emit(sseRefID, map); // this identifies the HTTP connection and respective SSEs we want to be listening to on JS
}
@WorkerThread
@Override
public Request onPreRetry(ServerSentEvent sse, Request sseRequest) {
return request;
}
@WorkerThread
@Override
public boolean onRetryTime(ServerSentEvent sse, long milliseconds) {
return true; // True to use the new retry time received by SSE
}
@WorkerThread
@Override
public boolean onRetryError(ServerSentEvent sse, Throwable throwable, Response response) {
return true; // True to retry, false otherwise
}
@WorkerThread
@Override
public void onClosed(ServerSentEvent sse) {
// Channel closed
sse.close();
}
});
this.sseManager.put(sseRefID, sse);
this.sseCount++;
promise.resolve(sseRefID);
}
@ReactMethod
public void close(String sseRefID, Promise promise) {
try {
this.sseManager.get(sseRefID).close();
this.sseManager.remove(sseRefID);
promise.resolve(true);
} catch (Exception e) {
System.out.println("failed to close connection");
promise.resolve(false);
}
}
}
on the react-native side
// NativeAndroidEventSource.js
import { NativeEventEmitter, NativeModules, Platform } from 'react-native';
export const eventEmitter = Platform.OS === 'ios'
? null
: new NativeEventEmitter(NativeModules.EventSource);
export default Platform.OS === 'ios' ? null : NativeModules.EventSource;
...
// other module
import AndroidEventSource from './NativeAndroidEventSource.js';
...
// inside an async function
const url = 'http:localhost:3000/sse';
const headers = { 'Content-Type': 'text/event-stream' };
const sseRefID = await AndroidEventSource.initRequest(
url,
headers
);
AndroidEventSource.addListener(sseRefID, { message } => {
console.log(message);
});
for ios
you can use the existing XHR-based EventSource polyfills
I am having the same issue but this is what I found. I am using react-native-event-source and following is a comment in its code in EventSource.js file:
https://github.com/jordanbyron/react-native-event-source/blob/master/EventSource.js#L47
And just as I killed the server all the events came trickling down, as killing the server kills the connections. So I'm guessing that onreadystatechange on android does not trigger unless the connection is closed from the server. I am not sure why and what but killing the server or closing the connections from the server side triggers the event listeners.
Unless Android changes the way it handles stuff this bug doesn't seem to be going away. For now there are some really good native libraries. I am using https://github.com/gpsgate/react-native-eventsource and it works like a dream. However its a non pure js solution so really relies upon updated code.
hey guys, I found these interesting library, to try to solve the android problem with pure js solution:
I understand the big problem of polyfill is XHR of RN doesn't work the same as html5
🤔 so it is some thing should change in the core of RN?
Hello,
I encountered the same problem, I tried 4 libraries and always the same observation: the mesages are displayed on iOS, but not on Android.
Any ideas ? thank you in advance
https://github.com/gpsgate/react-native-eventsource I try this on android and no response, I'm in rn 0.62
Try to disable Flipper network interceptor. Go to android/app/src/debug/java//ReactNativeFlipper.java and comment next lines of code:
// try to comment this code
NetworkingModule.setCustomClientBuilder(
new NetworkingModule.CustomClientBuilder() {
@Override
public void apply(OkHttpClient.Builder builder) {
builder.addNetworkInterceptor(new FlipperOkhttpInterceptor(networkFlipperPlugin));
}
}
);
i will tell to flipper team if this is a bug, or if they have some solution to solve it
Most of the sse React-Native libraries are using xhr
, and it doesn't seem to be working properly on Android.
I've struggled a lot lately with other unrelated issues and they all have been caused by xhr
(though it was working well on iOS) , after swapping it to node-fetch
everything started to work fine
Most of the sse React-Native libraries are using
xhr
, and it doesn't seem to be working properly on Android. I've struggled a lot lately with other unrelated issues and they all have been caused byxhr
(though it was working well on iOS) , after swapping it tonode-fetch
everything started to work fine
Hi can you share the node-fetch implementation of event source. Facing all the above issues for react-native-sse library in Android
Try to disable Flipper network interceptor. Go to android/app/src/debug/java//ReactNativeFlipper.java and comment next lines of code:
// try to comment this code NetworkingModule.setCustomClientBuilder( new NetworkingModule.CustomClientBuilder() { @Override public void apply(OkHttpClient.Builder builder) { builder.addNetworkInterceptor(new FlipperOkhttpInterceptor(networkFlipperPlugin)); } } );
The flipper network interceptor was in fact responsible for react-native-sse not working on Android. The above did not work for me, but the following did. In this code, I have added a boolean variable shouldAddInterceptor that you can set based on your conditions. If shouldAddInterceptor is true, the interceptor will be added. If false, it won't be added. You will need to replace shouldAddInterceptor with your condition for whether or not to add the Flipper network interceptor:
` public class ReactNativeFlipper { public static void initializeFlipper(Context context, ReactInstanceManager reactInstanceManager) { if (FlipperUtils.shouldEnableFlipper(context)) { final FlipperClient client = AndroidFlipperClient.getInstance(context);
client.addPlugin(new InspectorFlipperPlugin(context, DescriptorMapping.withDefaults()));
client.addPlugin(new DatabasesFlipperPlugin(context));
client.addPlugin(new SharedPreferencesFlipperPlugin(context));
client.addPlugin(CrashReporterPlugin.getInstance());
NetworkFlipperPlugin networkFlipperPlugin = new NetworkFlipperPlugin();
boolean shouldAddInterceptor = false; // set this based on your conditions
if (shouldAddInterceptor) {
NetworkingModule.setCustomClientBuilder(
new NetworkingModule.CustomClientBuilder() {
@Override
public void apply(OkHttpClient.Builder builder) {
builder.addNetworkInterceptor(new FlipperOkhttpInterceptor(networkFlipperPlugin));
}
});
}
client.addPlugin(networkFlipperPlugin);
client.start();
// Fresco Plugin needs to ensure that ImagePipelineFactory is initialized
// Hence we run if after all native modules have been initialized
ReactContext reactContext = reactInstanceManager.getCurrentReactContext();
if (reactContext == null) {
reactInstanceManager.addReactInstanceEventListener(
new ReactInstanceEventListener() {
@Override
public void onReactContextInitialized(ReactContext reactContext) {
reactInstanceManager.removeReactInstanceEventListener(this);
reactContext.runOnNativeModulesQueueThread(
new Runnable() {
@Override
public void run() {
client.addPlugin(new FrescoFlipperPlugin());
}
});
}
});
} else {
client.addPlugin(new FrescoFlipperPlugin());
}
}
} } `
You can just copy/paste this over existing code in the ReactNativeFlipper.java file. Just leave the imports how they are at the top of the file. After this, Android received server sent events as intended with only react-native-sse and react-native-url-polyfill packages. iOS worked fine without any changes.
Hey guys. I went ahead and implemented react-native-oksse. A wrapper around the java extension library oksse. Check it out.
In case you are using expo SDK 50, you could try with the plugins i've developed to patch the Android SSE issues while in debug variant (both are needed):
Edit: note to the react-native team, perhaps the flipper issue is caused by the same type of problem as described there: https://github.com/expo/expo/issues/27526 ?
This issue is stale because it has been open 180 days with no activity. Remove stale label or comment or this will be closed in 7 days.
This issue was closed because it has been stalled for 7 days with no activity.
Please provide all the information requested. Issues that do not follow this format are likely to stall.
Description
I am trying to use
rn-eventsource
andreact-native-event-source
and they always get 200 status when sending a request to the server. However! I am able to receive Events on:but not from:
I can get webpages with the browser provided
EventSource
to work on Chrome on ios and on Android.It seems like this is an issue with React-Native on Android.
React Native version:
Steps To Reproduce
Provide a detailed list of steps that reproduce the issue.
rn-eventsource
orreact-native-event-source
EventSource("https://..../sse_listen")
.onmessage
,.onopen
and.onerror
which doconsole.log
.onmessage
.onopen
and.onerror
Expected Results
Events appear on Android
More details: I'm pasting the comments I added to this thread : https://github.com/react-native-community/discussions-and-proposals/issues/99 ::