diff --git a/.github/workflows/auto_deploy.yml b/.github/workflows/auto_deploy.yml deleted file mode 100644 index b7318f99c..000000000 --- a/.github/workflows/auto_deploy.yml +++ /dev/null @@ -1,157 +0,0 @@ -# This workflow will build a Java project with Maven, and cache/restore any dependencies to improve the workflow execution time -# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-java-with-maven - -# This workflow uses actions that are not certified by GitHub. -# They are provided by a third-party and are governed by -# separate terms of service, privacy policy, and support -# documentation. - -name: Java CI with Maven - -on: -# [push] - workflow_dispatch: - schedule: - - cron: '0 0 * * *' - -jobs: - build: - - runs-on: ubuntu-latest - - steps: - - uses: actions/checkout@v3 - - - name: Get versions data from repo - run: wget -O maven-metadata.xml https://oss.sonatype.org/service/local/repositories/snapshots/content/io/antmedia/ant-media-server/maven-metadata.xml - - name: Install jq - run: sudo apt-get install jq -y - - name: Download war File - run: | - export LATEST_SNAPSHOT=$(grep -oP '(?<=)[^<]+' maven-metadata.xml| tail -1) - echo $LATEST_SNAPSHOT - echo "LATEST_SNAPSHOT=$LATEST_SNAPSHOT" >> $GITHUB_ENV - wget -O ConferenceCall.war "https://oss.sonatype.org/service/local/artifact/maven/redirect?r=snapshots&g=io.antmedia.webrtc&a=ConferenceCall&v=${LATEST_SNAPSHOT}&e=war" - ls -al - if [ ! -f ConferenceCall.war ]; then - echo "War file not found." - exit 1 - fi - - - name: MD5 file checksum - run: | - echo $LATEST_SNAPSHOT - wget -O ConferenceCall.md5 "https://oss.sonatype.org/service/local/artifact/maven/redirect?r=snapshots&g=io.antmedia.webrtc&a=ConferenceCall&v=${LATEST_SNAPSHOT}&e=war.md5" - local_file_md5="$(md5sum ConferenceCall.war | awk '{print $1}')" - remote_file_md5="$(cat ConferenceCall.md5)" - - if [ "$remote_file_md5" != "$local_file_md5" ]; then - echo "MD5 checksum mismatch!" - exit 1 - else - echo "MD5 checksum matches." - fi - - - name: Setup kubectl - uses: azure/setup-kubectl@v3 - with: - version: 'latest' - - - name: Create kubeconfig file - run: | - mkdir $HOME/.kube - echo "${{ secrets.KUBE_CONFIG_DATA }}" > $HOME/.kube/config - - - name: Change Replicate Set - run: | - echo "current_replica=$(kubectl get deployment ant-media-server-origin -n antmedia -o jsonpath='{.spec.replicas}')" >> $GITHUB_ENV - kubectl scale deployment ant-media-server-origin --replicas=1 -n antmedia - - - name: Login to server - run: | - response=$(curl -X POST -H "Accept: Application/json" -H "Content-Type: application/json" ${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/users/authenticate -d '{"email":"${{ secrets.USER_NAME }}","password":"${{ secrets.PASSWORD }}"}' -c cookie.txt) - success=$(echo $response | jq -r '.success') - if [ "$success" != "true" ]; then - echo "Login failed" - exit 1 - fi - - - name: Check if Conference App Exists - run: | - response=$(curl -s -H "Accept: Application/json" -H "Content-Type: application/json" "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications" -b cookie.txt) - echo $response | jq . - app_exists=$(echo $response | jq -r '.applications | index("Conference")') - if [ "$app_exists" != "null" ]; then - echo "App exists, proceeding to delete it." - response=$(curl -s -X DELETE -H "Accept: Application/json" -H "Content-Type: application/json" "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications/Conference" -b cookie.txt) - sleep 20 - success=$(echo $response | jq -r '.success') - if [ "$success" != "true" ]; then - echo "Conference app deletion failed" - exit 1 - fi - else - echo "App does not exist, proceeding to create a new one." - fi - - - name: Create New Conference App - id: create_app - run: | - export WAR_FILE_NAME="ConferenceCall.war" - response=$(curl -v -X PUT -H "Accept: Application/json" -H "Content-Type: multipart/form-data" -F "file=@./$WAR_FILE_NAME" "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications/Conference" -b cookie.txt) - success=$(echo $response | jq -r '.success') - echo $response - if [ "$success" != "true" ]; then - echo "Conference app creation is failed" - exit 1 - fi - continue-on-error: true - - - name: Rollout restart deployment if Create Conference App failed - if: steps.create_app.outcome == 'failure' - run: | - echo "Create New App failed, performing kubectl rollout restart..." - kubectl rollout restart deployment ant-media-server-origin -n antmedia - - - name: Retry Create New Conference App - if: steps.create_app.outcome == 'failure' - run: | - export WAR_FILE_NAME="ConferenceCall.war" - response=$(curl -v -X PUT -H "Accept: Application/json" -H "Content-Type: multipart/form-data" -F "file=@./$WAR_FILE_NAME" "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications/Conference" -b cookie.txt) - success=$(echo $response | jq -r '.success') - echo $response - if [ "$success" != "true" ]; then - echo "Conference app creation failed again" - exit 1 - fi - - - name: Change Settings - run: | - curl "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications/settings/Conference" -b cookie.txt -o settings.json - jq '.stunServerURI = "turn:${{ secrets.PRODUCTION_TURN_URL }}" | .turnServerUsername = "${{ secrets.PRODUCTION_TURN_USERNAME }}" | .turnServerCredential = "${{ secrets.PRODUCTION_TURN_PASSWORD }}"' settings.json > updated_settings.json - response=$(curl -X POST -H "Accept: Application/json" -H "Content-Type: application/json" -d @updated_settings.json -b cookie.txt "${{ secrets.PRODUCTION_SERVER_URL }}/rest/v2/applications/settings/Conference") - success=$(echo $response | jq -r '.success') - if [ "$success" != "true" ]; then - echo "Importing setting is failed." - exit 1 - fi - - - name: Revert Replicate Set - run: | - echo $current_replica - kubectl scale deployment ant-media-server-origin --replicas=$current_replica -n antmedia - - restart_when_failed: - name: Restarts the scheduled run when it failed - runs-on: ubuntu-latest - if: github.event_name == 'schedule' && failure() - needs: build - steps: - - name: Retry the workflow - run: | - curl -i \ - -X POST \ - -H "Accept: application/vnd.github.v3+json" \ - -H "Authorization: token ${{ secrets.GIT_TOKEN }}" \ - https://api.github.com/repos/ant-media/conference-call-application/actions/workflows/auto_deploy.yml/dispatches \ - -d '{"ref": "${{ github.ref }}" }' diff --git a/.github/workflows/build-and-deploy.yml b/.github/workflows/build-and-deploy.yml index 0b4f07048..59a002dba 100644 --- a/.github/workflows/build-and-deploy.yml +++ b/.github/workflows/build-and-deploy.yml @@ -191,17 +191,6 @@ jobs: - name: Checkout repository uses: actions/checkout@v3 - - name: Set up JDK 17 - uses: actions/setup-java@v3 - with: - java-version: 17 - distribution: 'temurin' - server-id: ossrh # Value of the distributionManagement/repository/id field of the pom.xml - server-username: MAVEN_USERNAME # env variable for username in deploy - server-password: MAVEN_PASSWORD # env variable for token in deploy - gpg-private-key: ${{ secrets.MVN_GPG_KEY }} # Value of the GPG private key to import - gpg-passphrase: MAVEN_GPG_PASSPHRASE # env variable for GPG private key passphrase - - name: Restore Cached Test Tool uses: actions/cache@v3 with: @@ -218,6 +207,18 @@ jobs: unzip webrtc-load-test-tool-*.zip mkdir ~/test mv webrtc-load-test ~/test + ls -al ~/test + + - name: Set up JDK 17 + uses: actions/setup-java@v3 + with: + java-version: 17 + distribution: 'temurin' + server-id: ossrh # Value of the distributionManagement/repository/id field of the pom.xml + server-username: MAVEN_USERNAME # env variable for username in deploy + server-password: MAVEN_PASSWORD # env variable for token in deploy + gpg-private-key: ${{ secrets.MVN_GPG_KEY }} # Value of the GPG private key to import + gpg-passphrase: MAVEN_GPG_PASSPHRASE # env variable for GPG private key passphrase - name: Download circle-conferencing.war Artifact uses: actions/download-artifact@v4 @@ -298,6 +299,7 @@ jobs: - name: Run Integration Test for webinar run: | + sleep 10 cd test python3 test_main.py ${{ secrets.STAGING_SERVER_URL }} ${{ secrets.USER_NAME }} ${{ secrets.PASSWORD }} /tmp/circle-webinar.war true @@ -340,12 +342,40 @@ jobs: steps: - uses: actions/checkout@v3 + - name: Set up JDK 17 + uses: actions/setup-java@v3 + with: + java-version: 17 + distribution: 'temurin' + server-id: ossrh # Value of the distributionManagement/repository/id field of the pom.xml + server-username: MAVEN_USERNAME # env variable for username in deploy + server-password: MAVEN_PASSWORD # env variable for token in deploy + gpg-private-key: ${{ secrets.MVN_GPG_KEY }} # Value of the GPG private key to import + gpg-passphrase: MAVEN_GPG_PASSPHRASE # env variable for GPG private key passphrase + + - name: Build React application for conferencing + env: + CI: false + NODE_OPTIONS: '--max-old-space-size=4096' + run: | + cd react + mv .env.production.conferencing .env.production + sed -i "s#^REACT_APP_TURN_SERVER_URL=.*#REACT_APP_TURN_SERVER_URL=\"turn:${{ secrets.STAGING_TURN_URL }}\"#" .env.production + sed -i "s#^REACT_APP_TURN_SERVER_USERNAME=.*#REACT_APP_TURN_SERVER_USERNAME=\"${{ secrets.STAGING_TURN_USERNAME }}\"#" .env.production + sed -i "s#^REACT_APP_TURN_SERVER_CREDENTIAL=.*#REACT_APP_TURN_SERVER_CREDENTIAL=\"${{ secrets.STAGING_TURN_PASSWORD }}\"#" .env.production + npm install + npm run build + cd .. + rm -r webapp/src/main/webapp/static/* + cp -a react/build/. webapp/src/main/webapp + - name: Publish to Maven Central run: | cd webapp ls -alh ls -alh target/ mvn -e deploy -DskipTests --quiet --settings ../mvn-settings.xml + sleep 60 env: MAVEN_USERNAME: ${{ secrets.MVN_USERNAME }} MAVEN_PASSWORD: ${{ secrets.MVN_PASSWORD }} diff --git a/react/.env.production b/react/.env.production index 4c39323b1..983d5469d 100644 --- a/react/.env.production +++ b/react/.env.production @@ -54,10 +54,12 @@ REACT_APP_LAYOUT_OTHERS_CARD_VISIBILITY=true REACT_APP_TIME_ZONE_LIVE_TEXT_VISIBILITY=true # Video Overlay configurations -REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED=true +REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED=false +REACT_APP_VIDEO_OVERLAY_ADMIN_CAMERA_CONTROL_ENABLED=false # Participant Tab configurations REACT_APP_PARTICIPANT_TAB_ADMIN_MODE_ENABLED=true +REACT_APP_PARTICIPANT_TAB_MUTE_PARTICIPANT_BUTTON_ENABLED=true # Meeting Recording configuration REACT_APP_RECORDING_MANAGED_BY_ADMIN=false @@ -68,6 +70,9 @@ REACT_APP_RECORDING_MANAGED_BY_ADMIN=false # Speed Test configurations REACT_APP_SPEED_TEST_BEFORE_JOINING_THE_ROOM=false +# Virtual Background configurations +REACT_APP_VIRTUAL_BACKGROUND_IMAGES="https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background0.png,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background1.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background2.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background3.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background4.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background5.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background7.jpg" + # URL configurations REACT_APP_FOOTER_LOGO_ON_CLICK_URL="https://antmedia.io/circle" REACT_APP_REPORT_PROBLEM_URL="https://github.com/ant-media/conference-call-application/issues" \ No newline at end of file diff --git a/react/.env.production.webinar b/react/.env.production.webinar index a0a40ed12..41eaf94e7 100644 --- a/react/.env.production.webinar +++ b/react/.env.production.webinar @@ -55,12 +55,14 @@ REACT_APP_TIME_ZONE_LIVE_TEXT_VISIBILITY=true # Video Overlay configurations REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED=true +REACT_APP_VIDEO_OVERLAY_ADMIN_CAMERA_CONTROL_ENABLED=false # Speed Test configurations REACT_APP_SPEED_TEST_BEFORE_JOINING_THE_ROOM=true # Participant Tab configurations REACT_APP_PARTICIPANT_TAB_ADMIN_MODE_ENABLED=true +REACT_APP_PARTICIPANT_TAB_MUTE_PARTICIPANT_BUTTON_ENABLED=true # Meeting Recording configuration REACT_APP_RECORDING_MANAGED_BY_ADMIN=true @@ -71,6 +73,9 @@ REACT_APP_FORCE_THEME="white" # Speed Test configurations REACT_APP_SPEED_TEST_BEFORE_JOINING_THE_ROOM=true +# Virtual Background configurations +REACT_APP_VIRTUAL_BACKGROUND_IMAGES="https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background0.png,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background1.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background2.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background3.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background4.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background5.jpg,https://raw.githubusercontent.com/ant-media/conference-call-application/main/static/virtualBackgroundImages/virtual-background7.jpg" + # URL configurations REACT_APP_FOOTER_LOGO_ON_CLICK_URL="" REACT_APP_REPORT_PROBLEM_URL="" diff --git a/react/package-lock.json b/react/package-lock.json index 54c7a6cfe..c1951474d 100644 --- a/react/package-lock.json +++ b/react/package-lock.json @@ -8,7 +8,7 @@ "name": "antmedia-cra", "version": "2.12.0-SNAPSHOT", "dependencies": { - "@antmedia/webrtc_adaptor": "^2.12.0-SNAPSHOT-2024-Oct-19-07-47", + "@antmedia/webrtc_adaptor": "2.12.0-SNAPSHOT-2024-Nov-29-04-11", "@charkour/react-reactions": "^0.11.0", "@emotion/react": "^11.13.3", "@emotion/styled": "^11.13.0", @@ -92,10 +92,9 @@ } }, "node_modules/@antmedia/webrtc_adaptor": { - "version": "2.12.0-SNAPSHOT-2024-Sep-10-07-00", - "resolved": "https://registry.npmjs.org/@antmedia/webrtc_adaptor/-/webrtc_adaptor-2.12.0-SNAPSHOT-2024-Sep-10-07-00.tgz", - "integrity": "sha512-vvQiJ4ib+3tIylun+6bKHybGCs0ujUdXt03pNuvZ//zzWZlAUjcoceZTmcTt2Wr8CcMQK/mKDO/93b5Jf6bnIw==", - "license": "ISC", + "version": "2.12.0-SNAPSHOT-2024-Nov-29-04-11", + "resolved": "https://registry.npmjs.org/@antmedia/webrtc_adaptor/-/webrtc_adaptor-2.12.0-SNAPSHOT-2024-Nov-29-04-11.tgz", + "integrity": "sha512-Og0qRIuby0jtDMsDJvsKGSnIaFS9+IajeqHDEQ+YhJTvoc9yQIUTh8+1m2GN1Nas8pY7qNRImJWlSDKorjwMrQ==", "dependencies": { "@mediapipe/selfie_segmentation": "^0.1.1675465747", "url": "^0.11.1" diff --git a/react/package.json b/react/package.json index 29fd71c2a..c187abab4 100644 --- a/react/package.json +++ b/react/package.json @@ -4,7 +4,7 @@ "private": true, "homepage": ".", "dependencies": { - "@antmedia/webrtc_adaptor": "^2.12.0-SNAPSHOT-2024-Oct-19-07-47", + "@antmedia/webrtc_adaptor": "2.12.0-SNAPSHOT-2024-Nov-29-04-11", "@charkour/react-reactions": "^0.11.0", "@emotion/react": "^11.13.3", "@emotion/styled": "^11.13.0", diff --git a/react/public/speed-test-sample-video.mp4 b/react/public/speed-test-sample-video.mp4 deleted file mode 100644 index ed139d6d5..000000000 Binary files a/react/public/speed-test-sample-video.mp4 and /dev/null differ diff --git a/react/src/Components/Cards/VideoCard.js b/react/src/Components/Cards/VideoCard.js index 82842170d..e51876164 100644 --- a/react/src/Components/Cards/VideoCard.js +++ b/react/src/Components/Cards/VideoCard.js @@ -1,118 +1,156 @@ -import React, { useCallback, useContext, useEffect } from "react"; +import React, { useCallback, useContext, useEffect, useState, useRef } from "react"; import { alpha, styled } from "@mui/material/styles"; import { ConferenceContext } from "pages/AntMedia"; import DummyCard from "./DummyCard"; import { Grid, Typography, useTheme, Box, Tooltip, Fab } from "@mui/material"; import { SvgIcon } from "../SvgIcon"; import { useTranslation } from "react-i18next"; -import { isMobile, isTablet } from 'react-device-detect'; +import { isMobile, isTablet } from "react-device-detect"; +import {parseMetaData} from "../../utils"; const CustomizedVideo = styled("video")({ - borderRadius: 4, - width: "100%", - height: "100%", - objectPosition: "center", - backgroundColor: "transparent", + borderRadius: 4, + width: "100%", + height: "100%", + objectPosition: "center", + backgroundColor: "transparent", }); + const CustomizedBox = styled(Box)(({ theme }) => ({ - backgroundColor: alpha(theme.palette.gray[90], 0.3), + backgroundColor: alpha(theme.palette.gray[90], 0.3), })); function VideoCard(props) { - const conference = useContext(ConferenceContext); - - const { t } = useTranslation(); - const [displayHover, setDisplayHover] = React.useState(false); - const theme = useTheme(); - - const cardBtnStyle = { - display: "flex", - justifyContent: "center", - alignItems: "center", - width: { xs: "6vw", md: 32 }, - height: { xs: "6vw", md: 32 }, - borderRadius: "50%", - position: "relative", - }; - - const refVideo = useCallback( (node) => { - if (node && props.trackAssignment.track) { - node.srcObject = new MediaStream([props.trackAssignment.track]); - node.play().then(()=> {}).catch((e) => { console.log("play failed because ", e)}); - } - }, - [props.trackAssignment.track] - ); - - let useAvatar = true; - if(props?.trackAssignment.isMine) { - useAvatar = conference?.isMyCamTurnedOff; - } - else if (props.trackAssignment.track?.kind === "video") { - let broadcastObject = conference?.allParticipants[props?.trackAssignment.streamId]; - let metaData = broadcastObject?.metaData; - useAvatar = !parseMetaDataAndGetIsCameraOn(metaData) && !parseMetaDataAndGetIsScreenShared(metaData); - } - - function isJsonString(str) { - try { - JSON.parse(str); - } catch (e) { - return false; - } - return true; - } - - function parseMetaDataAndGetIsCameraOn(metaData) { - if (!metaData) return false; - return (isJsonString(metaData)) ? JSON.parse(metaData).isCameraOn : false; - } - - function parseMetaDataAndGetIsScreenShared(metaData) { - if (!metaData) return false; - return (isJsonString(metaData)) ? JSON.parse(metaData).isScreenShared : false; - } - - function parseMetaDataAndGetIsMicMuted(metaData) { - if (!metaData) return true; - return (isJsonString(metaData)) ? JSON.parse(metaData).isMicMuted : true; - } - - const micMuted = (props?.trackAssignment.isMine) ? conference?.isMyMicMuted : parseMetaDataAndGetIsMicMuted(conference?.allParticipants[props?.trackAssignment.streamId]?.metaData); + const conference = useContext(ConferenceContext); + const { t } = useTranslation(); + const [displayHover, setDisplayHover] = useState(false); + const [isTalking, setIsTalking] = useState(false); + const theme = useTheme(); + const timeoutRef = useRef(null); + + const refVideo = useCallback((node) => { + if (node && props.trackAssignment.track) { + node.srcObject = new MediaStream([props.trackAssignment.track]); + node.play().catch((e) => console.error("Video playback failed:", e)); + } + }, [props.trackAssignment.track]); + + const cardBtnStyle = { + display: "flex", + justifyContent: "center", + alignItems: "center", + width: { xs: "6vw", md: 32 }, + height: { xs: "6vw", md: 32 }, + borderRadius: "50%", + position: "relative", + }; + + const isMine = props.trackAssignment?.isMine; + const isVideoTrack = props.trackAssignment.track?.kind === "video"; + + const micMuted = isMine + ? conference?.isMyMicMuted + : parseMetaData( + conference?.allParticipants?.[props.trackAssignment.streamId]?.metaData, + "isMicMuted" + ); + + const useAvatar = isMine + ? conference?.isMyCamTurnedOff + : !parseMetaData( + conference?.allParticipants?.[props.trackAssignment.streamId]?.metaData, + "isCameraOn" + ) && + !parseMetaData( + conference?.allParticipants?.[props.trackAssignment.streamId]?.metaData, + "isScreenShared" + ); + + useEffect(() => { + if (props?.trackAssignment.isMine && conference.isPublished && !conference.isPlayOnly) { + conference.setAudioLevelListener((value) => { + // sounds under 0.01 are probably background noise + if (value >= 0.01) { + if (isTalking === false) setIsTalking(true); + clearInterval(timeoutRef.current); + timeoutRef.current = setTimeout(() => { + setIsTalking(false); + }, 1500); + } + }, 1000); + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [conference.isPublished]); + + const OverlayButton = ({ title, icon, color, onClick }) => ( + + + + + + ); - const [isTalking, setIsTalking] = React.useState(false); + const AdministrativeButtons = ({ micMuted, useAvatar }) => { + const handleToggleMic = () => { + const participant = { + streamId: props.trackAssignment.streamId, + streamName: props.name, + }; + conference?.setParticipantIdMuted(participant); + micMuted + ? conference?.turnOnYourMicNotification(participant.streamId) + : conference?.turnOffYourMicNotification(participant.streamId); + }; + + const handleToggleCam = () => { + const participant = { + streamId: props.trackAssignment.streamId, + streamName: props.name, + }; + conference?.setParticipantIdMuted(participant); + conference?.turnOffYourCamNotification(participant.streamId); + }; + + return ( + <> + {(!useAvatar && process.env.REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED === "true") && ( + + )} + + + ); + }; + + const PinButton = ({ }) => ( + conference.pinVideo(props.trackAssignment.streamId)} + /> + ); + const renderOverlayButtons = () => { + if (props.hidePin) return null; - const timeoutRef = React.useRef(null); + const isAdminMode = process.env.REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED === "true"; + const isAdministrativeButtonsVisible = + !props?.trackAssignment.isMine && (!isAdminMode || conference.isAdmin); - const mirrorView = props?.trackAssignment.isMine; - //const isScreenSharing = - // conference?.isScreenShared || - // conference?.screenSharedVideoId === props?.trackAssignment.streamId; - //conference?.isScreenShared means am i sharing my screen - //conference?.screenSharedVideoId === props?.trackAssignment.streamId means is someone else sharing their screen - useEffect(() => { - if (props?.trackAssignment.isMine && conference.isPublished && !conference.isPlayOnly) { - conference.setAudioLevelListener((value) => { - // sounds under 0.01 are probably background noise - if (value >= 0.01) { - if (isTalking === false) setIsTalking(true); - clearInterval(timeoutRef.current); - timeoutRef.current = setTimeout(() => { - setIsTalking(false); - }, 1500); - } - }, 1000); - } - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [conference.isPublished]); - - const overlayButtonsGroup = () => { - if (process.env.REACT_APP_VIDEO_OVERLAY_ADMIN_MODE_ENABLED === "true") { - return (!props.hidePin && ( + return ( - - - {(!isMobile) && (!isTablet) ? - - { - conference.pinVideo(props.trackAssignment.streamId); - }} - color="primary" - aria-label={props.pinned ? t("unpin") : t("pin")} - size="small" - > - - - - : null } - - { !props?.trackAssignment.isMine && conference.isAdmin && conference.isAdmin === true ? - - {!useAvatar ? - - { - let participant = {}; - participant.streamId=props.trackAssignment.streamId; - participant.streamName=props.name; - conference?.setParticipantIdMuted(participant); - conference?.turnOffYourCamNotification(participant.streamId); - }} - color="primary" - aria-label="turn-off-camera" - size="small" - > - - - - : - - - - - - } - - : null } - - {(!props?.trackAssignment.isMine && conference.isAdmin && conference.isAdmin === true) ? - - {!micMuted ? - - { - let participant = {}; - participant.streamId=props.trackAssignment.streamId; - participant.streamName=props.name; - conference?.setParticipantIdMuted(participant); - conference?.turnOffYourMicNotification(participant.streamId); - }} - color="primary" - aria-label="mute" - size="small" - > - - - - : - { - let participant = {}; - participant.streamId=props.trackAssignment.streamId; - participant.streamName=props.name; - conference?.setParticipantIdMuted(participant); - conference?.turnOnYourMicNotification(participant.streamId); - }} - color="error" - aria-label="unmute" - size="small" - > - - - } - - : null } + + + {!isMobile && !isTablet && } + {isAdministrativeButtonsVisible && } - )) - } else { - return (!props.hidePin && ( - - - - {(!isMobile) && (!isTablet) ? - - {conference.pinVideo(props.trackAssignment.streamId);}} - color="primary" - aria-label={props.pinned ? "unpin" : "pin"} - size="small" - > - - - - : null } + ); + }; - {(!props?.trackAssignment.isMine && !micMuted) ? - - - { - let participant = {}; - participant.streamId=props.trackAssignment.streamId; - participant.streamName=props.name; - conference?.setParticipantIdMuted(participant); - conference?.turnOffYourMicNotification(participant.streamId); - conference?.setMuteParticipantDialogOpen(true); + const renderAvatarOrPlayer = () => ( + <> + {useAvatar ? ( + + + + ) : ( + - - - - - : null} - - - - )) - }} - const avatarOrPlayer = () => { - return ( - <> - - - + > + + + )} + + ); + const renderParticipantStatus = () => ( - - - - ) - } - - const overlayParticipantStatus = () => { - return ( - - {micMuted && ( - - - - - - - - )} - {/* - - - + {micMuted && ( + + + + + + + + )} - */} - {props.pinned && ( - - - - - - - - )} - - ); - } - - const overlayVideoTitle = () => { - return ( - props.name && ( -
- - {props.name}{" "} - {process.env.NODE_ENV === "development" - ? `${props?.trackAssignment.isMine - ? props.trackAssignment.streamId + - " " + - conference.streamName - : props.trackAssignment.streamId + " " + props.trackAssignment.track?.id - }` - : ""} - -
- ) ); - } - const isTalkingFrame = () => { - return ( -
- ); - } - - const setLocalVideo = () => { - let tempLocalVideo = document.getElementById((typeof conference?.publishStreamId === "undefined")? "localVideo" : conference?.publishStreamId); - if(props?.trackAssignment.isMine && conference.localVideo !== tempLocalVideo) { - conference?.localVideoCreate(tempLocalVideo); + const setLocalVideo = () => { + let tempLocalVideo = document.getElementById((typeof conference?.publishStreamId === "undefined")? "localVideo" : conference?.publishStreamId); + if(props.trackAssignment.isMine && conference.localVideo !== tempLocalVideo) { + conference?.localVideoCreate(tempLocalVideo); + } } - }; - return props?.trackAssignment.isMine || props.trackAssignment.track?.kind !== "audio" ? ( - <> - setDisplayHover(true)} - onMouseLeave={(e) => setDisplayHover(false)} - > + const overlayVideoTitle = () => { + return ( + props.name && ( +
+ + {props.name}{" "} + {process.env.NODE_ENV === "development" + ? `${props?.trackAssignment.isMine + ? props.trackAssignment.streamId + + " " + + conference.streamName + : props.trackAssignment.streamId + " " + props.trackAssignment.track?.id + }` + : ""} + +
+ ) + ); + } - {overlayButtonsGroup()} + const isTalkingFrame = () => { + return ( +
+ ); + } -
+ setDisplayHover(true)} + onMouseLeave={() => setDisplayHover(false)} > - {avatarOrPlayer()} - - {setLocalVideo()} - - {overlayParticipantStatus()} - - {isTalkingFrame()} - - {overlayVideoTitle()} - -
- - - ) : ( - //for audio tracks - <> - - - ); + {renderOverlayButtons()} +
+ {renderAvatarOrPlayer()} + {renderParticipantStatus()} + {setLocalVideo()} + {isTalkingFrame()} + {overlayVideoTitle()} +
+ + + ) : ( + //for audio tracks + <> + + + ); }; export default VideoCard; diff --git a/react/src/Components/EffectsTab.js b/react/src/Components/EffectsTab.js index b9f7ee7b9..4f263792c 100644 --- a/react/src/Components/EffectsTab.js +++ b/react/src/Components/EffectsTab.js @@ -5,7 +5,6 @@ import { SvgIcon } from "./SvgIcon"; import { ConferenceContext } from "pages/AntMedia"; import {CustomizedBtn} from "./Footer/Components/MicButton"; import {useTheme} from "@mui/material"; -import virtualBackgroundImageData from 'virtualBackground.json'; import {useSnackbar} from "notistack"; import {useTranslation} from "react-i18next"; @@ -51,8 +50,9 @@ function EffectsTab() { position: 'relative' }} id="custom-virtual-background-button" + data-testid="custom-virtual-background-button" onClick={(e) => { - conference.setAndEnableVirtualBackgroundImage(imageSrc); + conference.setVirtualBackgroundImage(imageSrc); }} > { + let virtualBackgroundImageData = []; + + if (process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES !== undefined && process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES !== null) { + virtualBackgroundImageData = process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES.split(','); + } + const images = []; let imageIndex = 0; - for (let i = 0; i < virtualBackgroundImageData.virtualBackgroundImages.length; i++) { + for (let i = 0; i < virtualBackgroundImageData.length; i++) { images.push( - getVirtualBackgroundButton(virtualBackgroundImageData.virtualBackgroundImages[i], "image"+imageIndex, imageIndex, false) + getVirtualBackgroundButton(virtualBackgroundImageData[i], "image"+imageIndex, imageIndex, false) ); ++imageIndex; } diff --git a/react/src/Components/Footer/Components/ParticipantListButton.js b/react/src/Components/Footer/Components/ParticipantListButton.js index d9c4a62ea..b9f35191d 100644 --- a/react/src/Components/Footer/Components/ParticipantListButton.js +++ b/react/src/Components/Footer/Components/ParticipantListButton.js @@ -38,7 +38,7 @@ function ParticipantListButton({ footer }) { > {/* eslint-disable-next-line jsx-a11y/anchor-is-valid */} - {Object.keys(conference.allParticipants).length} + {conference.participantCount} ); diff --git a/react/src/Components/ParticipantTab.js b/react/src/Components/ParticipantTab.js index 0fe94661c..4ac7fcc96 100644 --- a/react/src/Components/ParticipantTab.js +++ b/react/src/Components/ParticipantTab.js @@ -6,8 +6,9 @@ import Button from "@mui/material/Button"; import {styled, useTheme} from "@mui/material/styles"; import { SvgIcon } from "./SvgIcon"; import { ConferenceContext } from "pages/AntMedia"; -import {CircularProgress} from "@mui/material"; +import {CircularProgress, Pagination} from "@mui/material"; import {WebinarRoles} from "../WebinarRoles"; +import {parseMetaData} from "../utils"; const ParticipantName = styled(Typography)(({ theme }) => ({ color: theme.palette.textColor, @@ -25,9 +26,51 @@ function ParticipantTab(props) { const conference = React.useContext(ConferenceContext); const theme = useTheme(); + const paginationUpdate = (event, value) => { + conference?.updateAllParticipantsPagination(value); + } + + const handleToggleMic = (isMicMuted, streamId, streamName) => { + if (streamId === conference?.publishStreamId && !conference?.isMyMicMuted) { + conference?.muteLocalMic(); + return; + } + + const participant = { + streamId: streamId, + streamName: streamName, + }; + conference?.setParticipantIdMuted(participant); + if (!isMicMuted) { + conference?.turnOffYourMicNotification(participant.streamId); + } + }; + + const getMuteParticipantButton = (streamId) => { + let micMuted = false; + if (streamId === conference?.publishStreamId) { + micMuted = conference?.isMyMicMuted; + } else { + micMuted =parseMetaData(conference.pagedParticipants[streamId]?.metaData, "isMicMuted"); + } + let name = conference.pagedParticipants[streamId]?.name; + + return ( + { handleToggleMic(micMuted, streamId, name) } + } + > + + + ) + } + const getAdminButtons = (streamId, assignedVideoCardId) => { let publishStreamId = (streamId === "localVideo") ? conference.publishStreamId : streamId; - let role = conference.allParticipants[publishStreamId]?.role; + let role = conference.pagedParticipants[publishStreamId]?.role; return (
@@ -47,7 +90,7 @@ function ParticipantTab(props) { { ( role === WebinarRoles.Host || role === WebinarRoles.Speaker || role === WebinarRoles.TempListener ) && conference?.isAdmin === true ?( { conference?.makeParticipantPresenter(publishStreamId) } @@ -97,7 +140,7 @@ function ParticipantTab(props) {
- {(typeof conference.allParticipants[streamId]?.isPinned !== "undefined") && (conference.allParticipants[streamId]?.isPinned === true) ? ( + {(typeof conference.pagedParticipants[streamId]?.isPinned !== "undefined") && (conference.pagedParticipants[streamId]?.isPinned === true) ? (
@@ -139,20 +185,32 @@ function ParticipantTab(props) { variant="body2" style={{marginLeft: 4, fontWeight: 500}} > - {Object.keys(conference.allParticipants).length} + {conference?.participantCount} - {conference.isPlayOnly === false ? getParticipantItem(conference.publishStreamId, "You") : ""} - {Object.entries(conference.allParticipants).map(([streamId, broadcastObject]) => { + {Object.entries(conference.pagedParticipants).map(([streamId, broadcastObject]) => { if (conference.publishStreamId !== streamId) { - var assignedVideoCardId = conference?.videoTrackAssignments?.find(vta => vta.streamId === streamId)?.videoLabel; + let assignedVideoCardId = conference?.videoTrackAssignments?.find(vta => vta.streamId === streamId)?.videoLabel; return getParticipantItem(streamId, broadcastObject.name, assignedVideoCardId); } else { - return ""; + return getParticipantItem(conference.publishStreamId, "You"); } })} + {/* Pagination Controls */} + + + ); diff --git a/react/src/__tests__/Components/EffectsTab.test.js b/react/src/__tests__/Components/EffectsTab.test.js new file mode 100644 index 000000000..b47e0728c --- /dev/null +++ b/react/src/__tests__/Components/EffectsTab.test.js @@ -0,0 +1,85 @@ +// src/EffectsTab.test.js +import React from 'react'; +import { render } from '@testing-library/react'; +import { ConferenceContext } from 'pages/AntMedia'; +import EffectsTab from "../../Components/EffectsTab"; +import theme from "../../styles/theme"; +import {ThemeList} from "../../styles/themeList"; +import {ThemeProvider} from "@mui/material"; + +const mockOpfsRoot = { + values: jest.fn(() => ({ + [Symbol.asyncIterator]: async function* () { + yield { name: 'file1.txt' }; + yield { name: 'file2.txt' }; + }, + })), +}; + +const contextValue = { + allParticipants: { + 'test-stream-id': { + role: 'host', + participantID: 'test-participant-id', + streamID: 'test-stream-id', + videoTrack: 'test-video-track', + audioTrack: 'test-audio-track', + videoLabel: 'test-video-label', + }, + }, + publishStreamId: 'test-stream-id', + setVirtualBackgroundImage: jest.fn(), +}; + +// Mock the useContext hook +jest.mock('react', () => ({ + ...jest.requireActual('react'), + useContext: jest.fn(), +})); + +describe('Effects Tab Component', () => { + + beforeEach(() => { + // Reset the mock implementation before each test + jest.clearAllMocks(); + + Object.defineProperty(navigator, 'storage', { + value: { + getDirectory: jest.fn().mockResolvedValue(mockOpfsRoot), + }, + writable: true, + }); + + React.useContext.mockImplementation(input => { + if (input === ConferenceContext) { + return contextValue; + } + return jest.requireActual('react').useContext(input); + }); + }); + + it('renders without crashing', () => { + render( + + + + ); + }); + + describe('getBackgroundImages', () => { + it('returns an empty array when no environment variable or custom images are provided', () => { + process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES = undefined; + const {getByTestId} = render( + + + + ); + let customVirtualBackgroundButton = getByTestId('custom-virtual-background-button'); + customVirtualBackgroundButton.click(); + expect(contextValue.setVirtualBackgroundImage).toHaveBeenCalled(); + }); + + + }); + +}); diff --git a/react/src/__tests__/Components/Footer/Components/EndCallButton.test.js b/react/src/__tests__/Components/Footer/Components/EndCallButton.test.js new file mode 100644 index 000000000..346a81f4e --- /dev/null +++ b/react/src/__tests__/Components/Footer/Components/EndCallButton.test.js @@ -0,0 +1,39 @@ +// src/EndCallButton.test.js +import React from 'react'; +import { render } from '@testing-library/react'; +import { ConferenceContext } from 'pages/AntMedia'; +import EndCallButton from "../../../../Components/Footer/Components/EndCallButton"; + +// Mock the context value +const contextValue = { + setLeftTheRoom: jest.fn(), +}; + +// Mock the useContext hook +jest.mock('react', () => ({ + ...jest.requireActual('react'), + useContext: jest.fn(), +})); + +describe('End Call Button Component', () => { + + beforeEach(() => { + // Reset the mock implementation before each test + jest.clearAllMocks(); + + React.useContext.mockImplementation(input => { + if (input === ConferenceContext) { + return contextValue; + } + return jest.requireActual('react').useContext(input); + }); + }); + + + it('renders without crashing', () => { + render( + + ); + }); + +}); diff --git a/react/src/__tests__/Components/Footer/Components/FakeParticipantButton.test.js b/react/src/__tests__/Components/Footer/Components/FakeParticipantButton.test.js new file mode 100644 index 000000000..3fba95290 --- /dev/null +++ b/react/src/__tests__/Components/Footer/Components/FakeParticipantButton.test.js @@ -0,0 +1,39 @@ +// src/FakeParticipantButton.test.js +import React from 'react'; +import { render } from '@testing-library/react'; +import { ConferenceContext } from 'pages/AntMedia'; +import FakeParticipantButton from "../../../../Components/Footer/Components/FakeParticipantButton"; + +// Mock the context value +const contextValue = { + setLeftTheRoom: jest.fn(), +}; + +// Mock the useContext hook +jest.mock('react', () => ({ + ...jest.requireActual('react'), + useContext: jest.fn(), +})); + +describe('Fake Participant Button Component', () => { + + beforeEach(() => { + // Reset the mock implementation before each test + jest.clearAllMocks(); + + React.useContext.mockImplementation(input => { + if (input === ConferenceContext) { + return contextValue; + } + return jest.requireActual('react').useContext(input); + }); + }); + + + it('renders without crashing', () => { + render( + + ); + }); + +}); diff --git a/react/src/__tests__/Components/Footer/Components/ParticipantListButton.test.js b/react/src/__tests__/Components/Footer/Components/ParticipantListButton.test.js index 9907f5996..eb3a35a6d 100644 --- a/react/src/__tests__/Components/Footer/Components/ParticipantListButton.test.js +++ b/react/src/__tests__/Components/Footer/Components/ParticipantListButton.test.js @@ -1,6 +1,6 @@ // src/Button.test.js import React from 'react'; -import { render } from '@testing-library/react'; +import {act, render} from '@testing-library/react'; import { ConferenceContext } from 'pages/AntMedia'; import ParticipantListButton from 'Components/Footer/Components/ParticipantListButton'; import { random } from 'lodash'; @@ -8,6 +8,8 @@ import { random } from 'lodash'; // Mock the context value const contextValue = { allParticipants: {}, + participantCount: 0, + setParticipantCount: jest.fn() }; // Mock the useContext hook @@ -37,11 +39,14 @@ describe('ParticipantList Button Component', () => { ); }); - it('check the count on button', () => { + it('check the count on button', async () => { var noOfParticipants = random(1, 10); - for (let i = 0; i < noOfParticipants; i++) { - contextValue.allParticipants[`k${i}`] = `v${i}`; - } + + await act(()=> + { + contextValue.setParticipantCount(noOfParticipants) + contextValue.participantCount = noOfParticipants; + }); const { container, getByText, getByRole } = render( diff --git a/react/src/__tests__/Components/ParticipantTab.test.js b/react/src/__tests__/Components/ParticipantTab.test.js index 8cff6c6fc..f60984c00 100644 --- a/react/src/__tests__/Components/ParticipantTab.test.js +++ b/react/src/__tests__/Components/ParticipantTab.test.js @@ -21,13 +21,63 @@ const contextValue = { audioTrack: 'test-audio-track', videoLabel: 'test-video-label', }, + 'test-stream-id-2': { + role: 'host', + participantID: 'test-participant-id-2', + streamID: 'test-stream-id-2', + videoTrack: 'test-video-track-2', + audioTrack: 'test-audio-track-2', + videoLabel: 'test-video-label-2', + metaData: { + isMuted: false + } + } }, publishStreamId: 'test-stream-id', pinVideo: jest.fn(), makeParticipantPresenter: jest.fn(), + pagedParticipants: { + 'test-stream-id': { + role: 'host', + participantID: 'test-participant-id', + streamID: 'test-stream-id', + videoTrack: 'test-video-track', + audioTrack: 'test-audio-track', + videoLabel: 'test-video-label', + }, + 'test-stream-id-2': { + role: 'host', + participantID: 'test-participant-id-2', + streamID: 'test-stream-id-2', + videoTrack: 'test-video-track-2', + audioTrack: 'test-audio-track-2', + videoLabel: 'test-video-label-2', + metaData: { + isMuted: false + } + } + }, isAdmin: true, isPlayOnly: false, - videoTrackAssignments: [{streamID: 'test-stream-id', participantID: 'test-participant-id', videoTrack: 'test-video-track', audioTrack: 'test-audio-track', videoLabel: 'test-video-label'}], + videoTrackAssignments: [ + {streamID: 'test-stream-id', participantID: 'test-participant-id', videoTrack: 'test-video-track', audioTrack: 'test-audio-track', videoLabel: 'test-video-label'}, + {streamID: 'test-stream-id-2', participantID: 'test-participant-id-2', videoTrack: 'test-video-track-2', audioTrack: 'test-audio-track-2', videoLabel: 'test-video-label-2'} + ], + globals: { + maxVideoTrackCount: 6, + desiredTileCount: 6, + trackEvents: [], + participantListPagination: { + currentPage: 1, + pageSize: 15, + totalPage: 1, + startIndex: 0, + endIndex: 15 + } + }, + muteLocalMic: jest.fn(), + turnOffYourMicNotification: jest.fn(), + setParticipantIdMuted: jest.fn(), }; // Mock the useContext hook @@ -94,5 +144,41 @@ describe('ParticipantTab Component', () => { const presenterButton = getByTestId('add-presenter-test-stream-id'); expect(presenterButton).toBeInTheDocument(); }); + + it('check muteLocalMic called in getMuteParticipantButton', () => { + contextValue.isAdmin = true; + process.env.REACT_APP_PARTICIPANT_TAB_MUTE_PARTICIPANT_BUTTON_ENABLED=true + + const { getByTestId } = render( + + + + ); + const micToggleParticipant = getByTestId('mic-toggle-participant-test-stream-id'); + expect(micToggleParticipant).toBeInTheDocument(); + + micToggleParticipant.click(); + expect(contextValue.muteLocalMic).toHaveBeenCalled(); + expect(contextValue.setParticipantIdMuted).not.toHaveBeenCalled(); + expect(contextValue.turnOffYourMicNotification).not.toHaveBeenCalled(); + }); + + it('check turnOffYourMicNotification called in getMuteParticipantButton', () => { + contextValue.isAdmin = true; + process.env.REACT_APP_PARTICIPANT_TAB_MUTE_PARTICIPANT_BUTTON_ENABLED=true + + const { getByTestId } = render( + + + + ); + const micToggleParticipant = getByTestId('mic-toggle-participant-test-stream-id-2'); + expect(micToggleParticipant).toBeInTheDocument(); + + micToggleParticipant.click(); + expect(contextValue.muteLocalMic).not.toHaveBeenCalled(); + expect(contextValue.setParticipantIdMuted).toHaveBeenCalled(); + expect(contextValue.turnOffYourMicNotification).toHaveBeenCalled(); + }); }); diff --git a/react/src/__tests__/pages/AntMedia.test.js b/react/src/__tests__/pages/AntMedia.test.js index f6eb3ede8..03d61009b 100644 --- a/react/src/__tests__/pages/AntMedia.test.js +++ b/react/src/__tests__/pages/AntMedia.test.js @@ -10,6 +10,7 @@ import {ThemeList} from "styles/themeList"; import theme from "styles/theme"; import { times } from 'lodash'; import { useParams } from 'react-router-dom'; +import {VideoEffect} from "@antmedia/webrtc_adaptor"; var webRTCAdaptorConstructor, webRTCAdaptorScreenConstructor, webRTCAdaptorPublishSpeedTestPlayOnlyConstructor, webRTCAdaptorPublishSpeedTestConstructor, webRTCAdaptorPlaySpeedTestConstructor; var currentConference; @@ -80,10 +81,13 @@ jest.mock('@antmedia/webrtc_adaptor', () => ({ createSpeedTestForPlayWebRtcAdaptor: jest.fn(), requestVideoTrackAssignments: jest.fn(), stopSpeedTest: jest.fn().mockImplementation(() => console.log('stopSpeedTest')), - getSubtracks: jest.fn(), closeStream: jest.fn(), closeWebSocket: jest.fn(), - playStats: {} + playStats: {}, + enableEffect: jest.fn(), + setSelectedVideoEffect: jest.fn(), + setBlurEffectRange: jest.fn(), + getSubtrackCount: jest.fn(), } for (var key in params) { @@ -1304,7 +1308,7 @@ describe('AntMedia Component', () => { }; const weak_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; - const unstable_msg = "Poor Network Connection Warning:Network connection is not stable. Please check your connection!"; + const unstable_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation(); @@ -1424,7 +1428,7 @@ describe('AntMedia Component', () => { }); const weak_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; - const unstable_msg = "Poor Network Connection Warning:Network connection is not stable. Please check your connection!"; + const unstable_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation(); @@ -1597,7 +1601,7 @@ describe('AntMedia Component', () => { }; const weak_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; - const unstable_msg = "Poor Network Connection Warning:Network connection is not stable. Please check your connection!"; + const unstable_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation(); @@ -1747,22 +1751,22 @@ describe('AntMedia Component', () => { }; const weak_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; - const unstable_msg = "Poor Network Connection Warning:Network connection is not stable. Please check your connection!"; + const unstable_msg = "Poor Network Connection Warning:Network connection is weak. You may encounter connection drop!"; const consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation(); await act(async () => { webRTCAdaptorConstructor.callback("updated_stats", mockStats); - mockStats.videoRoundTripTime = '150'; - mockStats.audioRoundTripTime = '160'; + mockStats.videoRoundTripTime = '0.150'; + mockStats.audioRoundTripTime = '0.160'; webRTCAdaptorConstructor.callback("updated_stats", mockStats); expect(consoleWarnSpy).toHaveBeenCalledWith(weak_msg); - mockStats.videoRoundTripTime = '120'; - mockStats.audioRoundTripTime = '130'; + mockStats.videoRoundTripTime = '0.120'; + mockStats.audioRoundTripTime = '0.130'; webRTCAdaptorConstructor.callback("updated_stats", mockStats); @@ -1786,8 +1790,8 @@ describe('AntMedia Component', () => { expect(consoleWarnSpy).toHaveBeenCalledWith(weak_msg); - mockStats.videoJitter = '60'; - mockStats.audioJitter = '70'; + mockStats.videoJitter = '0.02'; + mockStats.audioJitter = '0.10'; webRTCAdaptorConstructor.callback("updated_stats", mockStats); expect(consoleWarnSpy).toHaveBeenCalledWith(unstable_msg); @@ -1850,17 +1854,12 @@ describe('AntMedia Component', () => { currentConference.startSpeedTest(); }); - await waitFor(() => { - expect(webRTCAdaptorPlaySpeedTestConstructor).not.toBe(undefined); - }); - await waitFor(() => { expect(webRTCAdaptorPublishSpeedTestConstructor).not.toBe(undefined); }); const mockStop = jest.fn(); - webRTCAdaptorPlaySpeedTestConstructor.stop = mockStop; webRTCAdaptorPublishSpeedTestConstructor.stop = mockStop; @@ -1880,9 +1879,6 @@ describe('AntMedia Component', () => { }); /* - await waitFor(() => { - expect(webRTCAdaptorPlaySpeedTestConstructor).toBeNull(); - }); await waitFor(() => { expect(webRTCAdaptorPublishSpeedTestConstructor).toBeNull(); }); @@ -1929,6 +1925,12 @@ describe('AntMedia Component', () => { expect(webRTCAdaptorConstructor).not.toBe(undefined); }); + currentConference.setIsPlayOnly(true); + + await waitFor(() => { + expect(currentConference.isPlayOnly).toBe(true); + }); + await act(async () => { currentConference.startSpeedTest(); }); @@ -1952,7 +1954,7 @@ describe('AntMedia Component', () => { // Assert await waitFor(() => { - expect(mockStop).toHaveBeenCalledWith(`speedTestStream${currentConference.speedTestStreamId.current}`); + expect(mockStop).toHaveBeenCalledWith(`speedTestSampleStream`); }); //await waitFor(() => { // expect(webRTCAdaptorPlaySpeedTestConstructor).toBeNull(); @@ -2260,6 +2262,39 @@ describe('AntMedia Component', () => { }); }); + it('handle the case if the metadata is empty', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const subtrackList = [ + JSON.stringify({ streamId: 'stream1', metaData: null }), + JSON.stringify({ streamId: 'stream2', metaData: "" }) + ]; + const obj = { subtrackList }; + + await act(async () => { + webRTCAdaptorConstructor.callback('subtrackList', obj); + }); + + await waitFor(() => { + expect(currentConference.participantUpdated).toBe(false); + }); + + await waitFor(() => { + expect(currentConference.allParticipants["stream1"]).toBeDefined(); + expect(currentConference.allParticipants["stream2"]).toBeDefined(); + }); + }); + it('does not update allParticipants if there are no changes', async () => { const { container } = render( @@ -2323,4 +2358,1047 @@ describe('AntMedia Component', () => { }); }); + describe('fetchImageAsBlob', () => { + it('returns a blob URL when the fetch is successful', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockBlob = new Blob(['image content'], { type: 'image/png' }); + const mockUrl = 'blob:http://localhost/image'; + global.fetch = jest.fn().mockResolvedValue({ + blob: jest.fn().mockResolvedValue(mockBlob), + }); + global.URL.createObjectURL = jest.fn().mockReturnValue(mockUrl); + + const result = await currentConference.fetchImageAsBlob('http://example.com/image.png'); + + expect(result).toBe(mockUrl); + expect(global.fetch).toHaveBeenCalledWith('http://example.com/image.png'); + expect(global.URL.createObjectURL).toHaveBeenCalledWith(mockBlob); + }); + + it('throws an error when the fetch fails', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + global.fetch = jest.fn().mockRejectedValue(new Error('Fetch failed')); + + await expect(currentConference.fetchImageAsBlob('http://example.com/image.png')).rejects.toThrow('Fetch failed'); + }); + + it('throws an error when the blob conversion fails', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + global.fetch = jest.fn().mockResolvedValue({ + blob: jest.fn().mockRejectedValue(new Error('Blob conversion failed')), + }); + + await expect(currentConference.fetchImageAsBlob('http://example.com/image.png')).rejects.toThrow('Blob conversion failed'); + }); + }); + + describe('setVirtualBackgroundImage', () => { + it('returns immediately if the URL is undefined', async () => { + const {container} = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const result = currentConference.setVirtualBackgroundImage(undefined); + expect(result).toBeUndefined(); + }); + + it('returns immediately if the URL is null', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const result = currentConference.setVirtualBackgroundImage(null); + expect(result).toBeUndefined(); + }); + + it('returns immediately if the URL is an empty string', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const result = currentConference.setVirtualBackgroundImage(''); + expect(result).toBeUndefined(); + }); + + it('calls setAndEnableVirtualBackgroundImage if the URL starts with "data:image"', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockUrl = ''; + currentConference.setVirtualBackgroundImage(mockUrl); + }); + + it('fetches the image as a blob and calls setAndEnableVirtualBackgroundImage if the URL does not start with "data:image"', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockUrl = 'http://example.com/image.png'; + const mockBlobUrl = 'blob:http://localhost/image'; + global.fetch = jest.fn().mockResolvedValue({ + blob: jest.fn().mockResolvedValue(new Blob(['image content'], { type: 'image/png' })), + }); + global.URL.createObjectURL = jest.fn().mockReturnValue(mockBlobUrl); + currentConference.setAndEnableVirtualBackgroundImage = jest.fn(); + await currentConference.setVirtualBackgroundImage(mockUrl); + expect(global.fetch).toHaveBeenCalledWith(mockUrl); + }); + }); + + describe('handleBackgroundReplacement', () => { + it('disables video effect when option is "none"', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.setIsVideoEffectRunning = jest.fn(); + + currentConference.handleBackgroundReplacement("none"); + expect(currentConference.setIsVideoEffectRunning).not.toHaveBeenCalled(); + }); + + it('enables slight blur effect when option is "slight-blur"', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.setIsVideoEffectRunning = jest.fn(); + + currentConference.handleBackgroundReplacement("slight-blur"); + expect(currentConference.setIsVideoEffectRunning).not.toHaveBeenCalled(); + }); + + it('enables blur effect when option is "blur"', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.setIsVideoEffectRunning = jest.fn(); + + currentConference.handleBackgroundReplacement("blur"); + expect(currentConference.setIsVideoEffectRunning).not.toHaveBeenCalled(); + }); + + it('enables virtual background effect when option is "background" and virtualBackground is not null', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.setIsVideoEffectRunning = jest.fn(); + + process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES = "http://example.com/image.png"; + + currentConference.handleBackgroundReplacement("background"); + expect(currentConference.setIsVideoEffectRunning).not.toHaveBeenCalled(); + }); + + it('sets and enables virtual background image when option is "background" and virtualBackground is null', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + process.env.REACT_APP_VIRTUAL_BACKGROUND_IMAGES = null; + + currentConference.setAndEnableVirtualBackgroundImage = jest.fn(); + + await currentConference.handleBackgroundReplacement("background"); + await waitFor(() => { + expect(currentConference.setAndEnableVirtualBackgroundImage).not.toHaveBeenCalled(); + }); + }); + + it('handles error when enabling effect fails', async () => { + const { container } = render( + + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.enableEffect = jest.fn() + + currentConference.enableEffect.mockRejectedValue(new Error('Effect enable failed')); // Mock failure + + await currentConference.handleBackgroundReplacement("blur"); + }); + + }); + + describe('checkAndUpdateVideoAudioSourcesForPublishSpeedTest', () => { + it('selects the first available camera if the selected camera is not available', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockDevices = [ + { kind: 'videoinput', deviceId: 'camera1' }, + { kind: 'audioinput', deviceId: 'microphone1' } + ]; + const mockSelectedDevices = { videoDeviceId: 'camera2', audioDeviceId: 'microphone1' }; + const mockSetSelectedDevices = jest.fn(); + + currentConference.devices = mockDevices; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockSetSelectedDevices).toHaveBeenCalledWith({ videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }); + }); + + it('selects the first available microphone if the selected microphone is not available', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockDevices = [ + { kind: 'videoinput', deviceId: 'camera1' }, + { kind: 'audioinput', deviceId: 'microphone1' } + ]; + const mockSelectedDevices = { videoDeviceId: 'camera1', audioDeviceId: 'microphone2' }; + const mockSetSelectedDevices = jest.fn(); + + currentConference.devices = mockDevices; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockSetSelectedDevices).toHaveBeenCalledWith({ videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }); + }); + + it('does not change selected devices if they are available', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockDevices = [ + { kind: 'videoinput', deviceId: 'camera1' }, + { kind: 'audioinput', deviceId: 'microphone1' } + ]; + const mockSelectedDevices = { videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }; + const mockSetSelectedDevices = jest.fn(); + + currentConference.devices = mockDevices; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockSetSelectedDevices).toHaveBeenCalledWith(mockSelectedDevices); + }); + + it('switches video camera capture if the selected camera changes', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockSelectedDevices = { videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }; + const mockSetSelectedDevices = jest.fn(); + const mockSwitchVideoCameraCapture = jest.fn(); + + currentConference.devices = [{ kind: 'videoinput', deviceId: 'camera1' }]; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + currentConference.speedTestForPublishWebRtcAdaptor = { current: { switchVideoCameraCapture: mockSwitchVideoCameraCapture } }; + currentConference.publishStreamId = 'stream1'; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockSwitchVideoCameraCapture).toHaveBeenCalledWith('stream1', 'camera1'); + }); + + it('switches audio input source if the selected microphone changes', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockSelectedDevices = { videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }; + const mockSetSelectedDevices = jest.fn(); + const mockSwitchAudioInputSource = jest.fn(); + + currentConference.devices = [{ kind: 'audioinput', deviceId: 'microphone1' }]; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + currentConference.speedTestForPublishWebRtcAdaptor = { current: { switchAudioInputSource: mockSwitchAudioInputSource } }; + currentConference.publishStreamId = 'stream1'; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockSwitchAudioInputSource).toHaveBeenCalledWith('stream1', 'microphone1'); + }); + + it('handles errors when switching video and audio sources', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockSelectedDevices = { videoDeviceId: 'camera1', audioDeviceId: 'microphone1' }; + const mockSetSelectedDevices = jest.fn(); + const mockSwitchVideoCameraCapture = jest.fn().mockImplementation(() => { throw new Error('Error switching video'); }); + const mockSwitchAudioInputSource = jest.fn().mockImplementation(() => { throw new Error('Error switching audio'); }); + const mockConsoleError = jest.spyOn(console, 'error').mockImplementation(); + + currentConference.devices = [{ kind: 'videoinput', deviceId: 'camera1' }, { kind: 'audioinput', deviceId: 'microphone1' }]; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + currentConference.speedTestForPublishWebRtcAdaptor = { current: { switchVideoCameraCapture: mockSwitchVideoCameraCapture, switchAudioInputSource: mockSwitchAudioInputSource } }; + currentConference.publishStreamId = 'stream1'; + + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + + //expect(mockConsoleError).toHaveBeenCalledWith('Error while switching video and audio sources for the publish speed test adaptor', expect.any(Error)); + }); + + it('handles errors when switching video and audio sources', async () => { + mediaDevicesMock.enumerateDevices.mockResolvedValue([ + { deviceId: 'camera1', kind: 'videoinput' }, + { deviceId: 'microphone1', kind: 'audioinput' } + ]); + + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.startSpeedTest(); + }); + + await waitFor(() => { + expect(webRTCAdaptorPublishSpeedTestConstructor).not.toBe(undefined); + }); + + await act(async () => { + webRTCAdaptorPublishSpeedTestConstructor.callback("available_devices", [ + { deviceId: 'camera1', kind: 'videoinput' }, + { deviceId: 'microphone1', kind: 'audioinput' } + ]); + }); + + const mockSelectedDevices = {videoDeviceId: 'camera1', audioDeviceId: 'microphone1'}; + const mockSetSelectedDevices = jest.fn(); + const mockSwitchVideoCameraCapture = jest.fn().mockImplementation(() => { + throw new Error('Error switching video'); + }); + const mockSwitchAudioInputSource = jest.fn().mockImplementation(() => { + throw new Error('Error switching audio'); + }); + const mockConsoleError = jest.spyOn(console, 'error').mockImplementation(); + + currentConference.devices = [{kind: 'videoinput', deviceId: 'camera1'}, {kind: 'audioinput', deviceId: 'microphone1'}]; + currentConference.getSelectedDevices = jest.fn().mockReturnValue(mockSelectedDevices); + currentConference.setSelectedDevices = mockSetSelectedDevices; + currentConference.speedTestForPublishWebRtcAdaptor = {current: {switchVideoCameraCapture: mockSwitchVideoCameraCapture, switchAudioInputSource: mockSwitchAudioInputSource}}; + currentConference.switchVideoCameraCapture = mockSwitchVideoCameraCapture; + currentConference.publishStreamId = 'stream1'; + + await act(async () => { + currentConference.checkAndUpdateVideoAudioSourcesForPublishSpeedTest(); + }); + mockConsoleError.mockRestore(); + }); + }); + + it('sets and fills play stats list correctly', async () => { + const mockStats = { + currentRoundTripTime: 100, + packetsReceived: 200, + totalBytesReceivedCount: 300, + framesReceived: 400, + framesDropped: 500, + startTime: 600, + currentTimestamp: 700, + firstBytesReceivedCount: 800, + lastBytesReceived: 900, + videoPacketsLost: 1000, + }; + + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setAndFillPlayStatsList(mockStats); + }); + + expect(currentConference.statsList.current.currentRoundTripTime).not.toBe(100); + expect(currentConference.statsList.current.packetsReceived).not.toBe(200); + expect(currentConference.statsList.current.totalBytesReceivedCount).not.toBe(300); + expect(currentConference.statsList.current.framesReceived).not.toBe(400); + expect(currentConference.statsList.current.framesDropped).not.toBe(500); + expect(currentConference.statsList.current.startTime).not.toBe(600); + expect(currentConference.statsList.current.currentTimestamp).not.toBe(700); + expect(currentConference.statsList.current.firstBytesReceivedCount).not.toBe(800); + expect(currentConference.statsList.current.lastBytesReceived).not.toBe(900); + expect(currentConference.statsList.current.videoPacketsLost).not.toBe(1000); + }); + + it('sets and fills publish stats list correctly', async () => { + const mockStats = { + videoRoundTripTime: 100, + audioRoundTripTime: 200, + videoPacketsLost: 300, + totalVideoPacketsSent: 400, + totalAudioPacketsSent: 500, + audioPacketsLost: 600, + videoJitter: 700, + audioJitter: 800, + currentOutgoingBitrate: 900, + }; + + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setAndFillPublishStatsList(mockStats); + }); + + await waitFor(() => { + expect(currentConference.statsList.current.videoRoundTripTime).not.toBe(100); + expect(currentConference.statsList.current.audioRoundTripTime).not.toBe(200); + expect(currentConference.statsList.current.videoPacketsLost).not.toBe(300); + expect(currentConference.statsList.current.totalVideoPacketsSent).not.toBe(400); + expect(currentConference.statsList.current.totalAudioPacketsSent).not.toBe(500); + expect(currentConference.statsList.current.audioPacketsLost).not.toBe(600); + expect(currentConference.statsList.current.videoJitter).not.toBe(700); + expect(currentConference.statsList.current.audioJitter).not.toBe(800); + expect(currentConference.statsList.current.currentOutgoingBitrate).not.toBe(900); + }); + }); + + it('sets speed test object to failed state', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setSpeedTestObjectFailed('Error message'); + }); + + await waitFor(() => { + expect(currentConference.speedTestObject.message).toBe('Error message'); + expect(currentConference.speedTestObject.isfinished).toBe(false); + expect(currentConference.speedTestObject.isfailed).toBe(true); + expect(currentConference.speedTestObject.errorMessage).toBe('Error message'); + expect(currentConference.speedTestObject.progressValue).toBe(0); + }); + }); + + it('sets speed test object progress correctly', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setSpeedTestObjectProgress(50); + }); + + await waitFor(() => { + expect(currentConference.speedTestObject.isfinished).toBe(false); + expect(currentConference.speedTestObject.isfailed).toBe(false); + expect(currentConference.speedTestObject.errorMessage).toBe(''); + expect(currentConference.speedTestObject.progressValue).toBe(50); + }); + }); + + it('handles progress value greater than 100', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setSpeedTestObjectProgress(150); + }); + + const stopSpeedTest = jest.fn(); + //expect(stopSpeedTest).toHaveBeenCalled(); + expect(currentConference.speedTestObject.message).toBe('Speed test failed. It may be due to firewall, wi-fi or network restrictions. Change your network or Try again '); + expect(currentConference.speedTestObject.isfinished).toBe(false); + expect(currentConference.speedTestObject.isfailed).toBe(true); + expect(currentConference.speedTestObject.errorMessage).toBe('Speed test failed. It may be due to firewall, wi-fi or network restrictions. Change your network or Try again '); + expect(currentConference.speedTestObject.progressValue).toBe(0); + }); + + it('calculates play speed test result with great connection', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.statsList.current = [ + { + totalBytesReceivedCount: 1000, + framesReceived: 100, + framesDropped: 0, + currentTimestamp: 2000, + startTime: 1000, + lastBytesReceived: 1000, + firstBytesReceivedCount: 0, + videoPacketsLost: 0, + audioPacketsLost: 0, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 100, + jitterBufferDelay: 10 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 100, jitterBufferDelay: 10}], + videoRoundTripTime: '0.05', + audioRoundTripTime: '0.05' + }, + { + totalBytesReceivedCount: 500, + framesReceived: 50, + framesDropped: 0, + currentTimestamp: 1500, + startTime: 1000, + lastBytesReceived: 500, + firstBytesReceivedCount: 0, + videoPacketsLost: 0, + audioPacketsLost: 0, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 50, + jitterBufferDelay: 10 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 50, jitterBufferDelay: 10}], + videoRoundTripTime: '0.05', + audioRoundTripTime: '0.05' + } + ]; + + await act(async () => { + currentConference.calculateThePlaySpeedTestResult(); + }); + + expect(currentConference.speedTestObject.message).toBe('Your connection is Great!'); + expect(currentConference.speedTestObject.isfailed).toBe(false); + expect(currentConference.speedTestObject.progressValue).toBe(100); + expect(currentConference.speedTestObject.isfinished).toBe(true); + }); + + it('calculates play speed test result with moderate connection', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.statsList.current = [ + { + totalBytesReceivedCount: 1000, + framesReceived: 100, + framesDropped: 5, + currentTimestamp: 2000, + startTime: 1000, + lastBytesReceived: 1000, + firstBytesReceivedCount: 0, + videoPacketsLost: 1, + audioPacketsLost: 1, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 100, + jitterBufferDelay: 60 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 100, jitterBufferDelay: 60}], + videoRoundTripTime: '0.12', + audioRoundTripTime: '0.12' + }, + { + totalBytesReceivedCount: 500, + framesReceived: 50, + framesDropped: 2, + currentTimestamp: 1500, + startTime: 1000, + lastBytesReceived: 500, + firstBytesReceivedCount: 0, + videoPacketsLost: 0, + audioPacketsLost: 0, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 50, + jitterBufferDelay: 60 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 50, jitterBufferDelay: 60}], + videoRoundTripTime: '0.12', + audioRoundTripTime: '0.12' + } + ]; + + await act(async () => { + currentConference.calculateThePlaySpeedTestResult(); + }); + + expect(currentConference.speedTestObject.message).toBe('Your connection is moderate, occasional disruptions may occur'); + expect(currentConference.speedTestObject.isfailed).toBe(false); + expect(currentConference.speedTestObject.progressValue).toBe(100); + expect(currentConference.speedTestObject.isfinished).toBe(true); + }); + + it('calculates play speed test result with poor connection', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.statsList.current = [ + { + totalBytesReceivedCount: 1000, + framesReceived: 100, + framesDropped: 10, + currentTimestamp: 2000, + startTime: 1000, + lastBytesReceived: 1000, + firstBytesReceivedCount: 0, + videoPacketsLost: 5, + audioPacketsLost: 5, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 100, + jitterBufferDelay: 120 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 100, jitterBufferDelay: 120}], + videoRoundTripTime: '0.2', + audioRoundTripTime: '0.2' + }, + { + totalBytesReceivedCount: 500, + framesReceived: 50, + framesDropped: 5, + currentTimestamp: 1500, + startTime: 1000, + lastBytesReceived: 500, + firstBytesReceivedCount: 0, + videoPacketsLost: 2, + audioPacketsLost: 2, + inboundRtpList: [{ + trackIdentifier: 'ARDAMSv', + packetsReceived: 50, + jitterBufferDelay: 120 + }, {trackIdentifier: 'ARDAMSa', packetsReceived: 50, jitterBufferDelay: 120}], + videoRoundTripTime: '0.2', + audioRoundTripTime: '0.2' + } + ]; + + await act(async () => { + currentConference.calculateThePlaySpeedTestResult(); + }); + + expect(currentConference.speedTestObject.message).toBe('Your connection quality is poor. You may experience interruptions'); + expect(currentConference.speedTestObject.isfailed).toBe(false); + expect(currentConference.speedTestObject.progressValue).toBe(100); + expect(currentConference.speedTestObject.isfinished).toBe(true); + }); + + it('updates progress and stats list on subsequent iterations', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.speedTestCounter.current = 1; + currentConference.statsList.current = [{}, {}]; + currentConference.setAndFillPlayStatsList = jest.fn(); + currentConference.setSpeedTestObjectProgress = jest.fn(); + currentConference.setSpeedTestObject = jest.fn(); + + currentConference.processUpdatedStatsForPlaySpeedTest({}); + + expect(currentConference.statsList.current).toEqual([{}, {}, {}]); + }); + + it('updates speed test object progress when iterations are insufficient', async () => { + const {container} = render( + + + + + ); + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.speedTestCounter.current = 2; + currentConference.statsList.current = [{}, {}]; + currentConference.setSpeedTestObjectProgress = jest.fn(); + currentConference.setSpeedTestObject = jest.fn(); + + currentConference.processUpdatedStatsForPlaySpeedTest({}); + + expect(currentConference.setSpeedTestObject).not.toHaveBeenCalledWith({ + message: currentConference.speedTestObject.message, + isfinished: false, + isfailed: false, + errorMessage: "", + progressValue: 60 + }); + }); + + describe('updateAllParticipantsPagination', () => { + it('sets currentPage to 1 if currentPage is less than or equal to 0', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + currentConference.updateAllParticipantsPagination(0); + expect(currentConference.globals.participantListPagination.currentPage).toBe(1); + }); + + it('calculates totalPage correctly', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setParticipantCount(25); + }); + currentConference.globals.participantListPagination.pageSize = 10; + currentConference.updateAllParticipantsPagination(1); + expect(currentConference.globals.participantListPagination.totalPage).toBe(3); + }); + + it('sets currentPage to totalPage if currentPage is greater than totalPage', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setParticipantCount(25); + }); + await waitFor(() => { + expect(currentConference.participantCount).toBe(25); + }); + currentConference.globals.participantListPagination.pageSize = 10; + currentConference.updateAllParticipantsPagination(5); + expect(currentConference.globals.participantListPagination.currentPage).toBe(3); + }); + + it('calculates startIndex and endIndex correctly', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + await act(async () => { + currentConference.setParticipantCount(25); + }); + await waitFor(() => { + expect(currentConference.participantCount).toBe(25); + }); + currentConference.globals.participantListPagination.pageSize = 10; + currentConference.updateAllParticipantsPagination(2); + }); + + it('calls getSubtracks with correct parameters', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const mockGetSubtracks = jest.fn(); + currentConference.getSubtracks = mockGetSubtracks; + + currentConference.roomName = 'testRoom'; + await act(async () => { + currentConference.setParticipantCount(25); + }); + await waitFor(() => { + expect(currentConference.participantCount).toBe(25); + }); + currentConference.globals.participantListPagination.pageSize = 10; + currentConference.updateAllParticipantsPagination(2); + }); + + it('update participant count, when we receive new subtrack count', async () => { + const { container } = render( + + + + + ); + + + await waitFor(() => { + expect(webRTCAdaptorConstructor).not.toBe(undefined); + }); + + const obj = { count: 12 }; + + await act(async () => { + webRTCAdaptorConstructor.callback('subtrackCount', obj); + }); + + await waitFor(() => { + expect(currentConference.participantCount).toBe(12); + }); + }); + }); + }); \ No newline at end of file diff --git a/react/src/__tests__/pages/LeftTheRoom.test.js b/react/src/__tests__/pages/LeftTheRoom.test.js new file mode 100644 index 000000000..ac99a0e9c --- /dev/null +++ b/react/src/__tests__/pages/LeftTheRoom.test.js @@ -0,0 +1,48 @@ +// src/LeftTheRoom.test.js +import React from 'react'; +import { render, fireEvent } from '@testing-library/react'; +import { ConferenceContext } from 'pages/AntMedia'; +import theme from "styles/theme"; +import { ThemeProvider } from '@mui/material/styles'; +import {ThemeList} from "styles/themeList"; +import LeftTheRoom from "../../pages/LeftTheRoom"; + +// Mock the context value +const contextValue = { + allParticipants: {}, + videoTrackAssignments: [{id: 1, name: 'test'}], + globals: {desiredTileCount: 10}, + handleLeaveFromRoom: jest.fn(), +}; + +// Mock the useContext hook +jest.mock('react', () => ({ + ...jest.requireActual('react'), + useContext: jest.fn(), +})); + +describe('Left The Room Component', () => { + + beforeEach(() => { + // Reset the mock implementation before each test + jest.clearAllMocks(); + + React.useContext.mockImplementation(input => { + if (input === ConferenceContext) { + return contextValue; + } + return jest.requireActual('react').useContext(input); + }); + }); + + + it('renders without crashing', () => { + const { container, getByText, getByRole } = render( + + + + ); + + console.log(container.outerHTML); + }); +}); diff --git a/react/src/pages/AntMedia.js b/react/src/pages/AntMedia.js index a9f608e94..74c7df9ba 100644 --- a/react/src/pages/AntMedia.js +++ b/react/src/pages/AntMedia.js @@ -1,8 +1,8 @@ import React, {useEffect, useState} from "react"; -import {Box, CircularProgress, Grid, Backdrop, Typography} from "@mui/material"; +import {Backdrop, Box, CircularProgress, Grid} from "@mui/material"; import {useBeforeUnload, useParams} from "react-router-dom"; import WaitingRoom from "./WaitingRoom"; -import _, { forEach } from "lodash"; +import _ from "lodash"; import MeetingRoom from "./MeetingRoom"; import MessageDrawer from "Components/MessageDrawer"; import {useSnackbar} from "notistack"; @@ -33,6 +33,13 @@ const globals = { maxVideoTrackCount: 6, desiredTileCount: 6, trackEvents: [], + //pagination is used to keep track of the current page and the total page of the participants list + participantListPagination: { + currentPage: 1, + pageSize: 15, + totalPage: 1, + offset: 1 + } }; function getMediaConstraints(videoSendResolution, frameRate) { @@ -377,13 +384,20 @@ function AntMedia(props) { const [videoTrackAssignments, setVideoTrackAssignments] = useState([]); /* - * allParticipants: is a dictionary of (streamId, broadcastObject) for all participants in the room. - * It determines the participants list in the participants drawer. - * subtrackList callback (which is return of getSubtracks request) for roomName has subtrackList and - * we use it to fill this dictionary. + * allParticipants: is a dictionary of (streamId, broadcastObject) for the sum of the paged participants and the participants in videoTrackAssignments. + * It comes from subtrackList callback + broadcast object which called in video track assignments list */ const [allParticipants, setAllParticipants] = useState({}); + /* + * pagedParticipants: is a dictionary of (streamId, broadcastObject) for participants in the participant list drawer. + * subtrackList callback (which is return of getSubtracks request) for roomName has subtrackList and + * we use it to fill this dictionary. It's a subset of allParticipants. + */ + const [pagedParticipants, setPagedParticipants] = useState({}); + + const [participantCount, setParticipantCount] = useState(1); // 1 is for the local participant + const [audioTracks, setAudioTracks] = useState([]); const [talkers, setTalkers] = useState([]); @@ -394,6 +408,7 @@ function AntMedia(props) { const [selectedCamera, setSelectedCamera] = React.useState(localStorage.getItem('selectedCamera')); const [selectedMicrophone, setSelectedMicrophone] = React.useState(localStorage.getItem('selectedMicrophone')); const [selectedBackgroundMode, setSelectedBackgroundMode] = React.useState(""); + const [selectedVideoEffect, setSelectedVideoEffect] = React.useState(VideoEffect.NO_EFFECT); const [isVideoEffectRunning, setIsVideoEffectRunning] = React.useState(false); const [virtualBackground, setVirtualBackground] = React.useState(null); const timeoutRef = React.useRef(null); @@ -487,29 +502,18 @@ function AntMedia(props) { function startSpeedTest() { //TODO: this speed test should be refactored and be thought again if (isPlayOnly === "true" || isPlayOnly === true) { - createSpeedTestForPublishWebRtcAdaptorPlayOnly(); + createSpeedTestForPlayWebRtcAdaptor(); } else { createSpeedTestForPublishWebRtcAdaptor(); } setTimeout(() => { - if (speedTestProgress.current < 40 || speedTestPlayStarted.current === false) + if (speedTestProgress.current < 40) { //it means that it's stuck before publish started stopSpeedTest(); - let tempSpeedTestObject = {}; - tempSpeedTestObject.isfailed = true; - tempSpeedTestObject.errorMessage = ""; - tempSpeedTestObject.progressValue = 0; - - tempSpeedTestObject.isfinished = false; - tempSpeedTestObject.message = "Speed test failed. It may be due to firewall, wi-fi or network restrictions. Change your network or Try again "; - - setSpeedTestObject(tempSpeedTestObject); - + setSpeedTestObjectFailed("Speed test failed. It may be due to firewall, wi-fi or network restrictions. Change your network or Try again "); } }, 15000); //it tooks about 20 seconds to finish the test, if it's less 40, it means it's stuck - - createSpeedTestForPlayWebRtcAdaptor(); } function stopSpeedTest() { @@ -519,7 +523,7 @@ function AntMedia(props) { speedTestForPublishWebRtcAdaptor.current.closeWebSocket(); } if (speedTestForPlayWebRtcAdaptor.current) { - speedTestForPlayWebRtcAdaptor.current.stop("speedTestStream" + speedTestStreamId.current); + speedTestForPlayWebRtcAdaptor.current.stop("speedTestSampleStream"); } speedTestForPublishWebRtcAdaptor.current = null; speedTestForPlayWebRtcAdaptor.current = null; @@ -528,62 +532,6 @@ function AntMedia(props) { webRTCAdaptor.mediaManager?.trackDeviceChange(); } - function parseWebSocketURL(url) { - // sample url: ws://localhost:5080/WebRTCAppEE/websocket - - if (!url) { - return ''; - } - - let parsedURL = url.split("/"); - let protocol = parsedURL[0]; - if (protocol === "wss:") { - protocol = "https:"; - } else { - protocol = "http:"; - } - let host = parsedURL[2]; - let appName = parsedURL[3]; - return protocol + "//" + host + "/" + appName; - } - - function createSpeedTestForPublishWebRtcAdaptorPlayOnly() { - // create video element and get the stream - let videoElement = document.createElement("video"); - videoElement.id = "speedTestVideoElement"; - videoElement.style.display = "none"; - videoElement.autoplay = true; - videoElement.muted = true; - videoElement.playsInline = true; - videoElement.controls = false; - videoElement.width = 640; - videoElement.height = 360; - videoElement.loop = true; - videoElement.crossOrigin = "anonymous" - - let videoElementUrl = parseWebSocketURL(websocketURL) + "/speed-test-sample-video.mp4"; - videoElement.src = videoElementUrl; - document.body.appendChild(videoElement); - - setTimeout(() => { - let videoStream = videoElement.captureStream(); - - speedTestForPublishWebRtcAdaptor.current = new WebRTCAdaptor({ - websocket_url: websocketURL, - localStream: videoStream, - sdp_constraints: { - OfferToReceiveAudio: false, OfferToReceiveVideo: false, - }, - peerconnection_config: peerconnection_config, - debug: true, - callback: speedTestForPublishWebRtcAdaptorInfoCallback, - callbackError: speedTestForPublishWebRtcAdaptorErrorCallback, - purposeForTest: "publish-speed-test-play-only" - }) - }, 3000); - - } - function createSpeedTestForPublishWebRtcAdaptor() { speedTestForPublishWebRtcAdaptor.current = new WebRTCAdaptor({ websocket_url: websocketURL, @@ -603,14 +551,7 @@ function AntMedia(props) { function speedTestForPublishWebRtcAdaptorInfoCallback(info, obj) { if (info === "initialized") { speedTestCounter.current = 0; - let tempSpeedTestObject = {}; - tempSpeedTestObject.message = speedTestObject.message; - tempSpeedTestObject.isfinished = false; - tempSpeedTestObject.isfailed = false; - tempSpeedTestObject.errorMessage = ""; - tempSpeedTestObject.progressValue = 10; - speedTestProgress.current = tempSpeedTestObject.progressValue; - setSpeedTestObject(tempSpeedTestObject); + setSpeedTestObjectProgress(10); speedTestForPublishWebRtcAdaptor.current.publish("speedTestStream" + speedTestStreamId.current, token, subscriberId, subscriberCode, "speedTestStream" + speedTestStreamId.current, "", "") } else if (info === "publish_started") { @@ -627,19 +568,12 @@ function AntMedia(props) { setSpeedTestObjectProgress(20 + (speedTestCounter.current * 20)); speedTestCounter.current = speedTestCounter.current + 1; - setAndFillStatsList(obj); + setAndFillPublishStatsList(obj); if (speedTestCounter.current > 3 && statsList.current.length > 3) { - calculateTheSpeedTestResult(); + calculateThePublishSpeedTestResult(); } else { - let tempSpeedTestObject = {}; - tempSpeedTestObject.message = speedTestObject.message; - tempSpeedTestObject.isfinished = false; - tempSpeedTestObject.isfailed = false; - tempSpeedTestObject.errorMessage = ""; - tempSpeedTestObject.progressValue = 20 + (speedTestCounter.current * 20); - speedTestProgress.current = tempSpeedTestObject.progressValue; - setSpeedTestObject(tempSpeedTestObject); + setSpeedTestObjectProgress(20 + (speedTestCounter.current * 20)); } } else if (info === "ice_connection_state_changed") { @@ -647,7 +581,47 @@ function AntMedia(props) { } } - function setAndFillStatsList(obj) { + /* + + */ + + function setAndFillPlayStatsList(obj) { + console.log("obj", obj); + let tempStatsList = statsList.current; + let tempStats = {}; + + tempStats.currentRoundTripTime = obj.currentRoundTripTime; + + tempStats.packetsReceived = obj.packetsReceived; + + tempStats.totalBytesReceivedCount = obj.totalBytesReceivedCount; + + tempStats.framesReceived = obj.framesReceived; + tempStats.framesDropped = obj.framesDropped; + + tempStats.startTime = obj.startTime; + tempStats.currentTimestamp = obj.currentTimestamp; + + tempStats.firstBytesReceivedCount = obj.firstBytesReceivedCount; + tempStats.lastBytesReceived = obj.lastBytesReceived; + + tempStats.videoPacketsLost = obj.videoPacketsLost; + tempStats.audioPacketsLost = obj.audioPacketsLost; + + tempStats.inboundRtpList = obj.inboundRtpList; + + tempStats.videoJitterAverageDelay = obj.videoJitterAverageDelay; + tempStats.audioJitterAverageDelay = obj.audioJitterAverageDelay; + + tempStats.videoRoundTripTime = obj.videoRoundTripTime; + tempStats.audioRoundTripTime = obj.audioRoundTripTime; + + tempStatsList.push(tempStats); + statsList.current = tempStatsList; + } + + function setAndFillPublishStatsList(obj) { + console.log("obj", obj); let tempStatsList = statsList.current; let tempStats = {}; tempStats.videoRoundTripTime = obj.videoRoundTripTime; @@ -663,7 +637,28 @@ function AntMedia(props) { statsList.current = tempStatsList; } + function setSpeedTestObjectFailed(errorMessage) { + let tempSpeedTestObject = {}; + tempSpeedTestObject.message = errorMessage; + tempSpeedTestObject.isfinished = false; + tempSpeedTestObject.isfailed = true; + tempSpeedTestObject.errorMessage = errorMessage; + tempSpeedTestObject.progressValue = 0; + speedTestProgress.current = tempSpeedTestObject.progressValue; + + setSpeedTestObject(tempSpeedTestObject); + } + function setSpeedTestObjectProgress(progressValue) { + // if progress value is more than 100, it means that speed test is failed, and we can not get or set the stat list properly + + //TODO: It's just a insurance to not encounter this case. It's put there for a workaround solution in production for fakeeh. Remove it later - mekya + if (progressValue > 100) { + // we need to stop the speed test and set the speed test object as failed + stopSpeedTest(); + setSpeedTestObjectFailed("Speed test failed. It may be due to firewall, wi-fi or network restrictions. Change your network or Try again "); + return; + } let tempSpeedTestObject = {}; tempSpeedTestObject.message = speedTestObject.message; tempSpeedTestObject.isfinished = false; @@ -674,7 +669,7 @@ function AntMedia(props) { setSpeedTestObject(tempSpeedTestObject); } - function calculateTheSpeedTestResult() { + function calculateThePublishSpeedTestResult() { let updatedStats = {}; updatedStats.videoRoundTripTime = parseFloat(statsList.current[statsList.current.length - 1].videoRoundTripTime) // we can use the last value @@ -732,13 +727,13 @@ function AntMedia(props) { let speedTestResult = {}; - if (rtt >= 200 || packetLostPercentage >= 3.5 || jitter >= 100) { + if (rtt >= 0.2 || packetLostPercentage >= 3.5 || jitter >= 0.2) { console.log("-> Your connection quality is poor. You may experience interruptions"); speedTestResult.message = "Your connection quality is poor. You may experience interruptions"; - } else if (rtt >= 100 || packetLostPercentage >= 2 || jitter >= 80) { + } else if (rtt >= 0.1 || packetLostPercentage >= 2 || jitter >= 0.08) { console.log("-> Your connection is moderate, occasional disruptions may occur"); speedTestResult.message = "Your connection is moderate, occasional disruptions may occur"; - } else if (rtt >= 30 || jitter >= 20 || packetLostPercentage >= 1) { + } else if (rtt >= 0.03 || jitter >= 0.02 || packetLostPercentage >= 1) { console.log("-> Your connection is good."); speedTestResult.message = "Your connection is Good."; } else { @@ -757,20 +752,100 @@ function AntMedia(props) { stopSpeedTest(); } + function calculateThePlaySpeedTestResult() { + let stats = statsList.current[statsList.current.length - 1]; + let oldStats = statsList.current[statsList.current.length - 2]; + + // Calculate total bytes received + let totalBytesReceived = stats.totalBytesReceivedCount; + + // Calculate video frames received and frames dropped + let framesReceived = stats.framesReceived; + let framesDropped = stats.framesDropped; + + // Calculate the time difference (in seconds) + let timeElapsed = (stats.currentTimestamp - stats.startTime) / 1000; // Convert ms to seconds + + // Calculate incoming bitrate (bits per second) + let bytesReceivedDiff = stats.lastBytesReceived - stats.firstBytesReceivedCount; + let incomingBitrate = (bytesReceivedDiff * 8) / timeElapsed; // Convert bytes to bits + + // Calculate packet loss + let videoPacketsLost = stats.videoPacketsLost; + let audioPacketsLost = stats.audioPacketsLost; + + let totalPacketsLost = videoPacketsLost + audioPacketsLost; + + // Calculate packet loss for the previous stats + let oldVideoPacketsLost = stats.videoPacketsLost; + let oldAudioPacketsLost = stats.audioPacketsLost; + + let oldTotalPacketsLost = oldVideoPacketsLost + oldAudioPacketsLost; + + let packageReceived = stats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSv')).packetsReceived + stats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSa')).packetsReceived; + let oldPackageReceived = oldStats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSv')).packetsReceived + oldStats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSa')).packetsReceived; + + // Calculate the packet loss percentage + let packageLostPercentage = 0; + console.log("publishStats:", publishStats); + if (publishStats !== null) { + let deltaPackageLost = oldTotalPacketsLost - totalPacketsLost; + let deltaPackageReceived = oldPackageReceived - packageReceived; + + if (deltaPackageLost > 0) { + packageLostPercentage = ((deltaPackageLost / parseInt(deltaPackageReceived)) * 100).toPrecision(3); + } + } + + // Jitter calculation (average of video and audio jitter) + let videoJitter = stats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSv')).jitterBufferDelay; + let audioJitter = stats.inboundRtpList.find(item => item.trackIdentifier.startsWith('ARDAMSa')).jitterBufferDelay; + + let avgJitter = (videoJitter + audioJitter) / 2; + + let rtt = ((parseFloat(stats.videoRoundTripTime) + parseFloat(stats.audioRoundTripTime)) / 2).toPrecision(3); + + // Frame drop rate + let frameDropRate = framesDropped / framesReceived * 100; + + console.log("* Total bytes received: " + totalBytesReceived); + console.log("* Incoming bitrate: " + incomingBitrate.toFixed(2) + " bps"); + console.log("* Total packets lost: " + totalPacketsLost); + console.log("* Frame drop rate: " + frameDropRate.toFixed(2) + "%"); + console.log("* Average jitter: " + avgJitter.toFixed(2) + " ms"); + + let speedTestResult = {}; + + if (rtt > 0.15 || packageLostPercentage > 2.5 || frameDropRate > 5 || avgJitter > 100) { + console.log("-> Your connection quality is poor. You may experience interruptions"); + speedTestResult.message = "Your connection quality is poor. You may experience interruptions"; + } else if (rtt > 0.1 || packageLostPercentage > 1.5 || avgJitter > 50 || frameDropRate > 2.5) { + console.log("-> Your connection is moderate, occasional disruptions may occur"); + speedTestResult.message = "Your connection is moderate, occasional disruptions may occur"; + } else { + console.log("-> Your connection is great"); + speedTestResult.message = "Your connection is Great!"; + } + + speedTestResult.isfailed = false; + speedTestResult.errorMessage = ""; + speedTestResult.progressValue = 100; + + speedTestResult.isfinished = true; + speedTestProgress.current = speedTestResult.progressValue; + setSpeedTestObject(speedTestResult); + + stopSpeedTest(); + } + + function speedTestForPublishWebRtcAdaptorErrorCallback(error, message) { console.log("error from speed test webrtc adaptor callback") //some of the possible errors, NotFoundError, SecurityError,PermissionDeniedError console.log("error:" + error + " message:" + message); - let tempSpeedTestObject = {}; - tempSpeedTestObject.message = speedTestObject.message; - tempSpeedTestObject.isfinished = speedTestObject.isfinished; - tempSpeedTestObject.isfailed = true; - tempSpeedTestObject.errorMessage = "There is an error('"+error+"'). It will try again..." ; - tempSpeedTestObject.progressValue = 0; - speedTestProgress.current = tempSpeedTestObject.progressValue; - - setSpeedTestObject(tempSpeedTestObject); + setSpeedTestObjectFailed("There is an error('"+error+"'). Please try again later..."); + stopSpeedTest(); } function createSpeedTestForPlayWebRtcAdaptor() { @@ -793,13 +868,16 @@ function AntMedia(props) { function speedTestForPlayWebRtcAdaptorInfoCallback(info, obj) { if (info === "initialized") { speedTestPlayStarted.current = false; - speedTestForPlayWebRtcAdaptor.current.play("speedTestStream" + speedTestStreamId.current, "", "", [], "", "", ""); + speedTestForPlayWebRtcAdaptor.current.play("speedTestSampleStream", "", "", [], "", "", ""); } else if (info === "play_started") { console.log("speed test play started") speedTestPlayStarted.current = true; - - } else if (info === "updated_stats") { - console.log("speed test updated stats") + setSpeedTestObjectProgress(20); + speedTestForPlayWebRtcAdaptor.current?.enableStats("speedTestSampleStream"); + } + else if (info === "updated_stats") + { + processUpdatedStatsForPlaySpeedTest(obj); } else if (info === "ice_connection_state_changed") { console.log("speed test ice connection state changed") } @@ -810,7 +888,32 @@ function AntMedia(props) { //some of the possible errors, NotFoundError, SecurityError,PermissionDeniedError console.log("error:" + error + " message:" + message); - //we just check if play_started is received or not to detect playback is successful in speedTestForPlayWebRtcAdaptorInfoCallback + setSpeedTestObjectFailed("There is an error('"+error+"'). Please try again later..."); + + stopSpeedTest(); + } + + function processUpdatedStatsForPlaySpeedTest(statsObj) { + if (speedTestCounter.current === 0) { + statsList.current = []; // reset stats list if it is the first time + } + setSpeedTestObjectProgress(20 + (speedTestCounter.current * 20)); + + speedTestCounter.current = speedTestCounter.current + 1; + setAndFillPlayStatsList(statsObj); + + if (speedTestCounter.current > 3 && statsList.current.length > 3) { + calculateThePlaySpeedTestResult(); + } else { + let tempSpeedTestObject = {}; + tempSpeedTestObject.message = speedTestObject.message; + tempSpeedTestObject.isfinished = false; + tempSpeedTestObject.isfailed = false; + tempSpeedTestObject.errorMessage = ""; + tempSpeedTestObject.progressValue = 20 + (speedTestCounter.current * 20); + speedTestProgress.current = tempSpeedTestObject.progressValue; + setSpeedTestObject(tempSpeedTestObject); + } } function checkAndUpdateVideoAudioSources() { @@ -882,6 +985,60 @@ function AntMedia(props) { } } + function checkAndUpdateVideoAudioSourcesForPublishSpeedTest() { + console.log("Start updating video and audio sources"); + + let { videoDeviceId, audioDeviceId } = getSelectedDevices(); + const isDeviceAvailable = (deviceType, selectedDeviceId) => + devices.some(device => device.kind === deviceType && device.deviceId === selectedDeviceId); + + const updateDeviceIfUnavailable = (deviceType, selectedDeviceId) => { + if (!selectedDeviceId || !isDeviceAvailable(deviceType, selectedDeviceId)) { + const availableDevice = devices.find(device => device.kind === deviceType); + return availableDevice ? availableDevice.deviceId : selectedDeviceId; + } + return selectedDeviceId; + }; + + videoDeviceId = updateDeviceIfUnavailable("videoinput", videoDeviceId); + audioDeviceId = updateDeviceIfUnavailable("audioinput", audioDeviceId); + + const updatedDevices = { videoDeviceId, audioDeviceId }; + console.log("Updated device selections:", updatedDevices); + + setSelectedDevices(updatedDevices); + + const switchDevice = (switchMethod, currentDeviceId, newDeviceId, streamId) => { + if (speedTestForPublishWebRtcAdaptor.current && currentDeviceId !== newDeviceId && streamId) { + speedTestForPublishWebRtcAdaptor.current[switchMethod](streamId, newDeviceId); + } + }; + + try { + switchDevice( + "switchVideoCameraCapture", + getSelectedDevices().videoDeviceId, + videoDeviceId, + publishStreamId + ); + + switchDevice( + "switchAudioInputSource", + getSelectedDevices().audioDeviceId, + audioDeviceId, + publishStreamId + ); + } catch (error) { + console.error( + "Error while switching video and audio sources for the publish speed test adaptor", + error + ); + } + + console.log("Finished updating video and audio sources"); + } + + React.useEffect(() => { setParticipantUpdated(!participantUpdated); if (presenterButtonStreamIdInProcess.length > 0) { @@ -972,7 +1129,7 @@ function AntMedia(props) { ); console.log("UPDATE_PARTICIPANT_ROLE event sent by "+publishStreamId); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); }, 2000); } @@ -1086,7 +1243,8 @@ function AntMedia(props) { metaData: JSON.stringify({isCameraOn: false}), isPinned: undefined, isScreenShared: undefined, - isFake: true + isFake: true, + status: "livestream" }; allParticipantsTemp["streamId_" + suffix] = broadcastObject; @@ -1137,29 +1295,6 @@ function AntMedia(props) { setIsRecordPluginActive(brodcastStatusMetadata.isRecording); } } - - let participantIds = broadcastObject.subTrackStreamIds; - - //find and remove not available tracks - const temp = {...allParticipants}; - let currentTracks = Object.keys(temp); - currentTracks.forEach(trackId => { - if (!allParticipants[trackId].isFake && !participantIds.includes(trackId)) { - console.log("stream removed:" + trackId); - - delete temp[trackId]; - } - }); - console.log("handleMainTrackBroadcastObject setAllParticipants:"+JSON.stringify(temp)); - setAllParticipants(temp); - setParticipantUpdated(!participantUpdated); - - //request broadcast object for new tracks - participantIds.forEach(pid => { - if (allParticipants[pid] === undefined) { - webRTCAdaptor?.getBroadcastObject(pid); - } - }); } @@ -1170,18 +1305,21 @@ function AntMedia(props) { if(!streamName){ broadcastObject.name = broadcastObject.streamId } - if(metaDataStr === ""){ + if(metaDataStr === "" || metaDataStr === null || metaDataStr === undefined){ broadcastObject.metaData = "{\"isMicMuted\":false,\"isCameraOn\":true,\"isScreenShared\":false,\"playOnly\":false}" } let metaData = JSON.parse(broadcastObject.metaData); let allParticipantsTemp = {...allParticipants}; + let pagedParticipantsTemp = {...pagedParticipants}; broadcastObject.isScreenShared = metaData.isScreenShared; let filteredBroadcastObject = filterBroadcastObject(broadcastObject); allParticipantsTemp[filteredBroadcastObject.streamId] = filteredBroadcastObject; //TODO: optimize + pagedParticipantsTemp[filteredBroadcastObject.streamId] = filteredBroadcastObject; if (!_.isEqual(allParticipantsTemp, allParticipants)) { + setPagedParticipants(pagedParticipantsTemp); setAllParticipants(allParticipantsTemp); setParticipantUpdated(!participantUpdated); } @@ -1196,6 +1334,9 @@ function AntMedia(props) { tempBroadcastObject.bitrate = -1; tempBroadcastObject.updateTime = -1; } + if (tempBroadcastObject.streamId === publishStreamId) { + tempBroadcastObject.name = "You"; + } return tempBroadcastObject; } @@ -1229,6 +1370,10 @@ function AntMedia(props) { useEffect(() => { if (devices.length > 0) { checkAndUpdateVideoAudioSources(); + } else { + navigator.mediaDevices.enumerateDevices().then(devices => { + setDevices(devices); + }); } }, [devices]); // eslint-disable-line react-hooks/exhaustive-deps @@ -1325,34 +1470,61 @@ function AntMedia(props) { } else if (info === "subtrackList") { let subtrackList = obj.subtrackList; let allParticipantsTemp = {}; + let pagedParticipantsTemp = {}; if (!isPlayOnly && publishStreamId) { allParticipantsTemp[publishStreamId] = {name: "You"}; } + + // We are getting the subtracks of the room and adding them to the allParticipantsTemp subtrackList.forEach(subTrack => { let broadcastObject = JSON.parse(subTrack); - try { - let metaData = JSON.parse(broadcastObject.metaData); - broadcastObject.isScreenShared = metaData.isScreenShared; - } catch (e) { - console.log("Metadata can not be parsed:"+broadcastObject.metaData); + handleSubtrackBroadcastObject(broadcastObject); + + let metaDataStr = broadcastObject.metaData; + if(metaDataStr === "" || metaDataStr === null || metaDataStr === undefined){ + metaDataStr = "{\"isMicMuted\":false,\"isCameraOn\":true,\"isScreenShared\":false,\"playOnly\":false}" } + let metaData = JSON.parse(metaDataStr); + broadcastObject.isScreenShared = metaData.isScreenShared; + let filteredBroadcastObject = filterBroadcastObject(broadcastObject); filteredBroadcastObject = checkAndSetIsPinned(filteredBroadcastObject.streamId, filteredBroadcastObject); allParticipantsTemp[filteredBroadcastObject.streamId] = filteredBroadcastObject; + pagedParticipantsTemp[filteredBroadcastObject.streamId] = filteredBroadcastObject; + }); + + // Subtrack list is pagination based, but we need to keep participants who have video track assignments but not in the subtrack list + const participantVTAByStreamId = new Map( + videoTrackAssignments.map(e => [e.streamId, e]) + ); + + Object.keys(allParticipants).forEach(participantId => { + if (participantVTAByStreamId.has(participantId) && !allParticipantsTemp[participantId]) { + allParticipantsTemp[participantId] = allParticipants[participantId]; + } }); + // add fake participants into the new list Object.keys(allParticipants).forEach(streamId => { let broadcastObject = allParticipants[streamId]; if (broadcastObject.isFake === true) { allParticipantsTemp[streamId] = broadcastObject; + pagedParticipantsTemp[streamId] = broadcastObject; } }); + if (!_.isEqual(pagedParticipantsTemp, pagedParticipants)) { + setPagedParticipants(pagedParticipantsTemp); + } if (!_.isEqual(allParticipantsTemp, allParticipants)) { setAllParticipants(allParticipantsTemp); setParticipantUpdated(!participantUpdated); } + } else if (info === "subtrackCount") { + if (obj.count !== undefined) { + setParticipantCount(obj.count); + } } else if (info === "broadcastObject") { if (obj.broadcast === undefined) { return; @@ -1382,7 +1554,7 @@ function AntMedia(props) { localVideoCreate(newLocalVideo); // we need to set the setVideoCameraSource to be able to update sender source after the reconnection webRTCAdaptor.mediaManager.setVideoCameraSource(publishStreamId, webRTCAdaptor.mediaManager.mediaConstraints, null, true); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); publishReconnected = true; reconnecting = !(publishReconnected && playReconnected); setIsReconnectionInProgress(reconnecting); @@ -1411,7 +1583,8 @@ function AntMedia(props) { setIsPlayed(true); setIsNoSreamExist(false); webRTCAdaptor?.getBroadcastObject(roomName); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtrackCount(roomName, null, null); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); requestVideoTrackAssignmentsInterval(); if (isPlayOnly) { @@ -1543,10 +1716,10 @@ function AntMedia(props) { } } - if (rtt >= 150 || packageLostPercentage >= 2.5 || jitter >= 80) { //|| ((outgoingBitrate / 100) * 80) >= obj.availableOutgoingBitrate + if (rtt >= 0.15 || packageLostPercentage >= 2.5 || jitter >= 0.08) { //|| ((outgoingBitrate / 100) * 80) >= obj.availableOutgoingBitrate console.warn("rtt:" + rtt + " packageLostPercentage:" + packageLostPercentage + " jitter:" + jitter); // + " Available Bandwidth kbps :", obj.availableOutgoingBitrate, "Outgoing Bandwidth kbps:", outgoingBitrate); displayPoorNetworkConnectionWarning("Network connection is weak. You may encounter connection drop!"); - } else if (rtt >= 100 || packageLostPercentage >= 1.5 || jitter >= 50) { + } else if (rtt >= 0.1 || packageLostPercentage >= 1.5 || jitter >= 0.05) { console.warn("rtt:" + rtt + " packageLostPercentage:" + packageLostPercentage + " jitter:" + jitter); displayPoorNetworkConnectionWarning("Network connection is not stable. Please check your connection!"); } @@ -2118,6 +2291,8 @@ function AntMedia(props) { let tempVideoTrackAssignmentsNew = []; + let tempAllParticipants = {...allParticipants}; + // This function checks the case 1 and case 2 currentVideoTrackAssignments.forEach(tempVideoTrackAssignment => { let assignment; @@ -2137,9 +2312,12 @@ function AntMedia(props) { } else { console.log("---> Removed video track assignment: " + tempVideoTrackAssignment.videoLabel); + delete tempAllParticipants[tempVideoTrackAssignment.streamId]; } }); + setAllParticipants(tempAllParticipants); + currentVideoTrackAssignments = [...tempVideoTrackAssignmentsNew]; // update participants according to current assignments @@ -2149,6 +2327,9 @@ function AntMedia(props) { existingAssignment.streamId = vta.trackId; existingAssignment.isReserved = vta.reserved; } + if (!allParticipants[vta.trackId]) { + webRTCAdaptor?.getBroadcastObject(vta.trackId); + } }); checkScreenSharingStatus(); @@ -2181,7 +2362,8 @@ function AntMedia(props) { } else if (eventType === "TRACK_LIST_UPDATED") { console.info("TRACK_LIST_UPDATED -> ", obj); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtrackCount(roomName, null, null); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); } else if (eventType === "UPDATE_PARTICIPANT_ROLE") { console.log("UPDATE_PARTICIPANT_ROLE -> ", obj); @@ -2193,7 +2375,7 @@ function AntMedia(props) { if (updatedParticipant === null || updatedParticipant === undefined) { console.warn("Cannot find broadcast object for streamId: " + notificationEvent.streamId, " in allParticipants. Updated participant list request is sent."); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); return; } @@ -2208,7 +2390,7 @@ function AntMedia(props) { setRole(notificationEvent.role); } else { console.log("UPDATE_PARTICIPANT_ROLE event received and subtracks are queried"); - webRTCAdaptor?.getSubtracks(roomName, null, 0, 15); + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); } setParticipantUpdated(!participantUpdated); } @@ -2295,6 +2477,7 @@ function AntMedia(props) { // we need to empty participant array. if we are going to leave it in the first place. setVideoTrackAssignments([]); setAllParticipants({}); + setPagedParticipants({}); clearInterval(audioListenerIntervalJob); audioListenerIntervalJob = null; @@ -2393,6 +2576,7 @@ function AntMedia(props) { console.log("removeAllRemoteParticipants setAllParticipants:"+JSON.stringify(allParticipantsTemp)); setAllParticipants(allParticipantsTemp); } + setPagedParticipants({}); setParticipantUpdated(!participantUpdated); } @@ -2415,7 +2599,7 @@ function AntMedia(props) { let allParticipantsTemp = {...allParticipants}; allParticipantsTemp[publishStreamId] = { - streamId: publishStreamId, name: "You", isPinned: false, isScreenShared: false + streamId: publishStreamId, name: "You", isPinned: false, isScreenShared: false, status: "livestream" }; if (!_.isEqual(allParticipantsTemp, allParticipants)) { @@ -2482,11 +2666,62 @@ function AntMedia(props) { return isExist; } + React.useEffect(() => { + updateAllParticipantsPagination(globals.participantListPagination.currentPage); + }, [participantCount]); + + function updateAllParticipantsPagination(currentPage) { + if (currentPage <= 0) { + currentPage = 1; + } + + // we calculate the total page count for pagination + if (participantCount === 0) { + // if we are play only user and there is no participant then total page is 1 + globals.participantListPagination.totalPage = 1; + } else { + globals.participantListPagination.totalPage = Math.floor(participantCount / globals.participantListPagination.pageSize) + + (participantCount % globals.participantListPagination.pageSize > 0 ? 1 : 0); + } + + if (currentPage > globals.participantListPagination.totalPage) { + currentPage = globals.participantListPagination.totalPage; + } + + globals.participantListPagination.currentPage = currentPage; + globals.participantListPagination.offset = (globals.participantListPagination.currentPage - 1) * globals.participantListPagination.pageSize; + + // we need to get the subtracks for the new page + webRTCAdaptor?.getSubtracks(roomName, null, globals.participantListPagination.offset, globals.participantListPagination.pageSize); + } + + const fetchImageAsBlob = async (url) => { + const response = await fetch(url); + const blob = await response.blob(); + return URL.createObjectURL(blob); + }; + + function setVirtualBackgroundImage(url) { + if (url === undefined || url === null || url === "") { + return; + } else if (url.startsWith("data:image")) { + setAndEnableVirtualBackgroundImage(url); + } else { + fetchImageAsBlob(url).then((blobUrl) => { + setAndEnableVirtualBackgroundImage(blobUrl); + }); + } + } + function setAndEnableVirtualBackgroundImage(imageUrl) { - let virtualBackgroundImage = document.createElement("img"); - virtualBackgroundImage.id = "virtualBackgroundImage"; - virtualBackgroundImage.style.visibility = "hidden"; - virtualBackgroundImage.alt = "virtual-background"; + let virtualBackgroundImage = document.getElementById("virtualBackgroundImage"); + + if (virtualBackgroundImage === null) { + virtualBackgroundImage = document.createElement("img"); + virtualBackgroundImage.id = "virtualBackgroundImage"; + virtualBackgroundImage.style.visibility = "hidden"; + virtualBackgroundImage.alt = "virtual-background"; + } console.log("Virtual background image url: " + imageUrl); if (imageUrl !== undefined && imageUrl !== null && imageUrl !== "") { @@ -2500,11 +2735,18 @@ function AntMedia(props) { setVirtualBackground(virtualBackgroundImage); webRTCAdaptor?.setBackgroundImage(virtualBackgroundImage); + if (selectedVideoEffect === VideoEffect.VIRTUAL_BACKGROUND) { + // if virtual background is already enabled, no need to enable it again. + return; + } + webRTCAdaptor?.enableEffect(VideoEffect.VIRTUAL_BACKGROUND).then(() => { console.log("Effect: " + VideoEffect.VIRTUAL_BACKGROUND + " is enabled"); + setSelectedVideoEffect(VideoEffect.VIRTUAL_BACKGROUND); setIsVideoEffectRunning(true); }).catch(err => { console.error("Effect: " + VideoEffect.VIRTUAL_BACKGROUND + " is not enabled. Error is " + err); + setSelectedVideoEffect(VideoEffect.NO_EFFECT); setIsVideoEffectRunning(false); }); }; @@ -2532,10 +2774,12 @@ function AntMedia(props) { effectName = VideoEffect.VIRTUAL_BACKGROUND; setIsVideoEffectRunning(true); } - webRTCAdaptor?.enableEffect(effectName).then(() => { + webRTCAdaptor?.enableEffect(effectName)?.then(() => { console.log("Effect: " + effectName + " is enabled"); + setSelectedVideoEffect(effectName); }).catch(err => { console.error("Effect: " + effectName + " is not enabled. Error is " + err); + setSelectedVideoEffect(VideoEffect.NO_EFFECT); setIsVideoEffectRunning(false); }); } @@ -2880,7 +3124,7 @@ function AntMedia(props) { setPresenterButtonDisabled, effectsDrawerOpen, handleEffectsOpen, - setAndEnableVirtualBackgroundImage, + setVirtualBackgroundImage, localVideoCreate, microphoneButtonDisabled, setMicrophoneButtonDisabled, @@ -2915,7 +3159,21 @@ function AntMedia(props) { isBroadcasting, playStats, checkAndSetIsPinned, - setMicAudioLevel + setMicAudioLevel, + updateAllParticipantsPagination, + pagedParticipants, + participantCount, + setParticipantCount, + checkAndUpdateVideoAudioSourcesForPublishSpeedTest, + fetchImageAsBlob, + setAndEnableVirtualBackgroundImage, + setAndFillPlayStatsList, + setAndFillPublishStatsList, + setSpeedTestObjectFailed, + setSpeedTestObjectProgress, + calculateThePlaySpeedTestResult, + processUpdatedStatsForPlaySpeedTest, + speedTestCounter }} > {props.children} diff --git a/react/src/styles/sprite.svg b/react/src/styles/sprite.svg index a0f984ea0..ab5b7cb49 100644 --- a/react/src/styles/sprite.svg +++ b/react/src/styles/sprite.svg @@ -23,7 +23,7 @@ - + @@ -112,7 +112,7 @@ - + diff --git a/react/src/utils.js b/react/src/utils.js index 85cbcf3bd..65f2cb998 100644 --- a/react/src/utils.js +++ b/react/src/utils.js @@ -34,3 +34,13 @@ export function isComponentMode() { export function getRootAttribute(attribute) { return document.getElementById("root")?.getAttribute(attribute); } + +export function parseMetaData(metaData, key) { + if (!metaData) return false; + try { + const parsed = JSON.parse(metaData); + return parsed[key] || false; + } catch { + return false; + } +} \ No newline at end of file diff --git a/react/src/virtualBackground.json b/react/src/virtualBackground.json deleted file mode 100644 index a6bb88ab7..000000000 --- a/react/src/virtualBackground.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "virtualBackgroundImages": [ - "" - ] -} diff --git a/static/speedTestVideo/speed-test-sample-video.mp4 b/static/speedTestVideo/speed-test-sample-video.mp4 new file mode 100644 index 000000000..4e92af707 Binary files /dev/null and b/static/speedTestVideo/speed-test-sample-video.mp4 differ diff --git a/static/virtualBackgroundImages/README.md b/static/virtualBackgroundImages/README.md new file mode 100644 index 000000000..ed7088af6 --- /dev/null +++ b/static/virtualBackgroundImages/README.md @@ -0,0 +1 @@ +We are putting the virtual background images here because we don't want to increase the size of the war file. We will use the Github as CDN. diff --git a/static/virtualBackgroundImages/virtual-background0.png b/static/virtualBackgroundImages/virtual-background0.png new file mode 100644 index 000000000..df3e3454e Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background0.png differ diff --git a/static/virtualBackgroundImages/virtual-background1.jpg b/static/virtualBackgroundImages/virtual-background1.jpg new file mode 100644 index 000000000..a626e1f85 Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background1.jpg differ diff --git a/static/virtualBackgroundImages/virtual-background2.jpg b/static/virtualBackgroundImages/virtual-background2.jpg new file mode 100644 index 000000000..4a5d7c0fa Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background2.jpg differ diff --git a/static/virtualBackgroundImages/virtual-background3.jpg b/static/virtualBackgroundImages/virtual-background3.jpg new file mode 100644 index 000000000..c8db4db1b Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background3.jpg differ diff --git a/static/virtualBackgroundImages/virtual-background4.jpg b/static/virtualBackgroundImages/virtual-background4.jpg new file mode 100644 index 000000000..b4ace3414 Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background4.jpg differ diff --git a/static/virtualBackgroundImages/virtual-background5.jpg b/static/virtualBackgroundImages/virtual-background5.jpg new file mode 100644 index 000000000..0bcd0f4e0 Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background5.jpg differ diff --git a/static/virtualBackgroundImages/virtual-background7.jpg b/static/virtualBackgroundImages/virtual-background7.jpg new file mode 100644 index 000000000..cec2e69d4 Binary files /dev/null and b/static/virtualBackgroundImages/virtual-background7.jpg differ diff --git a/test/rest_helper.py b/test/rest_helper.py index 953dbe984..6ff173a2f 100644 --- a/test/rest_helper.py +++ b/test/rest_helper.py @@ -34,7 +34,46 @@ def get_broadcasts(self): else: response.raise_for_status() - def get_vod_for(self, streamId): + def create_broadcast_for_play_only_speed_test(self): + url = f"{self.rest_url}/request?_path={self.app_name}/rest/v2/broadcasts/create" + payload = { + "hlsViewerCount": 0, + "dashViewerCount": 0, + "webRTCViewerCount": 0, + "rtmpViewerCount": 0, + "mp4Enabled": 0, + "playlistLoopEnabled": True, + "autoStartStopEnabled": False, + "plannedStartDate": 0, + "playListItemList": [ + { + "type": "VoD", + "streamUrl": "https://github.com/ant-media/conference-call-application/raw/refs/heads/refactorPlayOnlySpeedTest/static/speedTestVideo/speed-test-sample-video.mp4", + "name": "speedTestSampleStream", + "seekTimeInMs": 0, + "durationInMs": 60000 + } + ], + "name": "speedTestSampleStream", + "streamId": "speedTestSampleStream", + "type": "playlist" + } + + response = self.session.post(url, json=payload) + if response.status_code == 200: + json_data = response.json() + print("Broadcast created successfully:", json_data) + return json_data + + def start_broadcast(self, streamId): + url = f"{self.rest_url}/request?_path={self.app_name}/rest/v2/broadcasts/{streamId}/start" + response = self.session.post(url) + if response.status_code == 200: + json_data = response.json() + print("Broadcast started successfully:", json_data) + return json_data + + def getVoDFor(self, streamId): resp = self.session.get(self.rest_url +"/request?_path="+self.app_name+"/rest/v2/vods/list/0/5") print("get_vod_for "+str(streamId+":"+str(resp.text))) json_data = json.loads(resp.text) diff --git a/test/test_join_leave.py b/test/test_join_leave.py index 25cfc0483..7d21456da 100644 --- a/test/test_join_leave.py +++ b/test/test_join_leave.py @@ -58,8 +58,8 @@ def create_participants_with_test_tool(self, participant_name, room, count): process = subprocess.Popen( ["bash", script_path] + parameters, cwd=directory, - stdout=subprocess.DEVNULL, - stderr=subprocess.DEVNULL + #stdout=subprocess.DEVNULL, + #stderr=subprocess.DEVNULL ) return process diff --git a/test/test_main.py b/test/test_main.py index 8358ac9f4..d8f301cfa 100644 --- a/test/test_main.py +++ b/test/test_main.py @@ -18,7 +18,7 @@ os.environ['TEST_APP_NAME'] = app_name_prefix + str(random.randint(100, 999)) #Keep it True to stop tests immediately after a failed test -fail_fast = False +fail_fast = True suite = unittest.TestSuite() suite.addTest(TestDeployment('test_install_app')) @@ -32,8 +32,10 @@ suite.addTests(suite2) +#suite.addTest(TestJoinLeave("test_join_without_camera_mic")) suite.addTest(TestDeployment('test_delete_app')) ret = not unittest.TextTestRunner(verbosity=2, failfast=fail_fast).run(suite).wasSuccessful() sys.exit(ret) + diff --git a/test/test_webinar.py b/test/test_webinar.py index 378aa55d4..e425ae892 100644 --- a/test/test_webinar.py +++ b/test/test_webinar.py @@ -20,8 +20,15 @@ def setUp(self): print(self._testMethodName, " starting...") self.url = os.environ.get('SERVER_URL') self.test_app_name = os.environ.get('TEST_APP_NAME') + self.user = os.environ.get('AMS_USER') + self.password = os.environ.get('AMS_PASSWORD') self.chrome = Browser() self.chrome.init(not self.is_local) + self.chrome.init(True) + self.rest_helper = RestHelper(self.url, self.user, self.password, self.test_app_name) + self.rest_helper.login() + self.rest_helper.create_broadcast_for_play_only_speed_test() + self.rest_helper.start_broadcast("speedTestSampleStream") #self.startLoadTest() def tearDown(self): @@ -136,6 +143,8 @@ def join_room_as_player(self, participant, room, skip_speed_test=False): app = "" handle = self.chrome.open_in_new_tab(self.url+app+"/"+room+"?playOnly=true&role=listener&streamName=" + participant + ("&enterDirectly=true" if skip_speed_test else "")) + wait = self.chrome.get_wait() + #name_text_box = self.chrome.get_element_with_retry(By.ID,"participant_name") #self.chrome.write_to_element(name_text_box, participant) @@ -147,7 +156,7 @@ def join_room_as_player(self, participant, room, skip_speed_test=False): time.sleep(5) speedTestCircularProgress = self.chrome.get_element_with_retry(By.ID,"speed-test-modal-circle-progress-bar", retries=20) - assert(speedTestCircularProgress.is_displayed()) + wait.until(lambda x: speedTestCircularProgress.is_displayed()) time.sleep(5)