Browser hang on disconnect session

Hello,
we are having an issue with browsers hanging/crashing when users disconnect from a session. It seems to be narrowed down to when someone has shared their screen during the session. Using replaceTrack() with video track from:
.getUserMedia({ videoSource: "screen", audioSource: false}).then(stream => { const track = stream.getVideoTracks()[0]; publisher.replaceTrack(track) })
to toggle the screensharing.

Any ideas what could cause this behavior? The browser freezes up and needs to be forced to shut down. Some experience a black screen (Chrome).

Can you please describe how to reproduce the issue (all details are welcomed like browser versions, step by step description, etc…).

If you can share with us a sample app to reproduce the issue it would be awesome!!

Hey @micael.gallego, thanks for replying so fast. I will investigate further to see if the issue can be reproduced consistently, and try to get browser versions for the cases. I know the majority of the users are on latest Chrome on Win10. As a start, here is my hook implementation for openvidu-browser. I have tried to strip it down to the essentials and have not tested this edited version, but maybe someone will spot a problem in the code. toggleCamera() and toggleShareScreen() which calls setVideoTrack() is my current suspect for the issue. I have also been looking into whether the session cleans up properly and recently added the stopEvents() function with session.off() for the events. Not sure if this is necessary.

import { useState, useRef } from 'react';
    import axios from 'axios';
    import {
      OpenVidu,
      Connection,
      Publisher,
      Session,
      StreamManager,
      Device,
      PublisherProperties,
      StreamEvent,
      PublisherSpeakingEvent,
      ConnectionEvent
    } from 'openvidu-browser';

    const OPENVIDU_SERVER_URL = process.env.REACT_APP_MEDIASERVER_URL;

    // Server side
    const createSession = async (sessionId: string) => {
      return new Promise((resolve, reject) => {
    axios
      .get(
        OPENVIDU_SERVER_URL +
          '/api/MediaServer/Join/' +
          sessionId +
          '&autoRecord=false',
        {
          headers: {
            'Content-Type': 'application/json'
          }
        }
      )
      .then(response => {
        resolve(response.data);
      })
      .catch(error => {
        reject(error);
      });
      });
    };

    const useOpenVidu = () => {
      const openVidu = useRef<OpenVidu | null>(null);
      const session = useRef<Session | null>(null);
      const videoDevices = useRef<Device[]>([]);
      const videoDeviceIndex = useRef<number>(0);
      const [shareScreen, setShareScreen] = useState<boolean>(false);
      const [subscribers, setSubscribers] = useState<StreamManager[]>([]);
      const [mainStreamManager, setMainStreamManager] = useState<
    StreamManager | undefined
      >(undefined);
      const [publisher, setPublisher] = useState<Publisher | undefined>(undefined);
      const [connections, setConnections] = useState<Connection[]>([]);
      const [executing, setExecuting] = useState<boolean>(false);

      const setupEvents = () => {
    if (session.current) {
      session.current.on('connectionCreated', (event: any) => {
        const e = event as ConnectionEvent;
        setConnections((conns: Connection[]) => [...conns, e.connection]);
      });
      session.current.on('connectionDestroyed', (event: any) => {
        const e = event as ConnectionEvent;
        const id = e.connection.connectionId;
        setConnections(conns =>
          conns.filter(c => {
            return c.connectionId !== id;
          })
        );
      });
      session.current.on('streamCreated', (event: any) => {
        const e = event as StreamEvent;
        if (session.current) {
          const subscriber = session.current.subscribe(e.stream, 'undefined');
          setSubscribers(sub => [...sub, subscriber]);
        }
      });
      session.current.on('streamDestroyed', (event: any) => {
        const e = event as StreamEvent;
        setSubscribers(subs => subs.filter(s => s !== e.stream.streamManager));
      });
      session.current.on('sessionDisconnected', () => {
        stopEvents();
        cleanup();
      });
      session.current.on('streamPropertyChanged', () => {
        setSubscribers(subs => subs.slice());
      });
    } else {
      console.error('setupEvents() failed: no session reference');
    }
      };
      const stopEvents = () => {
    if (session.current) {
      session.current.off('connectionCreated');
      session.current.off('connectionDestroyed');
      session.current.off('streamCreated');
      session.current.off('streamDestroyed');
      session.current.off('sessionDisconnected');
      session.current.off('streamPropertyChanged');
    } else {
      console.error('stopEvents() failed: no session reference');
    }
      };

      const joinSession = async (userName: string, sessionId: string) => {
    openVidu.current = new OpenVidu();
    const devices = await openVidu.current.getDevices();

    const videoInputs = devices.filter(d => d.kind === 'videoinput');
    videoDevices.current = videoInputs;

    const newSession = openVidu.current!.initSession();
    session.current = newSession;
    setupEvents();

    connect(userName, sessionId).catch(() => {
      cleanup();
    });
      };

      const cleanup = () => {
    openVidu.current = null;
    session.current = null;
    videoDevices.current = [];
    setSubscribers([]);
    setMainStreamManager(undefined);
    setPublisher(undefined);
    setConnections([]);
    videoDevices.current = [];
    videoDeviceIndex.current = 0;
    setShareScreen(false);
    setExecuting(false);
      };

      const leaveSession = () => {
    if (session.current) {
      session.current.disconnect();
    }
    cleanup();
      };

      const connect = async (userName: string, sessionId: string) => {
    return new Promise((resolve, reject) => {
      createSession(sessionId)
        .then((response: any) => {
          if (session.current) {
            session.current
              .connect(response.token, {
                clientData: userName
              })
              .then(() => {
                publish()
                  .then(() => {
                    resolve({});
                  })
                  .catch(error => {
                    alert('Could not publish to session.\n' + error);
                    reject(error);
                  });
              });
          }
        })
        .catch(error => {
          alert('Could not connect to server.\n' + error);
          reject(error);
        });
    });
      };

      const publish = async () => {
    return new Promise((resolve, reject) => {
      let newPublisher: Publisher | null = null;
      const videoDevice =
        videoDevices.current.length > 0
          ? videoDevices.current[videoDeviceIndex.current].deviceId
          : false;
      if (openVidu.current) {
        newPublisher = openVidu.current.initPublisher('undefined', {
          videoSource: videoDevice,
          audioSource: undefined,
          publishAudio: true,
          publishVideo: true,
          resolution: '320x240',
          frameRate: 15,
          insertMode: 'APPEND',
          mirror: false
        });
      }

      if (session.current && newPublisher) {
        session.current
          .publish(newPublisher)
          .then(() => {
            setPublisher(newPublisher!);
            resolve(newPublisher);
          })
          .catch(error => {
            reject(error);
          });
      } else {
        reject('Session or publisher does not exist.');
      }
    });
      };

      const setVideoTrack = async (constraints: PublisherProperties) => {
    return new Promise((resolve, reject) => {
      if (openVidu.current && session.current && publisher !== undefined) {
        openVidu.current
          .getUserMedia({ ...constraints, audioSource: false })
          .then(stream => {
            const track = stream.getVideoTracks()[0];
            publisher.replaceTrack(track);
            resolve(track);
          })
          .catch(error => reject(error));
      }
    });
      };

      const toggleShareScreen = async () => {
    if (shareScreen && executing === false) {
      setExecuting(true);
      setVideoTrack({
        videoSource: videoDevices.current[videoDeviceIndex.current].deviceId
      }).then(() => setShareScreen(false));
    } else {
      setVideoTrack({ videoSource: 'screen' }).then(() => setShareScreen(true));
    }
    setExecuting(false);
      };

      const toggleCamera = () => {
    if (
      videoDevices.current.length > 1 &&
      publisher !== undefined &&
      executing === false
    ) {
      setExecuting(true);
      setShareScreen(false);
      const nextIndex =
        (videoDeviceIndex.current + 1) % videoDevices.current.length;

      setVideoTrack({
        videoSource: videoDevices.current[nextIndex].deviceId
      }).then(() => {
        videoDeviceIndex.current = nextIndex;
        setExecuting(false);
      });
    }
      };

      const setMainVideoStream = (stream: StreamManager | undefined) => {
    if (mainStreamManager !== stream) {
      setMainStreamManager(() => stream);
    }
      };

      return {
    joinSession,
    leaveSession,
    setMainVideoStream,
    setMainStreamManager,
    toggleShareScreen,
    shareScreen,
    toggleCamera,
    connections,
    mainStreamManager,
    publisher,
    subscribers
      };
    };

    export default useOpenVidu;

Please share with us a github repo with a working code and steps to reproduce the issue so we can try ourselves and debug it.

If the code is a modification to an existing demo/turorial much better, show we have to look at the modified code.

Best regards

Here is basically the same toggle screen sharing implementation that I have in my project. It’s a modified insecure react sample.
We had some clients black out this morning.
Edge Version 80.0.361.69
Chrome Version 80.0.3987.149
It happens after someone has shared screen and random events like a user leaves session, closes a tab, calls publishAudio()/publishVideo(). The browser window usually goes black, but if a user is still in session, audio comes through.
Could it be that there is a GPU spike that leaves the browser in a bad state or a memory issue, or if there may be conflict with the browsers own Stop sharing dialog.
chrome

Just out of curiosity, are Edge users able to connect to your OpenVidu Sessions right now? (of course using the brand new Edge browsers).

Regards

1 Like

apparently yes :slight_smile: i know opera browser is also used

It is hard to say. Please can you test with other WebRTC services like Skype web or whereby.com to see if the behaviour is the same. It looks to an entire problem of the browser… but if other services don’t have this problem, maybe we can investigate other ways to achieve the same behavior without crashing the browser.

It is the first time I see this issue… maybe is a Windows 10 only problem (we use linux).