org.kurento.tree.server.treemanager.LessLoadedElasticTM Maven / Gradle / Ivy
/*
* (C) Copyright 2016 Kurento (http://kurento.org/)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/
package org.kurento.tree.server.treemanager;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import org.kurento.client.IceCandidate;
import org.kurento.commons.exception.KurentoException;
import org.kurento.jsonrpc.Session;
import org.kurento.tree.client.TreeEndpoint;
import org.kurento.tree.client.TreeException;
import org.kurento.tree.server.app.TreeElementSession;
import org.kurento.tree.server.kms.Kms;
import org.kurento.tree.server.kms.Pipeline;
import org.kurento.tree.server.kms.Plumber;
import org.kurento.tree.server.kms.WebRtc;
import org.kurento.tree.server.kmsmanager.KmsLoad;
import org.kurento.tree.server.kmsmanager.KmsManager;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* This TreeManager has the following characteristics:
*
* - It allows N trees
* - Creates WebRtcEndpoint for sinks (viewers) only in non-source KMSs
* - Fills less loaded node.
* - It considers new KMSs after start.
*
*
* @author [email protected]
*/
public class LessLoadedElasticTM extends AbstractNTreeTM {
private static final Logger log = LoggerFactory.getLogger(LessLoadedElasticTM.class);
public class LessLoadedTreeInfo extends TreeInfo {
private KmsManager kmsManager;
private String treeId;
private Kms sourceKms;
private Pipeline sourcePipeline;
private List sourcePlumbers = new ArrayList<>();
private WebRtc source;
private List leafPipelines = new ArrayList<>();
private List leafPlumbers = new ArrayList<>();
private Map sinks = new ConcurrentHashMap<>();
private Map ownPipelineByKms = new ConcurrentHashMap<>();
private Map webRtcsById = new ConcurrentHashMap<>();
private int numSinks = 0;
public LessLoadedTreeInfo(String treeId, KmsManager kmsManager) {
this.treeId = treeId;
this.kmsManager = kmsManager;
if (kmsManager.getKmss().isEmpty()) {
throw new KurentoException("LessLoadedNElasticTM cannot be used without initial kmss");
}
sourceKms = kmsManager.getKmss().get(0);
}
@Override
public void release() {
source.release();
for (WebRtc webRtc : sinks.values()) {
webRtc.release();
}
}
@Override
public String setTreeSource(Session session, String offerSdp) {
if (source != null) {
removeTreeSource();
}
if (sourcePipeline == null) {
sourcePipeline = sourceKms.createPipeline();
ownPipelineByKms.put(sourceKms, sourcePipeline);
}
source = sourcePipeline.createWebRtc(new TreeElementSession(session, treeId, null));
String sdpAnswer = source.processSdpOffer(offerSdp);
source.gatherCandidates();
return sdpAnswer;
}
@Override
public void removeTreeSource() {
source.release();
source = null;
}
@Override
public TreeEndpoint addTreeSink(Session session, String sdpOffer) {
List kmss = kmsManager.getKmssSortedByLoad();
Kms selectedKms = kmss.get(0).getKms();
if (selectedKms == sourceKms) {
selectedKms = kmss.get(1).getKms();
}
Pipeline pipeline = getOrCreatePipeline(selectedKms);
if (pipeline.getKms().allowMoreElements()) {
String id = UUID.randomUUID().toString();
WebRtc webRtc = pipeline.createWebRtc(new TreeElementSession(session, treeId, id));
pipeline.getPlumbers().get(0).connect(webRtc);
String sdpAnswer = webRtc.processSdpOffer(sdpOffer);
webRtc.gatherCandidates();
webRtcsById.put(id, webRtc);
webRtc.setLabel("Sink " + id + ")");
return new TreeEndpoint(sdpAnswer, id);
} else {
throw new TreeException("Max number of viewers reached");
}
}
private Pipeline getOrCreatePipeline(Kms kms) {
Pipeline pipeline = ownPipelineByKms.get(kms);
if (pipeline == null) {
pipeline = kms.createPipeline();
ownPipelineByKms.put(kms, pipeline);
pipeline.setLabel(UUID.randomUUID().toString());
leafPipelines.add(pipeline);
Plumber[] plumbers = sourcePipeline.link(pipeline);
source.connect(plumbers[0]);
this.sourcePlumbers.add(plumbers[0]);
this.leafPlumbers.add(plumbers[1]);
}
return pipeline;
}
@Override
public void removeTreeSink(String sinkId) {
WebRtc webRtc = webRtcsById.get(sinkId);
Plumber plumber = (Plumber) webRtc.getSource();
webRtc.release();
if (plumber.getSinks().isEmpty()) {
Plumber remotePlumber = plumber.getLinkedTo();
Pipeline pipeline = plumber.getPipeline();
ownPipelineByKms.remove(pipeline.getKms());
pipeline.release();
remotePlumber.release();
}
}
@Override
public void addSinkIceCandidate(String sinkId, IceCandidate iceCandidate) {
webRtcsById.get(sinkId).addIceCandidate(iceCandidate);
}
@Override
public void addTreeIceCandidate(IceCandidate iceCandidate) {
source.addIceCandidate(iceCandidate);
}
}
private KmsManager kmsManager;
public LessLoadedElasticTM(KmsManager kmsManager) {
this.kmsManager = kmsManager;
}
@Override
public KmsManager getKmsManager() {
return kmsManager;
}
@Override
protected TreeInfo createTreeInfo(String treeId) {
return new LessLoadedTreeInfo(treeId, kmsManager);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy