diff --git a/bbb-screenshare/.classpath b/bbb-screenshare/.classpath new file mode 100755 index 0000000000000000000000000000000000000000..ddb4e05b80d8e90cdb0a2414d56ffe0cf2c51fd2 --- /dev/null +++ b/bbb-screenshare/.classpath @@ -0,0 +1,38 @@ +<?xml version="1.0" encoding="UTF-8"?> +<classpath> + <classpathentry kind="src" path="app/src/main/java"/> + <classpathentry kind="src" path="app/src/main/scala"/> + <classpathentry kind="src" path="app/src/test/java"/> + <classpathentry kind="src" path="jws/webstart/src/main/java"/> + <classpathentry kind="con" path="org.scala-ide.sdt.launching.SCALA_CONTAINER"/> + <classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER"/> + <classpathentry kind="lib" path="app/lib/aopalliance-1.0.jar"/> + <classpathentry kind="lib" path="app/lib/commons-fileupload-1.2.2.jar"/> + <classpathentry kind="lib" path="app/lib/commons-io-2.1.jar"/> + <classpathentry kind="lib" path="app/lib/commons-pool-1.5.6.jar"/> + <classpathentry kind="lib" path="app/lib/configgy-2.0.0.jar"/> + <classpathentry kind="lib" path="app/lib/easymock-2.4.jar"/> + <classpathentry kind="lib" path="app/lib/gson-1.7.1.jar"/> + <classpathentry kind="lib" path="app/lib/jcl-over-slf4j-1.7.9.jar"/> + <classpathentry kind="lib" path="app/lib/jedis-1.5.1.jar"/> + <classpathentry kind="lib" path="app/lib/jul-to-slf4j-1.7.9.jar"/> + <classpathentry kind="lib" path="app/lib/log4j-over-slf4j-1.7.9.jar"/> + <classpathentry kind="lib" path="app/lib/logback-classic-1.1.2.jar"/> + <classpathentry kind="lib" path="app/lib/logback-core-1.1.2.jar"/> + <classpathentry kind="lib" path="app/lib/mina-core-2.0.8.jar"/> + <classpathentry kind="lib" path="app/lib/mina-integration-beans-2.0.8.jar"/> + <classpathentry kind="lib" path="app/lib/mina-integration-jmx-2.0.8.jar"/> + <classpathentry kind="lib" path="app/lib/red5-io-1.0.6-SNAPSHOT.jar"/> + <classpathentry kind="lib" path="app/lib/red5-server-1.0.6-SNAPSHOT.jar"/> + <classpathentry kind="lib" path="app/lib/red5-server-common-1.0.6-SNAPSHOT.jar"/> + <classpathentry kind="lib" path="app/lib/scala-library-2.9.2.jar"/> + <classpathentry kind="lib" path="app/lib/servlet-api-2.5.jar"/> + <classpathentry kind="lib" path="app/lib/slf4j-api-1.7.9.jar"/> + <classpathentry kind="lib" path="app/lib/spring-aop-4.0.8.RELEASE.jar"/> + <classpathentry kind="lib" path="app/lib/spring-beans-4.0.8.RELEASE.jar"/> + <classpathentry kind="lib" path="app/lib/spring-context-4.0.8.RELEASE.jar"/> + <classpathentry kind="lib" path="app/lib/spring-core-4.0.8.RELEASE.jar"/> + <classpathentry kind="lib" path="app/lib/spring-web-4.0.8.RELEASE.jar"/> + <classpathentry kind="lib" path="app/lib/spring-webmvc-4.0.7.RELEASE.jar"/> + <classpathentry kind="output" path="bin"/> +</classpath> diff --git a/bbb-screenshare/.gitignore b/bbb-screenshare/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..20d12dd51408a0acf0f81cef518c9f7b5564e8c0 --- /dev/null +++ b/bbb-screenshare/.gitignore @@ -0,0 +1,8 @@ +.manager +.scala_dependencies +.classpath +.gradle/ +.project +app/build/ +lib/ +build diff --git a/bbb-screenshare/.project b/bbb-screenshare/.project new file mode 100755 index 0000000000000000000000000000000000000000..7e258648efc7fccd62009ea00e461fc2f8647d96 --- /dev/null +++ b/bbb-screenshare/.project @@ -0,0 +1,18 @@ +<?xml version="1.0" encoding="UTF-8"?> +<projectDescription> + <name>s-bbb-screenshare</name> + <comment></comment> + <projects> + </projects> + <buildSpec> + <buildCommand> + <name>org.scala-ide.sdt.core.scalabuilder</name> + <arguments> + </arguments> + </buildCommand> + </buildSpec> + <natures> + <nature>org.scala-ide.sdt.core.scalanature</nature> + <nature>org.eclipse.jdt.core.javanature</nature> + </natures> +</projectDescription> diff --git a/bbb-screenshare/app/.classpath.old b/bbb-screenshare/app/.classpath.old new file mode 100755 index 0000000000000000000000000000000000000000..263d60672590638b500b7dd3c7835f47d355c50c --- /dev/null +++ b/bbb-screenshare/app/.classpath.old @@ -0,0 +1,26 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<classpath> + <classpathentry kind="src" path="src/main/java"/> + <classpathentry kind="output" path="build/classes/main"/> + <classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER"/> + <classpathentry kind="src" path="/common" combineaccessrules="false"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/commons-fileupload/commons-fileupload/jars/commons-fileupload-1.2.1.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/commons-io/commons-io/jars/commons-io-1.4.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/javax.servlet/servlet-api/jars/servlet-api-2.5.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/log4j-over-slf4j/jars/log4j-over-slf4j-1.5.6.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/logback-classic/jars/logback-classic-0.9.14.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/logback-core/jars/logback-core-0.9.14.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/net/lag/configgy/configgy/jars/configgy-1.5.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/org.apache.mina/mina-core/jars/mina-core-2.0.0-RC1.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/org.apache.mina/mina-integration-spring/jars/mina-integration-spring-1.1.7.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/org.scala-lang/scala-library/jars/scala-library-2.7.7.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/org/red5/red5/jars/red5-0.91.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/slf4j-api/jars/slf4j-api-1.5.6.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-aop/jars/spring-aop-3.0.0.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-beans/jars/spring-beans-3.0.0.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-context/jars/spring-context-3.0.0.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-core/jars/spring-core-3.0.0.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-web/jars/spring-web-3.0.0.jar"/> + <classpathentry kind="lib" path="/home/firstuser/.gradle/cache/spring/spring-webmvc/jars/spring-webmvc-2.5.6.jar"/> +</classpath> diff --git a/bbb-screenshare/app/build.gradle b/bbb-screenshare/app/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..cebd3fe968f3dff9d5d5e55abb1a9761a2e91e8f --- /dev/null +++ b/bbb-screenshare/app/build.gradle @@ -0,0 +1,160 @@ +apply plugin: 'scala' +apply plugin: 'java' +apply plugin: 'war' +apply plugin: 'eclipse' + +version = '0.8' +jar.enabled = true + +def appName = 'bbb-screenshare' + +archivesBaseName = appName + +task resolveDeps(type: Copy) { + into('lib') + from configurations.default + from configurations.default.allArtifacts*.file +} + +repositories { + mavenCentral() + mavenLocal() + add(new org.apache.ivy.plugins.resolver.ChainResolver()) { + name = 'remote' + returnFirst = true + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "googlecode" + addArtifactPattern "http://red5.googlecode.com/svn/repository/[artifact](-[revision]).[ext]" + addArtifactPattern "http://red5.googlecode.com/svn/repository/[organisation]/[artifact](-[revision]).[ext]" + } + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "blindside-repos" + addArtifactPattern "http://blindside.googlecode.com/svn/repository/[artifact](-[revision]).[ext]" + addArtifactPattern "http://blindside.googlecode.com/svn/repository/[organisation]/[artifact](-[revision]).[ext]" + } + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "maven2-central" + m2compatible = true + addArtifactPattern "http://repo1.maven.org/maven2/[organisation]/[module]/[revision]/[artifact](-[revision]).[ext]" + addArtifactPattern "http://repo1.maven.org/maven2/[organisation]/[artifact]/[revision]/[artifact](-[revision]).[ext]" + } + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "testng_ibiblio_maven2" + m2compatible = true + addArtifactPattern "http://repo1.maven.org/maven2/[organisation]/[module]/[revision]/[artifact](-[revision])-jdk15.[ext]" + addArtifactPattern "http://repo1.maven.org/maven2/[organisation]/[artifact]/[revision]/[artifact](-[revision])-jdk15.[ext]" + } + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "netty-dependency" + m2compatible = true + addArtifactPattern "http://repository.jboss.org/nexus/content/groups/public-jboss/[organisation]/[module]/[revision]/[artifact](-[revision]).[ext]" + addArtifactPattern "http://repo1.maven.org/maven2/[organisation]/[artifact]/[revision]/[artifact](-[revision]).[ext]" + } + add(new org.apache.ivy.plugins.resolver.URLResolver()) { + name = "spring-bundles" + m2compatible = true + addArtifactPattern "http://repository.springsource.com/maven/bundles/external/[organisation]/[module]/[revision]/[artifact]-[revision].[ext]" + addArtifactPattern "http://repository.springsource.com/maven/bundles/release/[organisation]/[module]/[revision]/[artifact]-[revision].[ext]" + } + mavenRepo name: "jboss", urls: "http://repository.jboss.org/nexus/content/groups/public-jboss" + mavenRepo name: "sonatype-snapshot", urls: "http://oss.sonatype.org/content/repositories/snapshots" + mavenRepo name: "sonatype-releases", urls: "http://oss.sonatype.org/content/repositories/releases" + } +} + +dependencies { + // Servlet + providedCompile 'javax.servlet:servlet-api:2.5@jar' + + // Mina + providedCompile 'org.apache.mina:mina-core:2.0.8@jar' + providedCompile 'org.apache.mina:mina-integration-beans:2.0.8@jar' + providedCompile 'org.apache.mina:mina-integration-jmx:2.0.8@jar' + + // Spring + providedCompile 'org.springframework:spring-web:4.0.8.RELEASE@jar' + providedCompile 'org.springframework:spring-beans:4.0.8.RELEASE@jar' + providedCompile 'org.springframework:spring-context:4.0.8.RELEASE@jar' + providedCompile 'org.springframework:spring-core:4.0.8.RELEASE@jar' + + // Red5 + providedCompile 'org.red5:red5-server:1.0.6-SNAPSHOT@jar' + providedCompile 'org.red5:red5-server-common:1.0.6-SNAPSHOT@jar' + providedCompile 'org.red5:red5-io:1.0.6-SNAPSHOT@jar' + + // Logging + providedCompile 'ch.qos.logback:logback-core:1.1.2@jar' + providedCompile 'ch.qos.logback:logback-classic:1.1.2@jar' + providedCompile 'org.slf4j:log4j-over-slf4j:1.7.9@jar' + providedCompile 'org.slf4j:jcl-over-slf4j:1.7.9@jar' + providedCompile 'org.slf4j:jul-to-slf4j:1.7.9@jar' + providedCompile 'org.slf4j:slf4j-api:1.7.9@jar' + + + // Needed for the JVM shutdown hook but needs to be put into red5/lib dir. + // Otherwise we get exception on aop utils class not found. + providedCompile 'org.springframework:spring-aop:4.0.8.RELEASE@jar' + providedCompile 'aopalliance:aopalliance:1.0@jar' + + // Testing + //compile 'org.testng:testng:5.8@jar' + compile 'org.easymock:easymock:2.4@jar' + + // Testing + //testRuntime 'org/testng:testng:5.8@jar' + testRuntime 'org.easymock:easymock:2.4@jar' + + // Tunnelling servlet + compile 'org.springframework:spring-webmvc:4.0.7.RELEASE@jar' + + // Need to put commons-fileupload and commons-io in red5/lib dir. Otherwise, we get an + // java.lang.NoClassDefFoundError: org/apache/commons/fileupload/FileItemFactory or + // java.lang.NoClassDefFoundError: org/apache/commons/io/output/DeferredFileOutputStream + // ralam (Feb 27, 2013) + providedCompile 'commons-fileupload:commons-fileupload:1.2.2@jar' + providedCompile 'commons-io:commons-io:2.1@jar' + + // Libraries needed to run the scala tools + scalaTools 'org.scala-lang:scala-compiler:2.9.2' + scalaTools 'org.scala-lang:scala-library:2.9.2' + + // workaround for http://issues.gradle.org/browse/GRADLE-1273 + //compileScala.classpath = sourceSets.main.compileClasspath + files(sourceSets.main.classesDir) + //compileTestScala.classpath = sourceSets.test.compileClasspath + files(sourceSets.test.classesDir) + + // Libraries needed for scala api + compile 'org.scala-lang:scala-library:2.9.2' + compile 'net.lag:configgy:2.0.0@jar' + + //redis + compile 'redis.clients:jedis:1.5.1' + providedCompile 'commons-pool:commons-pool:1.5.6' + compile 'com.google.code.gson:gson:1.7.1' +} + +test { + useTestNG() +} + +war.doLast { + ant.unzip(src: war.archivePath, dest: "$buildDir/screenshare") +} + +task deploy() << { + def red5AppsDir = '/usr/share/red5/webapps' + def screenshareDir = new File("${red5AppsDir}/screenshare") + println "Deleting $screenshareDir" + ant.delete(dir: screenshareDir) + ant.mkdir(dir: screenshareDir) + ant.copy(todir: screenshareDir) { + fileset(dir: "$buildDir/screenshare") + } + def jwsLibDir = new File("${red5AppsDir}/screenshare/lib") + ant.mkdir(dir: jwsLibDir) + ant.copy(todir: jwsLibDir) { + fileset(dir: "jws/lib") + } + ant.copy(todir: screenshareDir) { + fileset(file: "jws/screenshare.jnlp") + } +} diff --git a/bbb-screenshare/app/deploy.sh b/bbb-screenshare/app/deploy.sh new file mode 100755 index 0000000000000000000000000000000000000000..408e1f44f41827b219f8b7e004016f5c17f2ac94 --- /dev/null +++ b/bbb-screenshare/app/deploy.sh @@ -0,0 +1,4 @@ +sudo chmod -R 777 /usr/share/red5/webapps +gradle clean war deploy +sudo chmod -R 777 /usr/share/red5/webapps + diff --git a/bbb-screenshare/app/jws/screenshare.jnlp b/bbb-screenshare/app/jws/screenshare.jnlp new file mode 100755 index 0000000000000000000000000000000000000000..f88969b672b728dc5b4a308ee4e624bddc01c074 --- /dev/null +++ b/bbb-screenshare/app/jws/screenshare.jnlp @@ -0,0 +1,50 @@ +<?xml version="1.0" encoding="UTF-8"?> +<jnlp spec="1.0+" codebase="." href=""> +<!-- + Keep href empty. Otherwise this jnlp file will always be cached. + http://www.coderanch.com/t/284889/JSP/java/Caching-JNLP +--> + <information> + <title>BigBlueButton Screen Share</title> + <vendor>BigBlueButton</vendor> + </information> + + <resources> + <j2se version="1.7+" href="http://java.sun.com/products/autodl/j2se"/> + <jar href="$$jnlpUrl/lib/javacv-screenshare-0.0.1.jar" main="true" /> + <jar href="$$jnlpUrl/lib/javacv.jar" /> + <jar href="$$jnlpUrl/lib/javacpp.jar" /> + <jar href="$$jnlpUrl/lib/ffmpeg.jar" /> + </resources> + + <resources os="Windows" arch="amd64"> + <nativelib href="$$jnlpUrl/lib/ffmpeg-windows-x86_64.jar" download="eager"/> + </resources> + + <resources os="Windows" arch="x86"> + <nativelib href="$$jnlpUrl/lib/ffmpeg-windows-x86.jar" download="eager"/> + </resources> + + <resources os="Linux" arch="x86_64 amd64"> + <nativelib href="$$jnlpUrl/lib/ffmpeg-linux-x86_64.jar" download="eager"/> + </resources> + + <resources os="Linux" arch="x86 i386 i486 i586 i686"> + <nativelib href="$$jnlpUrl/lib/ffmpeg-linux-x86.jar" download="eager"/> + </resources> + + + <application-desc + name="Desktop Sharing Demo Application" + main-class="org.bigbluebutton.screenshare.client.DeskshareMain"> + <argument>$$publishUrl</argument> + <argument>$$serverUrl</argument> + <argument>$$meetingId</argument> + <argument>$$streamId</argument> + <argument>$$fullScreen</argument> + <argument>$$codecOptions</argument> + <argument>$$errorMessage</argument> + </application-desc> + <security><all-permissions/></security> + <update check="always" policy="always"/> +</jnlp> diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiff.java b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiff.java new file mode 100755 index 0000000000000000000000000000000000000000..97fee6d3bf07ab0b540100052d73bebb029dbae8 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiff.java @@ -0,0 +1,642 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.jardiff; + +import java.io.*; +import java.util.*; +import java.util.jar.*; +import java.util.zip.*; + + +/** + * JarDiff is able to create a jar file containing the delta between two + * jar files (old and new). The delta jar file can then be applied to the + * old jar file to reconstruct the new jar file. + * <p> + * Refer to the JNLP spec for details on how this is done. + * + * @version 1.13, 06/26/03 + */ +public class JarDiff implements JarDiffConstants { + private static final int DEFAULT_READ_SIZE = 2048; + private static byte[] newBytes = new byte[DEFAULT_READ_SIZE]; + private static byte[] oldBytes = new byte[DEFAULT_READ_SIZE]; + private static ResourceBundle _resources = null; + + // The JARDiff.java is the stand-along jardiff.jar tool. Thus, we do not + // depend on Globals.java and other stuff here. Instead, we use an explicit + // _debug flag. + private static boolean _debug; + + public static ResourceBundle getResources() { + if (_resources == null) { + _resources = ResourceBundle.getBundle("jnlp/sample/jardiff/resources/strings"); + } + return _resources; + } + + /** + * Creates a patch from the two passed in files, writing the result + * to <code>os</code>. + */ + public static void createPatch(String oldPath, String newPath, + OutputStream os, boolean minimal) throws IOException{ + JarFile2 oldJar = new JarFile2(oldPath); + JarFile2 newJar = new JarFile2(newPath); + + try { + Iterator entries; + HashMap moved = new HashMap(); + HashSet visited = new HashSet(); + HashSet implicit = new HashSet(); + HashSet moveSrc = new HashSet(); + HashSet newEntries = new HashSet(); + + + // FIRST PASS + // Go through the entries in new jar and + // determine which files are candidates for implicit moves + // ( files that has the same filename and same content in old.jar + // and new.jar ) + // and for files that cannot be implicitly moved, we will either + // find out whether it is moved or new (modified) + entries = newJar.getJarEntries(); + if (entries != null) { + while (entries.hasNext()) { + JarEntry newEntry = (JarEntry)entries.next(); + String newname = newEntry.getName(); + + // Return best match of contents, will return a name match if possible + String oldname = oldJar.getBestMatch(newJar, newEntry); + if (oldname == null) { + // New or modified entry + if (_debug) { + System.out.println("NEW: "+ newname); + } + newEntries.add(newname); + } else { + // Content already exist - need to do a move + + // Should do implicit move? Yes, if names are the same, and + // no move command already exist from oldJar + if (oldname.equals(newname) && !moveSrc.contains(oldname)) { + if (_debug) { + System.out.println(newname + " added to implicit set!"); + } + implicit.add(newname); + } else { + // The 1.0.1/1.0 JarDiffPatcher cannot handle + // multiple MOVE command with same src. + // The work around here is if we are going to generate + // a MOVE command with duplicate src, we will + // instead add the target as a new file. This way + // the jardiff can be applied by 1.0.1/1.0 + // JarDiffPatcher also. + if (!minimal && (implicit.contains(oldname) || + moveSrc.contains(oldname) )) { + + // generate non-minimal jardiff + // for backward compatibility + + if (_debug) { + + System.out.println("NEW: "+ newname); + } + newEntries.add(newname); + } else { + // Use newname as key, since they are unique + if (_debug) { + System.err.println("moved.put " + newname + " " + oldname); + } + moved.put(newname, oldname); + moveSrc.add(oldname); + } + // Check if this disables an implicit 'move <oldname> <oldname>' + if (implicit.contains(oldname) && minimal) { + + if (_debug) { + System.err.println("implicit.remove " + oldname); + + System.err.println("moved.put " + oldname + " " + oldname); + + } + implicit.remove(oldname); + moved.put(oldname, oldname); + moveSrc.add(oldname); + } + + + } + } + } + } //if (entries != null) + + // SECOND PASS: <deleted files> = <oldjarnames> - <implicitmoves> - + // <source of move commands> - <new or modified entries> + ArrayList deleted = new ArrayList(); + entries = oldJar.getJarEntries(); + if (entries != null) { + while (entries.hasNext()) { + JarEntry oldEntry = (JarEntry)entries.next(); + String oldName = oldEntry.getName(); + if (!implicit.contains(oldName) && !moveSrc.contains(oldName) + && !newEntries.contains(oldName)) { + if (_debug) { + System.err.println("deleted.add " + oldName); + } + deleted.add(oldName); + } + } + } + + //DEBUG + if (_debug) { + //DEBUG: print out moved map + entries = moved.keySet().iterator(); + if (entries != null) { + System.out.println("MOVED MAP!!!"); + while (entries.hasNext()) { + String newName = (String)entries.next(); + String oldName = (String)moved.get(newName); + System.out.println("key is " + newName + " value is " + oldName); + } + } + + //DEBUG: print out IMOVE map + entries = implicit.iterator(); + if (entries != null) { + System.out.println("IMOVE MAP!!!"); + while (entries.hasNext()) { + String newName = (String)entries.next(); + System.out.println("key is " + newName); + } + } + } + + JarOutputStream jos = new JarOutputStream(os); + + // Write out all the MOVEs and REMOVEs + createIndex(jos, deleted, moved); + + // Put in New and Modified entries + entries = newEntries.iterator(); + if (entries != null) { + + while (entries.hasNext()) { + String newName = (String)entries.next(); + if (_debug) { + System.out.println("New File: " + newName); + } + writeEntry(jos, newJar.getEntryByName(newName), newJar); + } + } + + + jos.finish(); + jos.close(); + + } catch (IOException ioE){ + throw ioE; + } finally { + try { + oldJar.getJarFile().close(); + } catch (IOException e1) { + //ignore + } + try { + newJar.getJarFile().close(); + } catch (IOException e1) { + //ignore + } + } // finally + } + + /** + * Writes the index file out to <code>jos</code>. + * <code>oldEntries</code> gives the names of the files that were removed, + * <code>movedMap</code> maps from the new name to the old name. + */ + private static void createIndex(JarOutputStream jos, List oldEntries, + Map movedMap) throws + IOException { + StringWriter writer = new StringWriter(); + + writer.write(VERSION_HEADER); + writer.write("\r\n"); + + // Write out entries that have been removed + for (int counter = 0; counter < oldEntries.size(); counter++) { + String name = (String)oldEntries.get(counter); + + writer.write(REMOVE_COMMAND); + writer.write(" "); + writeEscapedString(writer, name); + writer.write("\r\n"); + } + + // And those that have moved + Iterator names = movedMap.keySet().iterator(); + + if (names != null) { + while (names.hasNext()) { + String newName = (String)names.next(); + String oldName = (String)movedMap.get(newName); + + writer.write(MOVE_COMMAND); + writer.write(" "); + writeEscapedString(writer, oldName); + writer.write(" "); + writeEscapedString(writer, newName); + writer.write("\r\n"); + + } + } + + JarEntry je = new JarEntry(INDEX_NAME); + byte[] bytes = writer.toString().getBytes("UTF-8"); + + writer.close(); + jos.putNextEntry(je); + jos.write(bytes, 0, bytes.length); + } + + private static void writeEscapedString(Writer writer, String string) + throws IOException { + int index = 0; + int last = 0; + char[] chars = null; + + while ((index = string.indexOf(' ', index)) != -1) { + if (last != index) { + if (chars == null) { + chars = string.toCharArray(); + } + writer.write(chars, last, index - last); + } + last = index; + index++; + writer.write('\\'); + } + if (last != 0) { + writer.write(chars, last, chars.length - last); + } + else { + // no spaces + writer.write(string); + } + } + + private static void writeEntry(JarOutputStream jos, JarEntry entry, + JarFile2 file) throws IOException { + writeEntry(jos, entry, file.getJarFile().getInputStream(entry)); + } + + private static void writeEntry(JarOutputStream jos, JarEntry entry, + InputStream data) throws IOException { + jos.putNextEntry(entry); + + try { + // Read the entry + int size = data.read(newBytes); + + while (size != -1) { + jos.write(newBytes, 0, size); + size = data.read(newBytes); + } + } catch(IOException ioE) { + throw ioE; + } finally { + try { + data.close(); + } catch(IOException e){ + //Ignore + } + + } + } + + + + + /** + * JarFile2 wraps a JarFile providing some convenience methods. + */ + private static class JarFile2 { + private JarFile _jar; + private List _entries; + private HashMap _nameToEntryMap; + private HashMap _crcToEntryMap; + + public JarFile2(String path) throws IOException { + _jar = new JarFile(new File(path)); + index(); + } + + public JarFile getJarFile() { + return _jar; + } + + public Iterator getJarEntries() { + return _entries.iterator(); + } + + public JarEntry getEntryByName(String name) { + return (JarEntry)_nameToEntryMap.get(name); + } + + /** + * Returns true if the two InputStreams differ. + */ + private static boolean differs(InputStream oldIS, InputStream newIS) + throws IOException { + int newSize = 0; + int oldSize; + int total = 0; + boolean retVal = false; + + try{ + while (newSize != -1) { + newSize = newIS.read(newBytes); + oldSize = oldIS.read(oldBytes); + + if (newSize != oldSize) { + if (_debug) { + System.out.println("\tread sizes differ: " + newSize + + " " + oldSize + " total " + total); + } + retVal = true; + break; + } + if (newSize > 0) { + while (--newSize >= 0) { + total++; + if (newBytes[newSize] != oldBytes[newSize]) { + if (_debug) { + System.out.println("\tbytes differ at " + + total); + } + retVal = true; + break; + } + if ( retVal ) { + //Jump out + break; + } + newSize = 0; + } + } + } + } catch(IOException ioE){ + throw ioE; + } finally { + try { + oldIS.close(); + } catch(IOException e){ + //Ignore + } + try { + newIS.close(); + } catch(IOException e){ + //Ignore + } + } + return retVal; + } + + public String getBestMatch(JarFile2 file, JarEntry entry) throws IOException { + // check for same name and same content, return name if found + if (contains(file, entry)) { + return (entry.getName()); + } + + // return name of same content file or null + return (hasSameContent(file,entry)); + } + + public boolean contains(JarFile2 f, JarEntry e) throws IOException { + + JarEntry thisEntry = getEntryByName(e.getName()); + + // Look up name in 'this' Jar2File - if not exist return false + if (thisEntry == null) + return false; + + // Check CRC - if no match - return false + if (thisEntry.getCrc() != e.getCrc()) + return false; + + // Check contents - if no match - return false + InputStream oldIS = getJarFile().getInputStream(thisEntry); + InputStream newIS = f.getJarFile().getInputStream(e); + boolean retValue = differs(oldIS, newIS); + + return !retValue; + } + + public String hasSameContent(JarFile2 file, JarEntry entry) throws + IOException { + + String thisName = null; + + Long crcL = new Long(entry.getCrc()); + + // check if this jar contains files with the passed in entry's crc + if (_crcToEntryMap.containsKey(crcL)) { + // get the Linked List with files with the crc + LinkedList ll = (LinkedList)_crcToEntryMap.get(crcL); + // go through the list and check for content match + ListIterator li = ll.listIterator(0); + if (li != null) { + while (li.hasNext()) { + JarEntry thisEntry = (JarEntry)li.next(); + + // check for content match + InputStream oldIS = getJarFile().getInputStream(thisEntry); + InputStream newIS = file.getJarFile().getInputStream(entry); + + if (!differs(oldIS, newIS)) { + thisName = thisEntry.getName(); + return thisName; + } + } + } + } + + return thisName; + + } + + + + + + private void index() throws IOException { + Enumeration entries = _jar.entries(); + + _nameToEntryMap = new HashMap(); + _crcToEntryMap = new HashMap(); + + _entries = new ArrayList(); + if (_debug) { + System.out.println("indexing: " + _jar.getName()); + } + if (entries != null) { + while (entries.hasMoreElements()) { + JarEntry entry = (JarEntry)entries.nextElement(); + + long crc = entry.getCrc(); + + Long crcL = new Long(crc); + + if (_debug) { + System.out.println("\t" + entry.getName() + " CRC " + + crc); + } + + _nameToEntryMap.put(entry.getName(), entry); + _entries.add(entry); + + // generate the CRC to entries map + if (_crcToEntryMap.containsKey(crcL)) { + // key exist, add the entry to the correcponding + // linked list + + // get the linked list + LinkedList ll = (LinkedList)_crcToEntryMap.get(crcL); + + // put in the new entry + ll.add(entry); + + // put it back in the hash map + _crcToEntryMap.put(crcL, ll); + } else { + // create a new entry in the hashmap for the new key + + // first create the linked list and put in the new + // entry + LinkedList ll = new LinkedList(); + ll.add(entry); + + // create the new entry in the hashmap + _crcToEntryMap.put(crcL, ll); + } + + } + } + } + + } + + + private static void showHelp() { + System.out.println("JarDiff: [-nonminimal (for backward compatibility with 1.0.1/1.0] [-creatediff | -applydiff] [-output file] old.jar new.jar"); + } + + // -creatediff -applydiff -debug -output file + public static void main(String[] args) throws IOException { + boolean diff = true; + boolean minimal = true; + String outputFile = "out.jardiff"; + + for (int counter = 0; counter < args.length; counter++) { + // for backward compatibilty with 1.0.1/1.0 + if (args[counter].equals("-nonminimal") || + args[counter].equals("-n")) { + minimal = false; + } + else if (args[counter].equals("-creatediff") || + args[counter].equals("-c")) { + diff = true; + } + else if (args[counter].equals("-applydiff") || + args[counter].equals("-a")) { + diff = false; + } + else if (args[counter].equals("-debug") || + args[counter].equals("-d")) { + _debug = true; + } + else if (args[counter].equals("-output") || + args[counter].equals("-o")) { + if (++counter < args.length) { + outputFile = args[counter]; + } + } + else if (args[counter].equals("-applydiff") || + args[counter].equals("-a")) { + diff = false; + } + else { + if ((counter + 2) != args.length) { + showHelp(); + System.exit(0); + } + if (diff) { + try { + OutputStream os = new FileOutputStream(outputFile); + + JarDiff.createPatch(args[counter], + args[counter + 1], os, minimal); + os.close(); + } catch (IOException ioe) { + try { + System.out.println(getResources().getString("jardiff.error.create") + " " + ioe); + } catch (MissingResourceException mre) { + } + } + } + else { + try { + OutputStream os = new FileOutputStream(outputFile); + + new JarDiffPatcher().applyPatch( + null, + args[counter], + args[counter + 1], + os); + os.close(); + } catch (IOException ioe) { + try { + System.out.println(getResources().getString("jardiff.error.apply") + " " + ioe); + } catch (MissingResourceException mre) { + } + } + } + System.exit(0); + } + } + showHelp(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffConstants.java b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffConstants.java new file mode 100755 index 0000000000000000000000000000000000000000..4483030b599e8da6e921fc794acfb7c474f389ae --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffConstants.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.jardiff; + +import java.io.*; +import java.util.*; +import java.util.jar.*; +import java.util.zip.*; + +/** + * Constants used by creating patch and applying patch for JarDiff. + * + * @version 1.8, 06/26/03 + */ +public interface JarDiffConstants { + public final String VERSION_HEADER = "version 1.0"; + public final String INDEX_NAME = "META-INF/INDEX.JD"; + public final String REMOVE_COMMAND = "remove"; + public final String MOVE_COMMAND = "move"; +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffPatcher.java b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffPatcher.java new file mode 100755 index 0000000000000000000000000000000000000000..e260ee120c60cc241d07c472ca38337ecc7a63c6 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/JarDiffPatcher.java @@ -0,0 +1,329 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.jardiff; + +import java.io.*; +import java.util.*; +import java.net.URL; +import java.util.jar.*; +import java.util.zip.*; + +/** + * JarDiff is able to create a jar file containing the delta between two + * jar files (old and new). The delta jar file can then be applied to the + * old jar file to reconstruct the new jar file. + * <p> + * Refer to the JNLP spec for details on how this is done. + * + * @version 1.11, 06/26/03 + */ +public class JarDiffPatcher implements JarDiffConstants, Patcher { + private static final int DEFAULT_READ_SIZE = 2048; + private static byte[] newBytes = new byte[DEFAULT_READ_SIZE]; + private static byte[] oldBytes = new byte[DEFAULT_READ_SIZE]; + private static ResourceBundle _resources = JarDiff.getResources(); + + public static ResourceBundle getResources() { + return JarDiff.getResources(); + } + + public void applyPatch(Patcher.PatchDelegate delegate, String oldJarPath, + String jarDiffPath, OutputStream result) throws IOException { + File oldFile = new File(oldJarPath); + File diffFile = new File(jarDiffPath); + JarOutputStream jos = new JarOutputStream(result); + JarFile oldJar = new JarFile(oldFile); + JarFile jarDiff = new JarFile(diffFile); + Set ignoreSet = new HashSet(); + Map renameMap = new HashMap(); + + + determineNameMapping(jarDiff, ignoreSet, renameMap); + + // get all keys in renameMap + Object[] keys = renameMap.keySet().toArray(); + + + // Files to implicit move + Set oldjarNames = new HashSet(); + + Enumeration oldEntries = oldJar.entries(); + if (oldEntries != null) { + while (oldEntries.hasMoreElements()) { + oldjarNames.add(((JarEntry)oldEntries.nextElement()).getName()); + } + } + + // size depends on the three parameters below, which is + // basically the counter for each loop that do the actual + // writes to the output file + // since oldjarNames.size() changes in the first two loop + // below, we need to adjust the size accordingly also when + // oldjarNames.size() changes + double size = oldjarNames.size() + keys.length + jarDiff.size(); + double currentEntry = 0; + + // Handle all remove commands + oldjarNames.removeAll(ignoreSet); + size -= ignoreSet.size(); + + + // Add content from JARDiff + Enumeration entries = jarDiff.entries(); + if (entries != null) { + while (entries.hasMoreElements()) { + JarEntry entry = (JarEntry)entries.nextElement(); + + + + if (!INDEX_NAME.equals(entry.getName())) { + + updateDelegate(delegate, currentEntry, size); + currentEntry++; + + writeEntry(jos, entry, jarDiff); + + // Remove entry from oldjarNames since no implicit + //move is needed + boolean wasInOld = oldjarNames.remove(entry.getName()); + + // Update progress counters. If it was in old, we do + // not need an implicit move, so adjust total size. + if (wasInOld) size--; + + } + else { + // no write is done, decrement size + size--; + } + } + } + + + + // go through the renameMap and apply move for each entry + for (int j = 0; j < keys.length; j++) { + + + + // Apply move <oldName> <newName> command + String newName = (String)keys[j]; + String oldName = (String)renameMap.get(newName); + + // Get source JarEntry + JarEntry oldEntry = oldJar.getJarEntry(oldName); + + if (oldEntry == null) { + String moveCmd = MOVE_COMMAND + oldName + " " + newName; + handleException("jardiff.error.badmove", moveCmd); + } + + // Create dest JarEntry + JarEntry newEntry = new JarEntry(newName); + newEntry.setTime(oldEntry.getTime()); + newEntry.setSize(oldEntry.getSize()); + newEntry.setCompressedSize(oldEntry.getCompressedSize()); + newEntry.setCrc(oldEntry.getCrc()); + newEntry.setMethod(oldEntry.getMethod()); + newEntry.setExtra(oldEntry.getExtra()); + newEntry.setComment(oldEntry.getComment()); + + + updateDelegate(delegate, currentEntry, size); + currentEntry++; + + writeEntry(jos, newEntry, oldJar.getInputStream(oldEntry)); + + // Remove entry from oldjarNames since no implicit + //move is needed + boolean wasInOld = oldjarNames.remove(oldName); + + // Update progress counters. If it was in old, we do + // not need an implicit move, so adjust total size. + if (wasInOld) size--; + + } + + // implicit move + Iterator iEntries = oldjarNames.iterator(); + if (iEntries != null) { + while (iEntries.hasNext()) { + + String name = (String)iEntries.next(); + JarEntry entry = oldJar.getJarEntry(name); + + updateDelegate(delegate, currentEntry, size); + currentEntry++; + + writeEntry(jos, entry, oldJar); + } + } + + updateDelegate(delegate, currentEntry, size); + + jos.finish(); + } + + private void updateDelegate(Patcher.PatchDelegate delegate, double currentSize, double size) { + if (delegate != null) { + delegate.patching((int)(currentSize/size)); + } + } + + private void determineNameMapping(JarFile jarDiff, Set ignoreSet, + Map renameMap) throws IOException { + InputStream is = jarDiff.getInputStream(jarDiff.getEntry(INDEX_NAME)); + + if (is == null) { + handleException("jardiff.error.noindex", null); + + } + LineNumberReader indexReader = new LineNumberReader + (new InputStreamReader(is, "UTF-8")); + String line = indexReader.readLine(); + + if (line == null || !line.equals(VERSION_HEADER)) { + handleException("jardiff.error.badheader", line); + + } + + while ((line = indexReader.readLine()) != null) { + if (line.startsWith(REMOVE_COMMAND)) { + List sub = getSubpaths(line.substring(REMOVE_COMMAND. + length())); + + if (sub.size() != 1) { + handleException("jardiff.error.badremove", line); + + } + ignoreSet.add(sub.get(0)); + } + else if (line.startsWith(MOVE_COMMAND)) { + List sub = getSubpaths(line.substring(MOVE_COMMAND.length())); + + if (sub.size() != 2) { + handleException("jardiff.error.badmove", line); + + } + // target of move should be the key + if (renameMap.put(sub.get(1), sub.get(0)) != null) { + // invalid move - should not move to same target twice + handleException("jardiff.error.badmove", line); + } + } + else if (line.length() > 0) { + handleException("jardiff.error.badcommand", line); + + } + } + } + + private void handleException(String errorMsg, String line) throws IOException { + try { + throw new IOException(getResources().getString(errorMsg) + " " + line); + } catch (MissingResourceException mre) { + System.err.println("Fatal error: " + errorMsg); + new Throwable().printStackTrace(System.err); + System.exit(-1); + } + } + + private List getSubpaths(String path) { + int index = 0; + int length = path.length(); + ArrayList sub = new ArrayList(); + + while (index < length) { + while (index < length && Character.isWhitespace + (path.charAt(index))) { + index++; + } + if (index < length) { + int start = index; + int last = start; + String subString = null; + + while (index < length) { + char aChar = path.charAt(index); + if (aChar == '\\' && (index + 1) < length && + path.charAt(index + 1) == ' ') { + + if (subString == null) { + subString = path.substring(last, index); + } + else { + subString += path.substring(last, index); + } + last = ++index; + } + else if (Character.isWhitespace(aChar)) { + break; + } + index++; + } + if (last != index) { + if (subString == null) { + subString = path.substring(last, index); + } + else { + subString += path.substring(last, index); + } + } + sub.add(subString); + } + } + return sub; + } + + private void writeEntry(JarOutputStream jos, JarEntry entry, + JarFile file) throws IOException { + writeEntry(jos, entry, file.getInputStream(entry)); + } + + private void writeEntry(JarOutputStream jos, JarEntry entry, + InputStream data) throws IOException { + //Create a new ZipEntry to clear the compressed size. 5079423 + jos.putNextEntry(new ZipEntry(entry.getName())); + + // Read the entry + int size = data.read(newBytes); + + while (size != -1) { + jos.write(newBytes, 0, size); + size = data.read(newBytes); + } + data.close(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/Patcher.java b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/Patcher.java new file mode 100755 index 0000000000000000000000000000000000000000..457cb9e4361e48ada2727a053e27445a6578c045 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/Patcher.java @@ -0,0 +1,60 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.jardiff; + +import java.io.IOException; +import java.io.OutputStream; +import java.net.URL; + +/** + * Patcher describes the necessary method to apply and create deltas. + * + * @version 1.8, 01/23/03 + */ +public interface Patcher { + /** + * Applies a patch previously created with <code>createPatch</code>. + * Pass in a delegate to be notified of the status of the patch. + */ + public void applyPatch(PatchDelegate delegate, String oldJarPath, + String deltaPath, OutputStream result) throws IOException; + + /** + * Callback used when patching a file. + */ + public interface PatchDelegate { + public void patching(int percentDone); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/resources/strings.properties b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/resources/strings.properties new file mode 100755 index 0000000000000000000000000000000000000000..e735688fa9925a0d569c9b19cf7d684b82fab1d8 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/jardiff/resources/strings.properties @@ -0,0 +1,41 @@ +# +# Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# -Redistribution of source code must retain the above copyright notice, this +# list of conditions and the following disclaimer. +# +# -Redistribution in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# Neither the name of Oracle nor the names of contributors may +# be used to endorse or promote products derived from this software without +# specific prior written permission. +# +# This software is provided "AS IS," without a warranty of any kind. ALL +# EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING +# ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE +# OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") +# AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE +# AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS +# DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST +# REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, +# INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY +# OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, +# EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. +# +# You acknowledge that this software is not designed, licensed or intended +# for use in the design, construction, operation or maintenance of any +# nuclear facility. +# + +jardiff.error.create=Unable to successfully create +jardiff.error.apply=Unable to successfully apply +jardiff.error.noindex=Invalid jardiff, no index! +jardiff.error.badheader=Invalid jardiff header: +jardiff.error.badremove=Invalid remove command: +jardiff.error.badmove=Invalid move command: +jardiff.error.badcommand=Invalid command: diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadRequest.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadRequest.java new file mode 100755 index 0000000000000000000000000000000000000000..cb46e81002bc11cf0bd57521ab6fcd0b7345f87a --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadRequest.java @@ -0,0 +1,227 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import java.io.File; +import java.util.ArrayList; +import javax.servlet.*; +import javax.servlet.http.*; + +/** + * The DownloadRequest incapsulates all the data in a request + * SQE: We need to address query string + */ +public class DownloadRequest { + // Arguments + private static final String ARG_ARCH = "arch"; + private static final String ARG_OS = "os"; + private static final String ARG_LOCALE = "locale"; + private static final String ARG_VERSION_ID = "version-id"; + private static final String ARG_CURRENT_VERSION_ID = "current-version-id"; + private static final String ARG_PLATFORM_VERSION_ID = "platform-version-id"; + private static final String ARG_KNOWN_PLATFORMS = "known-platforms"; + private static final String TEST_JRE = "TestJRE"; + + private String _path = null; + private String _version = null; + private String _currentVersionId = null; + private String[] _os = null; + private String[] _arch = null; + private String[] _locale = null; + private String[] _knownPlatforms = null; + private String _query = null; + private String _testJRE = null; + private boolean _isPlatformRequest = false; + private ServletContext _context = null; + private String _encoding = null; + + private HttpServletRequest _httpRequest = null; + + // HTTP Compression RFC 2616 : Standard headers + public static final String ACCEPT_ENCODING = "accept-encoding"; + + // Contruct Request object based on HTTP request + public DownloadRequest(HttpServletRequest request) { + this((ServletContext)null, request); + } + + public DownloadRequest(ServletContext context, HttpServletRequest request) { + _context = context; + _httpRequest = request; + _path = request.getRequestURI(); + _encoding = request.getHeader(ACCEPT_ENCODING); + String context_path = request.getContextPath(); + if (context_path != null) _path = _path.substring(context_path.length()); + if (_path == null) _path = request.getServletPath(); // This works for *.<ext> invocations + if (_path == null) _path = "/"; // No path given + _path = _path.trim(); + if (_context != null && !_path.endsWith("/")) { + String realPath = _context.getRealPath(_path); + // fix for 4474021 - getRealPath might returns NULL + if (realPath != null) { + File f = new File(realPath); + if (f != null && f.exists() && f.isDirectory()) { + _path += "/"; + } + } + } + // Append default file for a directory + if (_path.endsWith("/")) _path += "launch.jnlp"; + _version = getParameter(request, ARG_VERSION_ID); + _currentVersionId = getParameter(request, ARG_CURRENT_VERSION_ID); + _os = getParameterList(request, ARG_OS); + _arch = getParameterList(request, ARG_ARCH); + _locale = getParameterList(request, ARG_LOCALE); + _knownPlatforms = getParameterList(request, ARG_KNOWN_PLATFORMS); + String platformVersion = getParameter(request, ARG_PLATFORM_VERSION_ID); + _isPlatformRequest = (platformVersion != null); + if (_isPlatformRequest) _version = platformVersion; + _query = request.getQueryString(); + _testJRE = getParameter(request, TEST_JRE); + } + + /** Returns a DownloadRequest for the currentVersionId, that can be used + * to lookup the existing cached version + */ + private DownloadRequest(DownloadRequest dreq) { + _encoding = dreq._encoding; + _context = dreq._context; + _httpRequest = dreq._httpRequest; + _path = dreq._path; + _version = dreq._currentVersionId; + _currentVersionId = null; + _os = dreq._os; + _arch = dreq._arch; + _locale = dreq._locale; + _knownPlatforms = dreq._knownPlatforms; + _isPlatformRequest = dreq._isPlatformRequest; + _query = dreq._query; + _testJRE = dreq._testJRE; + } + + + private String getParameter(HttpServletRequest req, String key) { + String res = req.getParameter(key); + return (res == null) ? null : res.trim(); + } + + /** Converts a space delimitered string to a list of strings */ + static private String[] getStringList(String str) { + if (str == null) return null; + ArrayList list = new ArrayList(); + int i = 0; + int length = str.length(); + StringBuffer sb = null; + while(i < length) { + char ch = str.charAt(i); + if (ch == ' ') { + // A space was hit. Add string to list + if (sb != null) { + list.add(sb.toString()); + sb = null; + } + } else if (ch == '\\') { + // It is a delimiter. Add next character + if (i + 1 < length) { + ch = str.charAt(++i); + if (sb == null) sb = new StringBuffer(); + sb.append(ch); + } + } else { + if (sb == null) sb = new StringBuffer(); + sb.append(ch); + } + i++; // Next character + } + // Make sure to add the last part to the list too + if (sb != null) { + list.add(sb.toString()); + } + if (list.size() == 0) return null; + String[] results = new String[list.size()]; + return (String[])list.toArray(results); + } + + /* Split parameter at spaces. Convert '\ ' insto a space */ + private String[] getParameterList(HttpServletRequest req, String key) { + String res = req.getParameter(key); + return (res == null) ? null : getStringList(res.trim()); + } + + // Query + public String getPath() { return _path; } + public String getVersion() { return _version; } + public String getCurrentVersionId() { return _currentVersionId; } + public String getQuery() { return _query; } + public String getTestJRE() { return _testJRE; } + public String getEncoding() { return _encoding; } + public String[] getOS() { return _os; } + public String[] getArch() { return _arch; } + public String[] getLocale() { return _locale; } + public String[] getKnownPlatforms() { return _knownPlatforms; } + public boolean isPlatformRequest() { return _isPlatformRequest; } + public HttpServletRequest getHttpRequest() { return _httpRequest; } + + /** Returns a DownloadRequest for the currentVersionId, that can be used + * to lookup the existing cached version + */ + DownloadRequest getFromDownloadRequest() { + return new DownloadRequest(this); + } + + // Debug + public String toString() { + return "DownloadRequest[path=" + _path + + showEntry(" encoding=", _encoding) + + showEntry(" query=", _query) + + showEntry(" TestJRE=", _testJRE) + + showEntry(" version=", _version) + + showEntry(" currentVersionId=", _currentVersionId) + + showEntry(" os=", _os) + + showEntry(" arch=", _arch) + + showEntry(" locale=", _locale) + + showEntry(" knownPlatforms=", _knownPlatforms) + + " isPlatformRequest=" + _isPlatformRequest + "]"; + } + + private String showEntry(String msg, String value) { + if (value == null) return ""; + return msg + value; + } + + private String showEntry(String msg, String[] value) { + if (value == null) return ""; + return msg + java.util.Arrays.asList(value).toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadResponse.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadResponse.java new file mode 100755 index 0000000000000000000000000000000000000000..2b134f372efad84385b873908639446f5c121de0 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/DownloadResponse.java @@ -0,0 +1,296 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import java.io.*; +import java.util.*; +import java.net.URL; +import java.net.URLConnection; +import javax.servlet.http.HttpServletResponse; + +/** + * A class used to encapsulate a file response, and + * factory methods to create some common types. + */ +abstract public class DownloadResponse { + private static final String HEADER_LASTMOD = "Last-Modified"; + private static final String HEADER_JNLP_VERSION = "x-java-jnlp-version-id"; + private static final String JNLP_ERROR_MIMETYPE = "application/x-java-jnlp-error"; + + public static final int STS_00_OK = 0; + public static final int ERR_10_NO_RESOURCE = 10; + public static final int ERR_11_NO_VERSION = 11; + public static final int ERR_20_UNSUP_OS = 20; + public static final int ERR_21_UNSUP_ARCH = 21; + public static final int ERR_22_UNSUP_LOCALE = 22; + public static final int ERR_23_UNSUP_JRE = 23; + public static final int ERR_99_UNKNOWN = 99; + + // HTTP Compression RFC 2616 : Standard headers + public static final String CONTENT_ENCODING = "content-encoding"; + // HTTP Compression RFC 2616 : Standard header for HTTP/Pack200 Compression + public static final String GZIP_ENCODING = "gzip"; + public static final String PACK200_GZIP_ENCODING = "pack200-gzip"; + + public DownloadResponse() { /* do nothing */ } + + public String toString() { return getClass().getName(); } + + /** Post information to an HttpResponse */ + abstract void sendRespond(HttpServletResponse response) throws IOException; + + /** Factory methods for error responses */ + static DownloadResponse getNotFoundResponse() { return new NotFoundResponse(); } + static DownloadResponse getNoContentResponse() { return new NotFoundResponse(); } + static DownloadResponse getJnlpErrorResponse(int jnlpErrorCode) { return new JnlpErrorResponse(jnlpErrorCode); } + + /** Factory method for file download responses */ + + static DownloadResponse getNotModifiedResponse() { + return new NotModifiedResponse(); + } + + static DownloadResponse getHeadRequestResponse(String mimeType, + String versionId, long lastModified, int contentLength) { + return new HeadRequestResponse(mimeType, versionId, lastModified, + contentLength); + } + + static DownloadResponse getFileDownloadResponse(byte[] content, String mimeType, long timestamp, String versionId) { + return new ByteArrayFileDownloadResponse(content, mimeType, versionId, timestamp); + } + + static DownloadResponse getFileDownloadResponse(URL resource, String mimeType, long timestamp, String versionId) { + return new ResourceFileDownloadResponse(resource, mimeType, versionId, timestamp); + } + + static DownloadResponse getFileDownloadResponse(File file, String mimeType, long timestamp, String versionId) { + return new DiskFileDownloadResponse(file, mimeType, versionId, timestamp); + } + + // + // Private classes implementing the various types + // + + static private class NotModifiedResponse extends DownloadResponse { + public void sendRespond(HttpServletResponse response) throws + IOException { + response.sendError(HttpServletResponse.SC_NOT_MODIFIED); + } + } + + static private class NotFoundResponse extends DownloadResponse { + public void sendRespond(HttpServletResponse response) throws IOException { + response.sendError(HttpServletResponse.SC_NOT_FOUND); + } + } + + static private class NoContentResponse extends DownloadResponse { + public void sendRespond(HttpServletResponse response) throws IOException { + response.sendError(HttpServletResponse.SC_NO_CONTENT); + } + } + + static private class HeadRequestResponse extends DownloadResponse { + private String _mimeType; + private String _versionId; + private long _lastModified; + private int _contentLength; + + HeadRequestResponse(String mimeType, String versionId, + long lastModified, int contentLength) { + _mimeType = mimeType; + _versionId = versionId; + _lastModified = lastModified; + _contentLength = contentLength; + } + + /** Post information to an HttpResponse */ + public void sendRespond(HttpServletResponse response) throws + IOException { + // Set header information + response.setContentType(_mimeType); + response.setContentLength(_contentLength); + if (_versionId != null) { + response.setHeader(HEADER_JNLP_VERSION, _versionId); + } + if (_lastModified != 0) { + response.setDateHeader(HEADER_LASTMOD, _lastModified); + } + response.sendError(HttpServletResponse.SC_OK); + } + } + + static public class JnlpErrorResponse extends DownloadResponse { + private String _message; + + public JnlpErrorResponse(int jnlpErrorCode) { + String msg = Integer.toString(jnlpErrorCode); + String dsc = "No description"; + try { + dsc = JnlpDownloadServlet.getResourceBundle().getString("servlet.jnlp.err." + msg); + } catch (MissingResourceException mre) { /* ignore */} + _message = msg + " " + dsc; + } + + public void sendRespond(HttpServletResponse response) throws IOException { + response.setContentType(JNLP_ERROR_MIMETYPE); + PrintWriter pw = response.getWriter(); + pw.println(_message); + }; + + public String toString() { return super.toString() + "[" + _message + "]"; } + } + + static private abstract class FileDownloadResponse extends DownloadResponse { + private String _mimeType; + private String _versionId; + private long _lastModified; + private String _fileName; + + FileDownloadResponse(String mimeType, String versionId, long lastModified) { + _mimeType = mimeType; + _versionId = versionId; + _lastModified = lastModified; + _fileName = null; + } + + FileDownloadResponse(String mimeType, String versionId, long lastModified, String fileName) { + _mimeType = mimeType; + _versionId = versionId; + _lastModified = lastModified; + _fileName = fileName; + } + + + /** Information about response */ + String getMimeType() { return _mimeType; } + String getVersionId() { return _versionId; } + long getLastModified() { return _lastModified; } + abstract int getContentLength() throws IOException; + abstract InputStream getContent() throws IOException; + + /** Post information to an HttpResponse */ + public void sendRespond(HttpServletResponse response) throws IOException { + // Set header information + response.setContentType(getMimeType()); + response.setContentLength(getContentLength()); + if (getVersionId() != null) response.setHeader(HEADER_JNLP_VERSION, getVersionId()); + if (getLastModified() != 0) response.setDateHeader(HEADER_LASTMOD, getLastModified()); + if (_fileName != null) { + + if (_fileName.endsWith(".pack.gz")) { + response.setHeader(CONTENT_ENCODING, PACK200_GZIP_ENCODING ); + } else if (_fileName.endsWith(".gz")) { + response.setHeader(CONTENT_ENCODING, GZIP_ENCODING ); + } else { + response.setHeader(CONTENT_ENCODING, null); + } + } + + // Send contents + InputStream in = getContent(); + OutputStream out = response.getOutputStream(); + try { + byte[] bytes = new byte[32 * 1024]; + int read; + while ((read = in.read(bytes)) != -1) { + out.write(bytes, 0, read); + } + } finally { + if (in != null) in.close(); + } + } + + protected String getArgString() { + long length = 0; + try { + length = getContentLength(); + } catch(IOException ioe) { /* ignore */ } + return "Mimetype=" + getMimeType() + + " VersionId=" + getVersionId() + + " Timestamp=" + new Date(getLastModified()) + + " Length=" + length; + } + } + + static private class ByteArrayFileDownloadResponse extends FileDownloadResponse { + private byte[] _content; + + ByteArrayFileDownloadResponse(byte[] content, String mimeType, String versionId, long lastModified) { + super(mimeType, versionId, lastModified); + _content = content; + } + + int getContentLength() { return _content.length; } + InputStream getContent() { return new ByteArrayInputStream(_content); } + public String toString() { return super.toString() + "[ " + getArgString() + "]"; } + } + + static private class ResourceFileDownloadResponse extends FileDownloadResponse { + URL _url; + + ResourceFileDownloadResponse(URL url, String mimeType, String versionId, long lastModified) { + super(mimeType, versionId, lastModified, url.toString()); + _url= url; + } + + int getContentLength() throws IOException { + return _url.openConnection().getContentLength(); + } + InputStream getContent() throws IOException { + return _url.openConnection().getInputStream(); + } + public String toString() { return super.toString() + "[ " + getArgString() + "]"; } + } + + static private class DiskFileDownloadResponse extends FileDownloadResponse { + private File _file; + + DiskFileDownloadResponse(File file, String mimeType, String versionId, long lastModified) { + super(mimeType, versionId, lastModified, file.getName()); + _file = file; + } + + int getContentLength() throws IOException { + return (int)_file.length(); + } + + InputStream getContent() throws IOException { + return new BufferedInputStream(new FileInputStream(_file)); + } + + public String toString() { return super.toString() + "[ " + getArgString() + "]"; } + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ErrorResponseException.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ErrorResponseException.java new file mode 100755 index 0000000000000000000000000000000000000000..e9dc09a194af2eca64a4e7604eb26f0852b7b016 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ErrorResponseException.java @@ -0,0 +1,51 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; + +/** An exception that holds a DownloadResponse object. + * This exception can be thrown with the content describing + * the message that should be returned in the HTTP respond + */ +public class ErrorResponseException extends Exception { + private DownloadResponse _downloadResponse; + + public ErrorResponseException(DownloadResponse downloadResponse) { + _downloadResponse = downloadResponse; + } + + public DownloadResponse getDownloadResponse() { return _downloadResponse; } + + public String toString() { return _downloadResponse.toString(); } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JarDiffHandler.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JarDiffHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..06f2f975a739129eab46c4c9bf3a49412792aaad --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JarDiffHandler.java @@ -0,0 +1,388 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import java.io.*; +import java.util.*; +import jnlp.sample.jardiff.*; +import javax.servlet.*; +import javax.servlet.http.*; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + +import jnlp.sample.util.VersionString; +import java.net.URL; +/* + * A class that generates and caches information about JarDiff files + * + */ +public class JarDiffHandler { + final private Logger log = Red5LoggerFactory.getLogger(JarDiffHandler.class, "screenshare"); + + // Default size of download buffer + private static final int BUF_SIZE = 32 * 1024; + + // Default JARDiff mime type + private static final String JARDIFF_MIMETYPE = "application/x-java-archive-diff"; + + /** List of all generated JARDiffs */ + private HashMap _jarDiffEntries = null; + + /** Reference to ServletContext and logger object */ + private ServletContext _servletContext = null; + private String _jarDiffMimeType = null; + + /* Contains information about a particular JARDiff entry */ + private static class JarDiffKey implements Comparable{ + private String _name; // Name of file + private String _fromVersionId; // From version + private String _toVersionId; // To version + private boolean _minimal; // True if this is a minimal jardiff + + /** Constructor used to generate a query object */ + public JarDiffKey(String name, String fromVersionId, String toVersionId, boolean minimal) { + _name = name; + _fromVersionId = fromVersionId; + _toVersionId = toVersionId; + _minimal = minimal; + } + + // Query methods + public String getName() { return _name; } + public String getFromVersionId() { return _fromVersionId; } + public String getToVersionId() { return _toVersionId; } + public boolean isMinimal() { return _minimal; } + + // Collection framework interface methods + + public int compareTo(Object o) { + // All non JarDiff entries are less + if (!(o instanceof JarDiffKey)) return -1; + JarDiffKey other = (JarDiffKey)o; + + int n = _name.compareTo(other.getName()); + if (n != 0) return n; + + n = _fromVersionId.compareTo(other.getFromVersionId()); + if (n != 0) return n; + + if (_minimal != other.isMinimal()) return -1; + + return _toVersionId.compareTo(other.getToVersionId()); + } + + public boolean equals(Object o) { + return compareTo(o) == 0; + } + + public int hashCode() { + return _name.hashCode() + + _fromVersionId.hashCode() + + _toVersionId.hashCode(); + } + } + + static private class JarDiffEntry { + private File _jardiffFile; // Location of JARDiff file + + public JarDiffEntry(File jarDiffFile) { + _jardiffFile = jarDiffFile; + } + + public File getJarDiffFile() { return _jardiffFile; } + } + + /** Initialize JarDiff handler */ + public JarDiffHandler(ServletContext servletContext) { + _jarDiffEntries = new HashMap(); + _servletContext = servletContext; + + _jarDiffMimeType = _servletContext.getMimeType("xyz.jardiff"); + if (_jarDiffMimeType == null) _jarDiffMimeType = JARDIFF_MIMETYPE; + } + + /** Returns a JarDiff for the given request */ + public synchronized DownloadResponse getJarDiffEntry(ResourceCatalog catalog, DownloadRequest dreq, JnlpResource res) { + if (dreq.getCurrentVersionId() == null) return null; + + // check whether the request is from javaws 1.0/1.0.1 + // do not generate minimal jardiff if it is from 1.0/1.0.1 + boolean doJarDiffWorkAround = isJavawsVersion(dreq, "1.0*"); + + // First do a lookup to find a match + JarDiffKey key = new JarDiffKey(res.getName(), + dreq.getCurrentVersionId(), + res.getReturnVersionId(), + !doJarDiffWorkAround); + + + JarDiffEntry entry = (JarDiffEntry)_jarDiffEntries.get(key); + // If entry is not found, then the querty has not been made. + if (entry == null) { + if (log.isInfoEnabled()) { + log.info("servlet.log.info.jardiff.gen", + res.getName(), + dreq.getCurrentVersionId(), + res.getReturnVersionId()); + } + File f = generateJarDiff(catalog, dreq, res, doJarDiffWorkAround); + if (f == null) { + log.warn("servlet.log.warning.jardiff.failed", + res.getName(), + dreq.getCurrentVersionId(), + res.getReturnVersionId()); + } + // Store entry in table + entry = new JarDiffEntry(f); + _jarDiffEntries.put(key, entry); + } + + + + // Check for no JarDiff to return + if (entry.getJarDiffFile() == null) { + return null; + } else { + return DownloadResponse.getFileDownloadResponse(entry.getJarDiffFile(), + _jarDiffMimeType, + entry.getJarDiffFile().lastModified(), + res.getReturnVersionId()); + } + } + + + public static boolean isJavawsVersion(DownloadRequest dreq, String version) { + String javawsAgent = "javaws"; + String jwsVer = dreq.getHttpRequest().getHeader("User-Agent"); + + + // check the request is coming from javaws + if (!jwsVer.startsWith("javaws-")) { + // this is the new style User-Agent string + // User-Agent: JNLP/1.0.1 javaws/1.4.2 (b28) J2SE/1.4.2 + StringTokenizer st = new StringTokenizer(jwsVer); + while (st.hasMoreTokens()) { + String verString = st.nextToken(); + int index = verString.indexOf(javawsAgent); + if (index != -1) { + verString = verString.substring(index + javawsAgent.length() + 1); + return VersionString.contains(version, verString); + } + } + return false; + } + + // extract the version id from the download request + int startIndex = jwsVer.indexOf("-"); + + if (startIndex == -1) { + return false; + } + + int endIndex = jwsVer.indexOf("/"); + + if (endIndex == -1 || endIndex < startIndex) { + return false; + } + + String verId = jwsVer.substring(startIndex + 1, endIndex); + + + // check whether the versionString contains the versionId + return VersionString.contains(version, verId); + + } + + /** Download resource to the given file */ + private boolean download(URL target, File file) { + + log.debug("JarDiffHandler: Doing download"); + + boolean ret = true; + boolean delete = false; + // use bufferedstream for better performance + BufferedInputStream in = null; + BufferedOutputStream out = null; + try { + in = new BufferedInputStream(target.openStream()); + out = new BufferedOutputStream(new FileOutputStream(file)); + int read = 0; + int totalRead = 0; + byte[] buf = new byte[BUF_SIZE]; + while ((read = in.read(buf)) != -1) { + out.write(buf, 0, read); + totalRead += read; + } + + log.debug("total read: " + totalRead); + log.debug("Wrote URL " + target.toString() + " to file " + file); + + } catch(IOException ioe) { + + log.debug("Got exception while downloading resource: " + ioe); + + ret = false; + + if (file != null) delete = true; + + } finally { + + try { + in.close(); + in = null; + } catch (IOException ioe) { + log.debug("Got exception while downloading resource: " + ioe); + } + + try { + out.close(); + out = null; + } catch (IOException ioe) { + log.debug("Got exception while downloading resource: " + ioe); + } + + if (delete) { + file.delete(); + } + + } + return ret; + } + + // fix for 4720897 + // if the jar file resides in a war file, download it to a temp dir + // so it can be used to generate jardiff + private String getRealPath(String path) throws IOException{ + + URL fileURL = _servletContext.getResource(path); + + File tempDir = (File)_servletContext.getAttribute("javax.servlet.context.tempdir"); + + // download file into temp dir + if (fileURL != null) { + File newFile = File.createTempFile("temp", ".jar", tempDir); + if (download(fileURL, newFile)) { + String filePath = newFile.getPath(); + return filePath; + } + } + return null; + } + + + private File generateJarDiff(ResourceCatalog catalog, DownloadRequest dreq, JnlpResource res, boolean doJarDiffWorkAround) { + boolean del_old = false; + boolean del_new = false; + + // Lookup up file for request version + DownloadRequest fromDreq = dreq.getFromDownloadRequest(); + try { + JnlpResource fromRes = catalog.lookupResource(fromDreq); + + /* Get file locations */ + String newFilePath = _servletContext.getRealPath(res.getPath()); + String oldFilePath = _servletContext.getRealPath(fromRes.getPath()); + + // fix for 4720897 + if (newFilePath == null) { + newFilePath = getRealPath(res.getPath()); + if (newFilePath != null) del_new = true; + } + + if (oldFilePath == null) { + oldFilePath = getRealPath(fromRes.getPath()); + if (oldFilePath != null) del_old = true; + } + + if (newFilePath == null || oldFilePath == null) { + return null; + } + + // Create temp. file to store JarDiff file in + File tempDir = (File)_servletContext.getAttribute("javax.servlet.context.tempdir"); + + // fix for 4653036: JarDiffHandler() should use javax.servlet.context.tempdir to store the jardiff + File outputFile = File.createTempFile("jnlp", ".jardiff", tempDir); + + log.debug("Generating Jardiff between " + oldFilePath + " and " + + newFilePath + " Store in " + outputFile); + + // Generate JarDiff + OutputStream os = new FileOutputStream(outputFile); + + JarDiff.createPatch(oldFilePath, newFilePath, os, !doJarDiffWorkAround); + os.close(); + + try { + + // Check that Jardiff is smaller, or return null + if (outputFile.length() >= (new File(newFilePath).length())) { + log.debug("JarDiff discarded - since it is bigger"); + return null; + } + + // Check that Jardiff is smaller than the packed version of + // the new file, if the file exists at all + File newFilePacked = new File(newFilePath + ".pack.gz"); + if (newFilePacked.exists()) { + log.debug("generated jardiff size: " + outputFile.length()); + log.debug("packed requesting file size: " + newFilePacked.length()); + if (outputFile.length() >= newFilePacked.length()) { + log.debug("JarDiff discarded - packed version of requesting file is smaller"); + return null; + } + } + + log.debug("JarDiff generation succeeded"); + return outputFile; + + } finally { + // delete the temporarily downloaded file + if (del_new) { + new File(newFilePath).delete(); + } + + if (del_old) { + new File(oldFilePath).delete(); + } + } + } catch(IOException ioe) { + log.debug("Failed to genereate jardiff", ioe); + return null; + } catch(ErrorResponseException ere) { + log.debug("Failed to genereate jardiff", ere); + return null; + } + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpDownloadServlet.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpDownloadServlet.java new file mode 100755 index 0000000000000000000000000000000000000000..ad276853c978e6f747220fe8bead562704b898ac --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpDownloadServlet.java @@ -0,0 +1,249 @@ + + +package jnlp.sample.servlet; + +import java.io.*; +import java.util.*; +import java.net.*; +import javax.servlet.*; +import javax.servlet.http.*; +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import org.springframework.context.ApplicationContext; +import org.springframework.web.context.WebApplicationContext; + +/** + * This Servlet class is an implementation of JNLP Specification's + * Download Protocols. + * + * All requests to this servlet is in the form of HTTP GET commands. + * The parameters that are needed are: + * <ul> + * <li><code>arch</code>, + * <li><code>os</code>, + * <li><code>locale</code>, + * <li><code>version-id</code> or <code>platform-version-id</code>, + * <li><code>current-version-id</code>, + * <li>code>known-platforms</code> + * </ul> + * <p> + * + * @version 1.8 01/23/03 + */ +public class JnlpDownloadServlet extends HttpServlet { + final private Logger log = Red5LoggerFactory.getLogger(JnlpDownloadServlet.class, "screenshare"); + + // Localization + private static ResourceBundle _resourceBundle = null; + + // Servlet configuration + private static final String PARAM_JNLP_EXTENSION = "jnlp-extension"; + private static final String PARAM_JAR_EXTENSION = "jar-extension"; + + private JnlpFileHandler _jnlpFileHandler = null; + private JarDiffHandler _jarDiffHandler = null; + private ResourceCatalog _resourceCatalog = null; + + /** Initialize servlet */ + public void init(ServletConfig config) throws ServletException { + super.init(config); + + // Get extension from Servlet configuration, or use default + JnlpResource.setDefaultExtensions( + config.getInitParameter(PARAM_JNLP_EXTENSION), + config.getInitParameter(PARAM_JAR_EXTENSION)); + + _jnlpFileHandler = new JnlpFileHandler(config.getServletContext()); + _jarDiffHandler = new JarDiffHandler(config.getServletContext()); + _resourceCatalog = new ResourceCatalog(config.getServletContext()); + } + + public static synchronized ResourceBundle getResourceBundle() { + if (_resourceBundle == null) { + _resourceBundle = ResourceBundle.getBundle("jnlp/sample/servlet/resources/strings"); + } + return _resourceBundle; + } + + + public void doHead(HttpServletRequest request, HttpServletResponse response) + throws ServletException, IOException { + handleRequest(request, response, true); + } + + /** We handle get requests too - eventhough the spec. only requeres POST requests */ + public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { + handleRequest(request, response, false); + } + + private void handleRequest(HttpServletRequest request, + HttpServletResponse response, boolean isHead) throws IOException { + + String requestStr = request.getRequestURI(); + if (request.getQueryString() != null) requestStr += "?" + request.getQueryString().trim(); + + // Parse HTTP request + DownloadRequest dreq = new DownloadRequest(getServletContext(), request); + if (log.isInfoEnabled()) { + log.info("servlet.log.info.request", requestStr); + log.info("servlet.log.info.useragent", request.getHeader("User-Agent")); + } + if (log.isDebugEnabled()) { + log.debug(dreq.toString()); + } + + long ifModifiedSince = request.getDateHeader("If-Modified-Since"); + + // Check if it is a valid request + try { + // Check if the request is valid + validateRequest(dreq); + + // Decide what resource to return + JnlpResource jnlpres = locateResource(dreq); + log.debug("JnlpResource: " + jnlpres); + + + if (log.isInfoEnabled()) { + log.info("servlet.log.info.goodrequest", jnlpres.getPath()); + } + + DownloadResponse dres = null; + + if (isHead) { + + int cl = + jnlpres.getResource().openConnection().getContentLength(); + + // head request response + dres = DownloadResponse.getHeadRequestResponse( + jnlpres.getMimeType(), jnlpres.getVersionId(), + jnlpres.getLastModified(), cl); + + } else if (ifModifiedSince != -1 && + (ifModifiedSince / 1000) >= + (jnlpres.getLastModified() / 1000)) { + // We divide the value returned by getLastModified here by 1000 + // because if protocol is HTTP, last 3 digits will always be + // zero. However, if protocol is JNDI, that's not the case. + // so we divide the value by 1000 to remove the last 3 digits + // before comparison + + // return 304 not modified if possible + log.debug("return 304 Not modified"); + dres = DownloadResponse.getNotModifiedResponse(); + + } else { + + // Return selected resource + dres = constructResponse(jnlpres, dreq); + } + + dres.sendRespond(response); + + } catch(ErrorResponseException ere) { + if (log.isInfoEnabled()) { + log.info("servlet.log.info.badrequest", requestStr); + } + if (log.isDebugEnabled()) { + log.debug("Response: "+ ere.toString()); + } + // Return response from exception + ere.getDownloadResponse().sendRespond(response); + } catch(Throwable e) { + log.error("servlet.log.fatal.internalerror", e); + response.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR); + } + } + + /** Make sure that it is a valid request. This is also the place to implement the + * reverse IP lookup + */ + private void validateRequest(DownloadRequest dreq) throws ErrorResponseException { + String path = dreq.getPath(); + if (path.endsWith(ResourceCatalog.VERSION_XML_FILENAME) || + path.indexOf("__") != -1 ) { + throw new ErrorResponseException(DownloadResponse.getNoContentResponse()); + } + } + + /** Interprets the download request and convert it into a resource that is + * part of the Web Archive. + */ + private JnlpResource locateResource(DownloadRequest dreq) throws IOException, ErrorResponseException { + if (dreq.getVersion() == null) { + return handleBasicDownload(dreq); + } else { + return handleVersionRequest(dreq); + } + } + + private JnlpResource handleBasicDownload(DownloadRequest dreq) throws ErrorResponseException, IOException { + log.debug("Basic Protocol lookup"); + // Do not return directory names for basic protocol + if (dreq.getPath() == null || dreq.getPath().endsWith("/")) { + throw new ErrorResponseException(DownloadResponse.getNoContentResponse()); + } + // Lookup resource + JnlpResource jnlpres = new JnlpResource(getServletContext(), dreq.getPath()); + if (!jnlpres.exists()) { + throw new ErrorResponseException(DownloadResponse.getNoContentResponse()); + } + return jnlpres; + } + + private JnlpResource handleVersionRequest(DownloadRequest dreq) throws IOException, ErrorResponseException { + log.debug("Version-based/Extension based lookup"); + return _resourceCatalog.lookupResource(dreq); + } + + /** Given a DownloadPath and a DownloadRequest, it constructs the data stream to return + * to the requester + */ + private DownloadResponse constructResponse(JnlpResource jnlpres, DownloadRequest dreq) throws IOException { + String path = jnlpres.getPath(); + if (jnlpres.isJnlpFile()) { + // It is a JNLP file. It need to be macro-expanded, so it is handled differently + boolean supportQuery = JarDiffHandler.isJavawsVersion(dreq, "1.5+"); + log.debug("SupportQuery in Href: " + supportQuery); + + // only support query string in href for 1.5 and above + if (supportQuery) { + return _jnlpFileHandler.getJnlpFileEx(jnlpres, dreq); + } else { + return _jnlpFileHandler.getJnlpFile(jnlpres, dreq); + } + } + + // Check if a JARDiff can be returned + if (dreq.getCurrentVersionId() != null && jnlpres.isJarFile()) { + DownloadResponse response = _jarDiffHandler.getJarDiffEntry(_resourceCatalog, dreq, jnlpres); + if (response != null) { + log.info("servlet.log.info.jardiff.response"); + return response; + } + } + + // check and see if we can use pack resource + JnlpResource jr = new JnlpResource(getServletContext(), + jnlpres.getName(), + jnlpres.getVersionId(), + jnlpres.getOSList(), + jnlpres.getArchList(), + jnlpres.getLocaleList(), + jnlpres.getPath(), + jnlpres.getReturnVersionId(), + dreq.getEncoding()); + + log.debug("Real resource returned: " + jr); + + // Return WAR file resource + return DownloadResponse.getFileDownloadResponse(jr.getResource(), + jr.getMimeType(), + jr.getLastModified(), + jr.getReturnVersionId()); + } + + +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpFileHandler.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpFileHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..543866d9bbd3d3952c4fd978acdd893107374092 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpFileHandler.java @@ -0,0 +1,495 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; + +import java.util.*; +import java.util.regex.*; +import java.net.*; +import java.io.*; +import javax.servlet.*; +import javax.servlet.http.*; +import javax.xml.parsers.*; +import org.xml.sax.*; +import javax.xml.transform.*; +import javax.xml.transform.dom.DOMSource; +import javax.xml.transform.stream.StreamResult; + +import org.bigbluebutton.app.screenshare.ScreenShareInfo; +import org.bigbluebutton.app.screenshare.ScreenShareInfoResponse; +import org.bigbluebutton.app.screenshare.server.servlet.JnlpConfigurator; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import org.springframework.context.ApplicationContext; +import org.springframework.web.context.WebApplicationContext; +import org.w3c.dom.*; + +/* The JNLP file handler implements a class that keeps + * track of JNLP files and their specializations + */ +public class JnlpFileHandler { + final private Logger log = Red5LoggerFactory.getLogger(JnlpFileHandler.class, "screenshare"); + + private static final String JNLP_MIME_TYPE = "application/x-java-jnlp-file"; + private static final String HEADER_LASTMOD = "Last-Modified"; + + private ServletContext _servletContext; + private HashMap<String, JnlpFileEntry> _jnlpFiles = null; + + private boolean hasConfigurator = false; + private JnlpConfigurator configurator = null; + + /** Initialize JnlpFileHandler for the specific ServletContext */ + public JnlpFileHandler(ServletContext servletContext) { + _servletContext = servletContext; + _jnlpFiles = new HashMap<String, JnlpFileEntry>(); + } + + private static class JnlpFileEntry { + // Response + DownloadResponse _response; + // Keeps track of cache is out of date + private long _lastModified; + + // Constructor + JnlpFileEntry(DownloadResponse response, long lastmodfied) { + _response = response; + _lastModified = lastmodfied; + } + + public DownloadResponse getResponse() { return _response; } + long getLastModified() { return _lastModified; } + } + + /* Main method to lookup an entry */ + public synchronized DownloadResponse getJnlpFile(JnlpResource jnlpres, DownloadRequest dreq) + throws IOException { + log.debug("In getJnlpFile"); + + String path = jnlpres.getPath(); + URL resource = jnlpres.getResource(); + long lastModified = jnlpres.getLastModified(); + + log.debug("lastModified: " + lastModified + " " + new Date(lastModified)); + if (lastModified == 0) { + log.warn("servlet.log.warning.nolastmodified", path); + } + + // fix for 4474854: use the request URL as key to look up jnlp file + // in hash map + String reqUrl = HttpUtils.getRequestURL(dreq.getHttpRequest()).toString(); + + // Read information from WAR file + long timeStamp = new java.util.Date().getTime();; + String mimeType = _servletContext.getMimeType(path); + if (mimeType == null) mimeType = JNLP_MIME_TYPE; + + StringBuffer jnlpFileTemplate = new StringBuffer(); + URLConnection conn = resource.openConnection(); + BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream(), "UTF-8")); + String line = br.readLine(); + if (line != null && line.startsWith("TS:")) { + timeStamp = parseTimeStamp(line.substring(3)); + log.debug("Timestamp: " + timeStamp + " " + new Date(timeStamp)); + if (timeStamp == 0) { + log.warn("servlet.log.warning.notimestamp", path); + timeStamp = lastModified; + } + line = br.readLine(); + } + while(line != null) { + jnlpFileTemplate.append(line); + line = br.readLine(); + } + + if (! hasConfigurator) { + configurator = getConfigurator(); + if (configurator != null) hasConfigurator = true; + } + + String jnlpFileContent = specializeJnlpTemplate(dreq.getHttpRequest(), path, jnlpFileTemplate.toString()); + + // Convert to bytes as a UTF-8 encoding + byte[] byteContent = jnlpFileContent.getBytes("UTF-8"); + + timeStamp = new java.util.Date().getTime(); + + // Create entry + DownloadResponse resp = DownloadResponse.getFileDownloadResponse(byteContent, + mimeType, + timeStamp, + jnlpres.getReturnVersionId()); + + log.debug("JNLP: mime=[" + mimeType + "] timestamp=[" + timeStamp + "] version=[" + jnlpres.getReturnVersionId() + "]"); + return resp; + } + + /* Main method to lookup an entry (NEW for JavaWebStart 1.5+) */ + public synchronized DownloadResponse getJnlpFileEx(JnlpResource jnlpres, DownloadRequest dreq) + throws IOException { + + log.debug("In getJnlpFileEx"); + + String path = jnlpres.getPath(); + URL resource = jnlpres.getResource(); + long lastModified = jnlpres.getLastModified(); + + + log.debug("lastModified: " + lastModified + " " + new Date(lastModified)); + if (lastModified == 0) { + log.warn("servlet.log.warning.nolastmodified", path); + } + + if (! hasConfigurator) { + configurator = getConfigurator(); + if (configurator != null) hasConfigurator = true; + } + + // fix for 4474854: use the request URL as key to look up jnlp file + // in hash map + String reqUrl = HttpUtils.getRequestURL(dreq.getHttpRequest()).toString(); + // SQE: To support query string, we changed the hash key from Request URL to (Request URL + query string) + if (dreq.getQuery() != null) + reqUrl += dreq.getQuery(); + + // Read information from WAR file + long timeStamp = lastModified; + String mimeType = _servletContext.getMimeType(path); + if (mimeType == null) mimeType = JNLP_MIME_TYPE; + + StringBuffer jnlpFileTemplate = new StringBuffer(); + URLConnection conn = resource.openConnection(); + BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream(), "UTF-8")); + String line = br.readLine(); + if (line != null && line.startsWith("TS:")) { + timeStamp = parseTimeStamp(line.substring(3)); + log.debug("Timestamp: " + timeStamp + " " + new Date(timeStamp)); + if (timeStamp == 0) { + log.warn("servlet.log.warning.notimestamp", path); + timeStamp = lastModified; + } + line = br.readLine(); + } + while(line != null) { + jnlpFileTemplate.append(line); + line = br.readLine(); + } + + String jnlpFileContent = specializeJnlpTemplate(dreq.getHttpRequest(), path, jnlpFileTemplate.toString()); + + /* SQE: We need to add query string back to href in jnlp file. We also need to handle JRE requirement for + * the test. We reconstruct the xml DOM object, modify the value, then regenerate the jnlpFileContent. + */ + String query = dreq.getQuery(); + String testJRE = dreq.getTestJRE(); + log.debug("Double check query string: " + query); + // For backward compatibility: Always check if the href value exists. + // Bug 4939273: We will retain the jnlp template structure and will NOT add href value. Above old + // approach to always check href value caused some test case not run. + if (query != null) { + byte [] cb = jnlpFileContent.getBytes("UTF-8"); + ByteArrayInputStream bis = new ByteArrayInputStream(cb); + try { + DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); + DocumentBuilder builder = factory.newDocumentBuilder(); + Document document = builder.parse(bis); + if (document != null && document.getNodeType() == Node.DOCUMENT_NODE) { + boolean modified = false; + Element root = document.getDocumentElement(); + + if (root.hasAttribute("href") && query != null) { + String href = root.getAttribute("href"); + root.setAttribute("href", href + "?" + query); + modified = true; + } + // Update version value for j2se tag + if (testJRE != null) { + NodeList j2seNL = root.getElementsByTagName("j2se"); + if (j2seNL != null) { + Element j2se = (Element) j2seNL.item(0); + String ver = j2se.getAttribute("version"); + if (ver.length() > 0) { + j2se.setAttribute("version", testJRE); + modified = true; + } + } + } + TransformerFactory tFactory = TransformerFactory.newInstance(); + Transformer transformer = tFactory.newTransformer(); + DOMSource source = new DOMSource(document); + StringWriter sw = new StringWriter(); + StreamResult result = new StreamResult(sw); + transformer.transform(source, result); + jnlpFileContent = sw.toString(); + log.debug("Converted jnlpFileContent: " + jnlpFileContent); + // Since we modified the file on the fly, we always update the timestamp value with current time + if (modified) { + timeStamp = new java.util.Date().getTime(); + log.debug("Last modified on the fly: " + timeStamp); + } + } + } catch (Exception e) { + log.debug(e.toString(), e); + } + } + + // Convert to bytes as a UTF-8 encoding + byte[] byteContent = jnlpFileContent.getBytes("UTF-8"); + + // Create entry + DownloadResponse resp = DownloadResponse.getFileDownloadResponse(byteContent, + mimeType, + timeStamp, + jnlpres.getReturnVersionId()); + + return resp; + } + + /* This method performs the following substituations + * $$name + * $$codebase + * $$context + */ + private String specializeJnlpTemplate(HttpServletRequest request, String respath, String jnlpTemplate) { + log.debug("Query string = [" + request.getQueryString().trim() + "]"); + + String errorMessage = "NO_ERRORS"; + String authParam = request.getParameter("authToken"); + String authToken = "unknown"; + if (authParam != null && authParam != "") { + authToken = authParam.trim(); + } else { + errorMessage = "MISSING_AUTHTOKEN"; + } + + String fullScreen = request.getParameter("fullScreen"); + if (fullScreen != null && fullScreen != "") { + fullScreen = fullScreen.trim(); + } else { + errorMessage = "MISSING_FULLSCREEN"; + } + + String meetingId = request.getParameter("meetingId"); + if (meetingId != null && meetingId != "") { + meetingId = meetingId.trim(); + } else { + errorMessage = "MISSING_MEETINGID"; + } + + ScreenShareInfo sInfo = configurator.getScreenShareInfo(meetingId, authToken); + String publishUrl = "unknown"; + String streamId = "unknown"; + if (sInfo == null) { + errorMessage = "ERROR_GETTING_INFO_USING_TOKEN"; + } else { + publishUrl = sInfo.publishUrl; + streamId = sInfo.streamId; + } + + String jnlpUrl = configurator.getJnlpUrl(); + + String codecOptions = configurator.getCodecOptions(); + log.debug("Codec Options = [" + codecOptions + "]"); + + String urlprefix = getUrlPrefix(request); + int idx = respath.lastIndexOf('/'); // + String name = respath.substring(idx + 1); // Exclude / + String codebase = respath.substring(0, idx + 1); // Include / + jnlpTemplate = substitute(jnlpTemplate, "$$name", name); + jnlpTemplate = substitute(jnlpTemplate, "$$jnlpUrl", jnlpUrl); + jnlpTemplate = substitute(jnlpTemplate, "$$serverUrl", jnlpUrl); + jnlpTemplate = substitute(jnlpTemplate, "$$publishUrl", publishUrl); + jnlpTemplate = substitute(jnlpTemplate, "$$fullScreen", fullScreen); + jnlpTemplate = substitute(jnlpTemplate, "$$meetingId", meetingId); + jnlpTemplate = substitute(jnlpTemplate, "$$streamId", streamId); + jnlpTemplate = substitute(jnlpTemplate, "$$codecOptions", codecOptions); + jnlpTemplate = substitute(jnlpTemplate, "$$errorMessage", errorMessage); + // fix for 5039951: Add $$hostname macro + jnlpTemplate = substitute(jnlpTemplate, "$$hostname", request.getServerName()); + jnlpTemplate = substitute(jnlpTemplate, "$$codebase", urlprefix + request.getContextPath() + codebase); + jnlpTemplate = substitute(jnlpTemplate, "$$context", urlprefix + request.getContextPath()); + // fix for 6256326: add $$site macro to sample jnlp servlet + jnlpTemplate = substitute(jnlpTemplate, "$$site", urlprefix); + + + log.debug(jnlpTemplate); + return jnlpTemplate; + } + + // This code is heavily inspired by the stuff in HttpUtils.getRequestURL + private String getUrlPrefix(HttpServletRequest req) { + StringBuffer url = new StringBuffer(); + String scheme = req.getScheme(); + int port = req.getServerPort(); + url.append(scheme); // http, https + url.append("://"); + url.append(req.getServerName()); + if ((scheme.equals("http") && port != 80) + || (scheme.equals("https") && port != 443)) { + url.append(':'); + url.append(req.getServerPort()); + } + return url.toString(); + } + + private String substitute(String target, String key, String value) { + int start = 0; + do { + int idx = target.indexOf(key, start); + if (idx == -1) return target; + target = target.substring(0, idx) + value + target.substring(idx + key.length()); + start = idx + value.length(); + } while(true); + } + + /** Parses a ISO 8601 Timestamp. The format of the timestamp is: + * + * YYYY-MM-DD hh:mm:ss or YYYYMMDDhhmmss + * + * Hours (hh) is in 24h format. ss are optional. Time are by default relative + * to the current timezone. Timezone information can be specified + * by: + * + * - Appending a 'Z', e.g., 2001-12-19 12:00Z + * - Appending +hh:mm, +hhmm, +hh, -hh:mm -hhmm, -hh to + * indicate that the locale timezone used is either the specified + * amound before or after GMT. For example, + * + * 12:00Z = 13:00+1:00 = 0700-0500 + * + * The method returns 0 if it cannot pass the string. Otherwise, it is + * the number of milliseconds size sometime in 1969. + */ + private long parseTimeStamp(String timestamp) { + int YYYY = 0; + int MM = 0; + int DD = 0; + int hh = 0; + int mm = 0; + int ss = 0; + + timestamp = timestamp.trim(); + try { + // Check what format is used + if (matchPattern("####-##-## ##:##", timestamp)) { + YYYY = getIntValue(timestamp, 0, 4); + MM = getIntValue(timestamp, 5, 7); + DD = getIntValue(timestamp, 8, 10); + hh = getIntValue(timestamp, 11, 13); + mm = getIntValue(timestamp, 14, 16); + timestamp = timestamp.substring(16); + if (matchPattern(":##", timestamp)) { + ss = getIntValue(timestamp, 1, 3); + timestamp = timestamp.substring(3); + } + } else if (matchPattern("############", timestamp)) { + YYYY = getIntValue(timestamp, 0, 4); + MM = getIntValue(timestamp, 4, 6); + DD = getIntValue(timestamp, 6, 8); + hh = getIntValue(timestamp, 8, 10); + mm = getIntValue(timestamp, 10, 12); + timestamp = timestamp.substring(12); + if (matchPattern("##", timestamp)) { + ss = getIntValue(timestamp, 0, 2); + timestamp = timestamp.substring(2); + } + } else { + // Unknown format + return 0; + } + } catch(NumberFormatException e) { + // Bad number + return 0; + } + + String timezone = null; + // Remove timezone information + timestamp = timestamp.trim(); + if (timestamp.equalsIgnoreCase("Z")) { + timezone ="GMT"; + } else if (timestamp.startsWith("+") || timestamp.startsWith("-")) { + timezone = "GMT" + timestamp; + } + + if (timezone == null) { + // Date is relative to current locale + Calendar cal = Calendar.getInstance(); + cal.set(YYYY, MM - 1, DD, hh, mm, ss); + return cal.getTime().getTime(); + } else { + // Date is relative to a timezone + Calendar cal = Calendar.getInstance(TimeZone.getTimeZone(timezone)); + cal.set(YYYY, MM - 1, DD, hh, mm, ss); + return cal.getTime().getTime(); + } + } + + private int getIntValue(String key, int start, int end) { + return Integer.parseInt(key.substring(start, end)); + } + + private boolean matchPattern(String pattern, String key) { + // Key must be longer than pattern + if (key.length() < pattern.length()) return false; + for(int i = 0; i < pattern.length(); i++) { + char format = pattern.charAt(i); + char ch = key.charAt(i); + if (!((format == '#' && Character.isDigit(ch)) || (format == ch))) { + return false; + } + } + return true; + } + + private JnlpConfigurator getConfigurator() { + JnlpConfigurator jnlpConfigurator = null; + if (_servletContext != null) { + //Grab a reference to the application context + ApplicationContext appCtx = (ApplicationContext) _servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE); + + //Get the bean holding the parameter + jnlpConfigurator = (JnlpConfigurator) appCtx.getBean("jnlpConfigurator"); + if (jnlpConfigurator != null) { + log.debug("JnlpConfigurator initialized."); + } else { + log.error("Failed to initialize JnlpConfigurator."); + } + } else { + log.error("Failed to initialize JnlpConfigurator. Context is undefined."); + } + + return jnlpConfigurator; + } + +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpResource.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpResource.java new file mode 100755 index 0000000000000000000000000000000000000000..af585c140a00a2e0d1bf8559284145885d453ecf --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/JnlpResource.java @@ -0,0 +1,266 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import javax.servlet.ServletContext; +import java.net.URL; +import java.io.File; +import java.io.IOException; +import java.net.URLConnection; +import java.util.*; + +/** + * A JnlpResource encapsulate the information about a resource that is + * needed to process a JNLP Download Request. + * + * The pattern matching arguments are: name, version-id, os, arch, and locale. + * + * The outgoing arguments are: + * - path to resource in (WAR File) + * - product version-id (Version-id to return or null. Typically same as version-id above) + * - mime-type for content + * - lastModified date of WAR file resource + * + */ +public class JnlpResource { + private static final String JNLP_MIME_TYPE = "application/x-java-jnlp-file"; + private static final String JAR_MIME_TYPE = "application/x-java-archive"; + + private static final String JAR_MIME_TYPE_NEW = "application/java-archive"; + + // Default extension for the JNLP file + private static final String JNLP_EXTENSION = ".jnlp"; + private static final String JAR_EXTENSION = ".jar"; + + private static String _jnlpExtension = JNLP_EXTENSION; + private static String _jarExtension = JAR_EXTENSION; + + public static void setDefaultExtensions(String jnlpExtension, String jarExtension) { + if (jnlpExtension != null && jnlpExtension.length() > 0) { + if (!jnlpExtension.startsWith(".")) jnlpExtension = "." + jnlpExtension; + _jnlpExtension = jnlpExtension; + } + if (jarExtension != null && jarExtension.length() > 0) { + if (!jarExtension .startsWith(".")) jarExtension = "." + jarExtension ; + _jarExtension = jarExtension; + } + } + + /* Pattern matching arguments */ + private String _name; // Name of resource with path (this is the same as path for non-version based) + private String _versionId; // Version-id for resource, or null if none + private String[] _osList; // List of OSes for which resource should be returned + private String[] _archList; // List of architectures for which the resource should be returned + private String[] _localeList; // List of locales for which the resource should be returned + /* Information used for reply */ + private String _path; // Path to resource in WAR file (unique) + private URL _resource; // URL to resource in WAR file (unique - same as above really) + private long _lastModified; // Last modified in WAR file + private String _mimeType; // Mime-type for resource + private String _returnVersionId; // Version Id to return + private String _encoding; // Accept encoding + + public JnlpResource(ServletContext context, String path) { + this(context, null, null, null, null, null, path, null); + } + + public JnlpResource(ServletContext context, + String name, + String versionId, + String[] osList, + String[] archList, + String[] localeList, + String path, + String returnVersionId) { + this(context, name, versionId, osList, archList, localeList, path, + returnVersionId, null); + } + + public JnlpResource(ServletContext context, + String name, + String versionId, + String[] osList, + String[] archList, + String[] localeList, + String path, + String returnVersionId, + String encoding) { + // Matching arguments + _encoding = encoding; + _name = name; + _versionId = versionId; + _osList = osList; + _archList = archList; + _localeList = localeList; + + _returnVersionId = returnVersionId; + + /* Check for existance and get last modified timestamp */ + try { + String orig_path = path.trim(); + String search_path = orig_path; + _resource = context.getResource(orig_path); + _mimeType = getMimeType(context, orig_path); + if (_resource != null) { + + boolean found = false; + // pack200 compression + if (encoding != null && _mimeType != null && + (_mimeType.compareTo(JAR_MIME_TYPE) == 0 || _mimeType.compareTo(JAR_MIME_TYPE_NEW) == 0) && + encoding.toLowerCase().indexOf(DownloadResponse.PACK200_GZIP_ENCODING) > -1){ + search_path = orig_path + ".pack.gz"; + _resource = context.getResource(search_path); + // Get last modified time + if (_resource != null) { + _lastModified = getLastModified(context, _resource, search_path); + if (_lastModified != 0) { + _path = search_path; + found = true; + } else { + _resource = null; + } + } + } + + // gzip compression + if (found == false && encoding != null && + encoding.toLowerCase().indexOf(DownloadResponse.GZIP_ENCODING) > -1){ + search_path = orig_path + ".gz"; + _resource = context.getResource(search_path); + // Get last modified time + if (_resource != null) { + _lastModified = getLastModified(context, _resource, search_path); + if (_lastModified != 0) { + _path = search_path; + found = true; + } else { + _resource = null; + } + } + } + + if (found == false) { + // no compression + search_path = orig_path; + _resource = context.getResource(search_path); + // Get last modified time + if (_resource != null) { + _lastModified = getLastModified(context, _resource, search_path); + if (_lastModified != 0) { + _path = search_path; + found = true; + } else { + _resource = null; + } + } + } + } + } catch(IOException ioe) { + _resource = null; + } + } + + long getLastModified(ServletContext context, URL resource, String path) { + long lastModified = 0; + URLConnection conn; + try { + // Get last modified time + conn = resource.openConnection(); + lastModified = conn.getLastModified(); + } catch (Exception e) { + // do nothing + } + + if (lastModified == 0) { + // Arguably a bug in the JRE will not set the lastModified for file URLs, and + // always return 0. This is a workaround for that problem. + String filepath = context.getRealPath(path); + if (filepath != null) { + File f = new File(filepath); + if (f.exists()) { + lastModified = f.lastModified(); + } + } + } + return lastModified; + } + + /* Get resource specific attributes */ + public String getPath() { return _path; } + public URL getResource() { return _resource; } + public String getMimeType() { return _mimeType; } + public long getLastModified() { return _lastModified; } + public boolean exists() { return _resource != null; } + public boolean isJnlpFile() { return _path.endsWith(_jnlpExtension); } + public boolean isJarFile() { return _path.endsWith(_jarExtension); } + + /* Get JNLP version specific attributes */ + public String getName() { return _name; } + public String getVersionId() { return _versionId; } + public String[] getOSList() { return _osList; } + public String[] getArchList() { return _archList; } + public String[] getLocaleList() { return _localeList; } + public String getReturnVersionId() { return _returnVersionId; } + + private String getMimeType(ServletContext context, String path) { + String mimeType = context.getMimeType(path); + if (mimeType != null) return mimeType; + if (path.endsWith(_jnlpExtension)) return JNLP_MIME_TYPE; + if (path.endsWith(_jarExtension)) return JAR_MIME_TYPE; + return "application/unknown"; + } + + /** Print info about an entry */ + public String toString() { + return "JnlpResource[WAR Path: " + _path + + showEntry(" versionId=",_versionId) + + showEntry(" name=", _name) + + " lastModified=" + new Date(_lastModified) + + showEntry(" osList=", _osList) + + showEntry(" archList=", _archList) + + showEntry(" localeList=", _localeList) + "]" + + showEntry(" returnVersionId=", _returnVersionId) + "]"; + + } + + private String showEntry(String msg, String value) { + if (value == null) return ""; + return msg + value; + } + + private String showEntry(String msg, String[] value) { + if (value == null) return ""; + return msg + java.util.Arrays.asList(value).toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/Logger.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/Logger.java new file mode 100755 index 0000000000000000000000000000000000000000..cdb1d4d8cf6c891fb9ea33dca5d62a1e2ae8870c --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/Logger.java @@ -0,0 +1,200 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import java.text.MessageFormat; +import java.util.*; +import java.io.*; +import javax.servlet.*; + +/* A loging object used by the servlets */ +public class Logger { + // Logging levels + public final static int NONE = 0; + public static final String NONE_KEY = "NONE"; + public final static int FATAL = 1; + public static final String FATAL_KEY = "FATAL"; + public final static int WARNING = 2; + public static final String WARNING_KEY = "WARNING"; + public final static int INFORMATIONAL = 3; + public static final String INFORMATIONAL_KEY = "INFORMATIONAL"; + public final static int DEBUG = 4; + public static final String DEBUG_KEY = "DEBUG"; + + // Configuration parameters + private final static String LOG_LEVEL = "logLevel"; + private final static String LOG_PATH = "logPath"; + + private int _loggingLevel = FATAL; + private ServletContext _servletContext = null; + private String _logFile = null; + private String _servletName = null; + + // Localization + ResourceBundle _resources = null; + + + /** Initialize logging object. It reads the logLevel and pathLevel init parameters. + * Default is logging level FATAL, and logging using the ServletContext.log + */ + public Logger(ServletConfig config, ResourceBundle resources) { + _resources = resources; + _servletContext = config.getServletContext(); + _servletName = config.getServletName(); + _logFile = config.getInitParameter(LOG_PATH); + if (_logFile != null) { + _logFile = _logFile.trim(); + if (_logFile.length() == 0) _logFile = null; + } + String level = config.getInitParameter(LOG_LEVEL); + if (level != null) { + level = level.trim().toUpperCase(); + if (level.equals(NONE_KEY)) _loggingLevel = NONE; + if (level.equals(FATAL_KEY)) _loggingLevel = FATAL; + if (level.equals(WARNING_KEY)) _loggingLevel = WARNING; + if (level.equals(INFORMATIONAL_KEY)) _loggingLevel = INFORMATIONAL; + if (level.equals(DEBUG_KEY)) _loggingLevel = DEBUG; + } + } + + // Logging API. Fatal, Warning, and Informational are localized + public void addFatal(String key, Throwable throwable) { + logEvent(FATAL, getString(key), throwable); + } + + public void addWarning(String key, String arg) { + logL10N(WARNING, key, arg, (Throwable)null); + } + + public void addWarning(String key, String arg, Throwable t) { + logL10N(WARNING, key, arg, t); + } + + public void addWarning(String key, String arg1, String arg2) { + logL10N(WARNING, key, arg1, arg2); + } + + public void addWarning(String key, String arg1, String arg2, String arg3) { + logL10N(WARNING, key, arg1, arg2, arg3); + } + + public void addInformational(String key) { + logEvent(INFORMATIONAL, getString(key), (Throwable)null); + } + + public void addInformational(String key, String arg) { + logL10N(INFORMATIONAL, key, arg, (Throwable)null); + } + + public void addInformational(String key, String arg1, String arg2, String arg3) { + logL10N(INFORMATIONAL, key, arg1, arg2, arg3); + } + + // Debug messages are not localized + public void addDebug(String msg) { logEvent(DEBUG, msg, null); } + + public void addDebug(String msg, Throwable throwable) { + logEvent(DEBUG, msg, throwable); + } + + // Query to test for level + boolean isNoneLevel() { return _loggingLevel >= NONE; } + boolean isFatalevel() { return _loggingLevel >= FATAL; } + boolean isWarningLevel() { return _loggingLevel >= WARNING; } + boolean isInformationalLevel() { return _loggingLevel >= INFORMATIONAL; } + boolean isDebugLevel() { return _loggingLevel >= DEBUG; } + + // Returns a string from the resources + private String getString(String key) { + try { + return _resources.getString(key); + } catch (MissingResourceException mre) { + return "Missing resource for: " + key; + } + } + + private void logL10N(int level, String key, String arg, Throwable e) { + Object[] messageArguments = { arg }; + logEvent(level, applyPattern(key, messageArguments), e); + } + + private void logL10N(int level, String key, String arg1, String arg2) { + Object[] messageArguments = { arg1, arg2 }; + logEvent(level, applyPattern(key, messageArguments), null); + } + + private void logL10N(int level, String key, String arg1, String arg2, String arg3) { + Object[] messageArguments = { arg1, arg2, arg3 }; + logEvent(level, applyPattern(key, messageArguments), null); + } + + /** Helper function that applies the messageArguments to a message from the resource object */ + private String applyPattern(String key, Object[] messageArguments) { + String message = getString(key); + MessageFormat formatter = new MessageFormat(message); + String output = formatter.format(message, messageArguments); + return output; + } + + // The method that actually does the logging */ + private synchronized void logEvent(int level, String string, Throwable throwable) { + // Check if the event should be logged + if (level > _loggingLevel) return; + + if (_logFile != null) { + // No logfile specified, log using servlet context + PrintWriter pw = null; + try { + pw = new PrintWriter(new FileWriter(_logFile, true)); + pw.println(_servletName + "(" + level + "): " + string); + if (throwable != null) { + throwable.printStackTrace(pw); + } + pw.close(); + // Do a return here. An exception will cause a fall through to + // do _servletContex logging API + return; + } catch (IOException ioe) { + /* just ignore */ + } + } + + // Otherwise, write to servlet context log + if (throwable == null) { + _servletContext.log(string); + } else { + _servletContext.log(string, throwable); + } + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ResourceCatalog.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ResourceCatalog.java new file mode 100755 index 0000000000000000000000000000000000000000..978107a518859c4d463738a1d88aec6b963dc31f --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/ResourceCatalog.java @@ -0,0 +1,514 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; +import java.util.HashMap; +import java.util.List; +import java.util.ArrayList; +import java.io.File; +import java.io.BufferedInputStream; +import javax.servlet.ServletContext; +import javax.xml.parsers.*; +import org.xml.sax.*; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import org.w3c.dom.*; +import jnlp.sample.util.VersionString; +import jnlp.sample.util.VersionID; + +public class ResourceCatalog { + final private Logger log = Red5LoggerFactory.getLogger(ResourceCatalog.class, "screenshare"); + + public static final String VERSION_XML_FILENAME = "version.xml"; + + private ServletContext _servletContext = null; + + private HashMap _entries; + + /** Class to contain the information we know + * about a specific directory + */ + static private class PathEntries { + /* Version-based entries at this particular path */ + private List _versionXmlList; + private List _directoryList; + private List _platformList; + /* Last time this entry was updated */ + private long _lastModified; // Last modified time of entry; + + public PathEntries(List versionXmlList, List directoryList, List platformList, long lastModified) { + _versionXmlList = versionXmlList; + _directoryList = directoryList; + _platformList = platformList; + _lastModified = lastModified; + } + + + public void setDirectoryList(List dirList) { + _directoryList = dirList; + } + + public List getVersionXmlList() { return _versionXmlList; } + public List getDirectoryList() { return _directoryList; } + public List getPlatformList() { return _platformList; } + + public long getLastModified() { return _lastModified; } + } + + public ResourceCatalog(ServletContext servletContext) { + _entries = new HashMap(); + _servletContext = servletContext; + } + + + public JnlpResource lookupResource(DownloadRequest dreq) throws ErrorResponseException { + // Split request up into path and name + String path = dreq.getPath(); + String name = null; + String dir = null; + int idx = path.lastIndexOf('/'); + if (idx == -1) { + name = path; + } else { + name = path.substring(idx + 1); // Exclude '/' + dir = path.substring(0, idx + 1); // Include '/' + } + + // Lookup up already parsed entries, and san directory for entries if neccesary + PathEntries pentries = (PathEntries)_entries.get(dir); + JnlpResource xmlVersionResPath = new JnlpResource(_servletContext, dir + VERSION_XML_FILENAME); + if (pentries == null || (xmlVersionResPath.exists() && xmlVersionResPath.getLastModified() > pentries.getLastModified())) { + log.info("servlet.log.scandir", dir); + List dirList = scanDirectory(dir, dreq); + // Scan XML file + List versionList = new ArrayList(); + List platformList = new ArrayList(); + parseVersionXML(versionList, platformList, dir, xmlVersionResPath); + pentries = new PathEntries(versionList, dirList, platformList, xmlVersionResPath.getLastModified()); + _entries.put(dir, pentries); + } + + // Search for a match + JnlpResource[] result = new JnlpResource[1]; + + if (dreq.isPlatformRequest()) { + int sts = findMatch(pentries.getPlatformList(), name, dreq, result); + if (sts != DownloadResponse.STS_00_OK) { + throw new ErrorResponseException(DownloadResponse.getJnlpErrorResponse(sts)); + } + } else { + // First lookup in versions.xml file + int sts1 = findMatch(pentries.getVersionXmlList(), name, dreq, result); + if (sts1 != DownloadResponse.STS_00_OK) { + // Then lookup in directory + int sts2 = findMatch(pentries.getDirectoryList(), name, dreq, result); + if (sts2 != DownloadResponse.STS_00_OK) { + + // fix for 4450104 + // try rescan and see if it helps + pentries.setDirectoryList(scanDirectory(dir, dreq)); + sts2 = findMatch(pentries.getDirectoryList(), name, dreq, result); + // try again after rescanning directory + if (sts2 != DownloadResponse.STS_00_OK) { + // Throw the most specific error code + throw new ErrorResponseException(DownloadResponse.getJnlpErrorResponse(Math.max(sts1, sts2))); + } + } + } + } + return result[0]; + } + + /** This method finds the best match, or return the best error code. The + * result parameter must be an array with room for one element. + * + * If a match is found, the method returns DownloadResponse.STS_00_OK + * If one or more entries matches on: name, version-id, os, arch, and locale, + * then the one with the highest version-id is set in the result[0] field. + * + * If a match is not found, it returns an error code, either: ERR_10_NO_RESOURCE, + * ERR_11_NO_VERSION, ERR_20_UNSUP_OS, ERR_21_UNSUP_ARCH, ERR_22_UNSUP_LOCALE, + * ERR_23_UNSUP_JRE. + * + */ + public int findMatch(List list, String name, DownloadRequest dreq, JnlpResource[] result) { + if (list == null) return DownloadResponse.ERR_10_NO_RESOURCE; + // Setup return values + VersionID bestVersionId = null; + int error = DownloadResponse.ERR_10_NO_RESOURCE; + VersionString vs = new VersionString(dreq.getVersion()); + // Iterate through entries + for(int i = 0; i < list.size(); i++) { + JnlpResource respath = (JnlpResource)list.get(i); + VersionID vid = new VersionID(respath.getVersionId()); + int sts = matchEntry(name, vs, dreq, respath, vid); + if (sts == DownloadResponse.STS_00_OK) { + if (result[0] == null || vid.isGreaterThan(bestVersionId)) { + result[0] = respath; + bestVersionId = vid; + } + } else { + error = Math.max(error, sts); + } + } + return (result[0] != null) ? DownloadResponse.STS_00_OK : error; + } + + public int matchEntry(String name, VersionString vs, DownloadRequest dreq, JnlpResource jnlpres, VersionID vid) { + if (!name.equals(jnlpres.getName())) { + return DownloadResponse.ERR_10_NO_RESOURCE; + } + if (!vs.contains(vid)) { + return DownloadResponse.ERR_11_NO_VERSION; + } + if (!prefixMatchLists(jnlpres.getOSList(), dreq.getOS())) { + return DownloadResponse.ERR_20_UNSUP_OS; + } + if (!prefixMatchLists(jnlpres.getArchList(), dreq.getArch())) { + return DownloadResponse.ERR_21_UNSUP_ARCH; + } + if (!prefixMatchLists(jnlpres.getLocaleList(), dreq.getLocale())) { + return DownloadResponse.ERR_22_UNSUP_LOCALE; + } + return DownloadResponse.STS_00_OK; + } + + + private static boolean prefixMatchStringList(String[] prefixList, String target) { + // No prefixes matches everything + if (prefixList == null) return true; + // No target, but a prefix list does not match anything + if (target == null) return false; + for(int i = 0; i < prefixList.length; i++) { + if (target.startsWith(prefixList[i])) return true; + } + return false; + } + + /* Return true if at least one of the strings in 'prefixes' are a prefix + * to at least one of the 'keys'. + */ + public boolean prefixMatchLists(String[] prefixes, String[] keys) { + // The prefixes are part of the server resources. If none is given, + // everything matches + if (prefixes == null) return true; + // If no os keyes was given, and the server resource is keyed of this, + // then return false. + if (keys == null) return false; + // Check for a match on a key + for(int i = 0; i < keys.length; i++) { + if (prefixMatchStringList(prefixes, keys[i])) return true; + } + return false; + } + + /** This method scans the directory pointed to by the + * given path and creates a list of ResourcePath elements + * that contains information about all the entries + * + * The version-based information is encoded in the file name + * given the following format: + * + * entry ::= <name> __ ( <options> ). <ext> + * options ::= <option> ( __ <options> )? + * option ::= V<version-id> + * | O<os> + * | A<arch> + * | L<locale> + * + */ + + + private String jnlpGetPath(DownloadRequest dreq) { + // fix for 4474021 + // try to manuually generate the filename + // extract file name + String path = dreq.getPath(); + String filename = path.substring(path.lastIndexOf("/") + 1); + path = path.substring(0, path.lastIndexOf("/") + 1); + String name = filename; + String ext = null; + + if (filename.lastIndexOf(".") != -1) { + ext = filename.substring(filename.lastIndexOf(".") + 1); + + filename = filename.substring(0, filename.lastIndexOf(".")); + + } + if (dreq.getVersion() != null) { + filename += "__V" + dreq.getVersion(); + } + + String[] temp = dreq.getOS(); + + if (temp != null) { + for (int i=0; i<temp.length; i++) { + filename += "__O" + temp[i]; + } + } + + temp = dreq.getArch(); + + if (temp != null) { + for (int i=0; i<temp.length; i++) { + filename += "__A" + temp[i]; + } + } + temp = dreq.getLocale(); + + if (temp != null) { + for (int i=0; i<temp.length; i++) { + filename += "__L" + temp[i]; + } + } + + if (ext != null) { + filename += "." + ext; + } + + path += filename; + + return path; + } + + public List scanDirectory(String dirPath, DownloadRequest dreq) { + ArrayList list = new ArrayList(); + + // fix for 4474021 + if (_servletContext.getRealPath(dirPath) == null) { + String path = jnlpGetPath(dreq); + + String name = dreq.getPath().substring(path.lastIndexOf("/") + 1); + + JnlpResource jnlpres = new JnlpResource(_servletContext, name, dreq.getVersion(), dreq.getOS(), dreq.getArch(), dreq.getLocale(), path, dreq.getVersion()); + + // the file does not exist + if (jnlpres.getResource() == null) return null; + + list.add(jnlpres); + return list; + } + File dir = new File(_servletContext.getRealPath(dirPath)); + log.debug("File directory: " + dir); + if (dir.exists() && dir.isDirectory()) { + File[] entries = dir.listFiles(); + for(int i = 0; i < entries.length; i++) { + JnlpResource jnlpres = parseFileEntry(dirPath, entries[i].getName()); + if (jnlpres != null) { + if (log.isDebugEnabled()) { + log.debug("Read file resource: " + jnlpres); + } + list.add(jnlpres); + } + } + } + return list; + } + + private JnlpResource parseFileEntry(String dir, String filename) { + int idx = filename .indexOf("__"); + if (idx == -1) return null; + + // Cut out name + String name = filename.substring(0, idx); + String rest = filename.substring(idx); + + // Cut out extension + idx = rest.lastIndexOf('.'); + String extension = ""; + if (idx != -1 ) { + extension = rest.substring(idx); + rest = rest .substring(0, idx); + } + + // Parse options + String versionId = null; + ArrayList osList = new ArrayList(); + ArrayList archList = new ArrayList(); + ArrayList localeList = new ArrayList(); + while(rest.length() > 0) { + /* Must start with __ at this point */ + if (!rest.startsWith("__")) return null; + rest = rest.substring(2); + // Get option and argument + char option = rest.charAt(0); + idx = rest.indexOf("__"); + String arg = null; + if (idx == -1) { + arg = rest.substring(1); + rest = ""; + } else { + arg = rest.substring(1, idx); + rest = rest.substring(idx); + } + switch(option) { + case 'V': versionId = arg; break; + case 'O': osList.add(arg); break; + case 'A': archList.add(arg); break; + case 'L': localeList.add(arg); break; + default: return null; // error + } + } + + return new JnlpResource(_servletContext, + name + extension, /* Resource name in URL request */ + versionId, + listToStrings(osList), + listToStrings(archList), + listToStrings(localeList), + dir + filename, /* Resource name in WAR file */ + versionId); + } + + private String[] listToStrings(List list) { + if (list.size() == 0) return null; + return (String[])list.toArray(new String[list.size()]); + } + + // Returns false if parsing failed + private void parseVersionXML(final List versionList, final List platformList, + final String dir, final JnlpResource versionRes) { + if (!versionRes.exists()) return; + + // Parse XML into a more understandable format + XMLNode root = null; + try { + DocumentBuilderFactory docBuilderFactory = DocumentBuilderFactory.newInstance(); + DocumentBuilder docBuilder = docBuilderFactory.newDocumentBuilder(); + Document doc = docBuilder.parse(new BufferedInputStream(versionRes.getResource().openStream())); + doc.getDocumentElement().normalize(); + + // Convert document into an XMLNode structure, since we already got utility methods + // to handle these. We should really use the data-binding stuff here - but that will come + // later + // + root = XMLParsing.convert(doc.getDocumentElement()); + } catch (SAXParseException err) { + log.warn("servlet.log.warning.xml.parsing", + versionRes.getPath(), + Integer.toString(err.getLineNumber()), + err.getMessage()); + return; + } catch (Throwable t) { + log.warn("servlet.log.warning.xml.reading", versionRes.getPath(), t); + return; + } + + // Check that root element is a <jnlp> tag + if (!root.getName().equals("jnlp-versions")) { + log.warn("servlet.log.warning.xml.missing-jnlp", versionRes.getPath()); + return; + } + + // Visit all <resource> elements + XMLParsing.visitElements(root, "<resource>", new XMLParsing.ElementVisitor() { + public void visitElement(XMLNode node) { + XMLNode pattern = XMLParsing.findElementPath(node, "<pattern>"); + if (pattern == null) { + log.warn("servlet.log.warning.xml.missing-pattern", versionRes.getPath()); + } else { + // Parse pattern + String name = XMLParsing.getElementContent(pattern , "<name>", ""); + String versionId = XMLParsing.getElementContent(pattern , "<version-id>"); + String[] os = XMLParsing.getMultiElementContent(pattern, "<os>"); + String[] arch = XMLParsing.getMultiElementContent(pattern, "<arch>"); + String[] locale = XMLParsing.getMultiElementContent(pattern, "<locale>"); + // Get return request + String file = XMLParsing.getElementContent(node, "<file>"); + if (versionId == null || file == null) { + log.warn("servlet.log.warning.xml.missing-elems", versionRes.getPath()); + } else { + JnlpResource res = new JnlpResource(_servletContext, + name, + versionId, + os, + arch, + locale, + dir + file, + versionId); + if (res.exists()) { + versionList.add(res); + if (log.isDebugEnabled()) { + log.debug("Read resource: " + res); + } + } else { + log.warn("servlet.log.warning.missing-file", file, versionRes.getPath()); + } + } + } + } + }); + + // Visit all <resource> elements + XMLParsing.visitElements(root, "<platform>", new XMLParsing.ElementVisitor() { + public void visitElement(XMLNode node) { + XMLNode pattern = XMLParsing.findElementPath(node, "<pattern>"); + if (pattern == null) { + log.warn("servlet.log.warning.xml.missing-pattern", versionRes.getPath()); + } else { + // Parse pattern + String name = XMLParsing.getElementContent(pattern , "<name>", ""); + String versionId = XMLParsing.getElementContent(pattern , "<version-id>"); + String[] os = XMLParsing.getMultiElementContent(pattern, "<os>"); + String[] arch = XMLParsing.getMultiElementContent(pattern, "<arch>"); + String[] locale = XMLParsing.getMultiElementContent(pattern, "<locale>"); + // Get return request + String file = XMLParsing.getElementContent(node, "<file>"); + String productId = XMLParsing.getElementContent(node, "<product-version-id>"); + + if (versionId == null || file == null || productId == null) { + log.warn("servlet.log.warning.xml.missing-elems2", versionRes.getPath()); + } else { + JnlpResource res = new JnlpResource(_servletContext, + name, + versionId, + os, + arch, + locale, + dir + file, + productId); + if (res.exists()) { + platformList.add(res); + if (log.isDebugEnabled()) { + log.debug("Read platform resource: " + res); + } + } else { + log.warn("servlet.log.warning.missing-file", file, versionRes.getPath()); + } + } + } + } + }); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLAttribute.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLAttribute.java new file mode 100755 index 0000000000000000000000000000000000000000..76a1027620bfd8caf134d0f8a0e5cb3df8aabc21 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLAttribute.java @@ -0,0 +1,82 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; + +/** Class that contains information about a specific attribute + */ +public class XMLAttribute { + private String _name; + private String _value; + private XMLAttribute _next; + + public XMLAttribute(String name, String value) { + _name = name; + _value = value; + _next = null; + } + + public XMLAttribute(String name, String value, XMLAttribute next) { + _name = name; + _value = value; + _next = next; + } + + public String getName() { return _name; } + public String getValue() { return _value; } + public XMLAttribute getNext() { return _next; } + public void setNext(XMLAttribute next) { _next = next; } + + public boolean equals(Object o) { + if (o == null || !(o instanceof XMLAttribute)) return false; + XMLAttribute other = (XMLAttribute)o; + return + match(_name, other._name) && + match(_value, other._value) && + match(_next, other._next); + } + + private static boolean match(Object o1, Object o2) { + if (o1 == null) return (o2 == null); + return o1.equals(o2); + } + + public String toString() { + if (_next != null) { + return _name + "=\"" + _value + "\" " + _next.toString(); + } else { + return _name + "=\"" + _value + "\""; + } + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLNode.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLNode.java new file mode 100755 index 0000000000000000000000000000000000000000..ee5ca60051d47204b59af1641491b9e7a44ffcf9 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLNode.java @@ -0,0 +1,150 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; + +import java.io.PrintWriter; +import java.io.StringWriter; + +/** Class that contains information about an XML Node + */ +public class XMLNode { + private boolean _isElement; // Element/PCTEXT + private String _name; + private XMLAttribute _attr; + private XMLNode _parent; // Parent Node + private XMLNode _nested; // Nested XML tags + private XMLNode _next; // Following XML tag on the same level + + /** Creates a PCTEXT node */ + public XMLNode(String name) { + this(name, null, null, null); + _isElement = false; + } + + /** Creates a ELEMENT node */ + public XMLNode(String name, XMLAttribute attr) { + this(name, attr, null, null); + } + + /** Creates a ELEMENT node */ + public XMLNode(String name, XMLAttribute attr, XMLNode nested, XMLNode next) { + _isElement = true; + _name = name; + _attr = attr; + _nested = nested; + _next = next; + _parent = null; + } + + public String getName() { return _name; } + public XMLAttribute getAttributes() { return _attr; } + public XMLNode getNested() { return _nested; } + public XMLNode getNext() { return _next; } + public boolean isElement() { return _isElement; } + + public void setParent(XMLNode parent) { _parent = parent; } + public XMLNode getParent() { return _parent; } + + public void setNext(XMLNode next) { _next = next; } + public void setNested(XMLNode nested) { _nested = nested; } + + public boolean equals(Object o) { + if (o == null || !(o instanceof XMLNode)) return false; + XMLNode other = (XMLNode)o; + boolean result = + match(_name, other._name) && + match(_attr, other._attr) && + match(_nested, other._nested) && + match(_next, other._next); + return result; + } + + public String getAttribute(String name) { + XMLAttribute cur = _attr; + while(cur != null) { + if (name.equals(cur.getName())) return cur.getValue(); + cur = cur.getNext(); + } + return ""; + } + + private static boolean match(Object o1, Object o2) { + if (o1 == null) return (o2 == null); + return o1.equals(o2); + } + + public void printToStream(PrintWriter out) { + printToStream(out, 0); + } + + public void printToStream(PrintWriter out, int n) { + if (!isElement()) { + out.print(_name); + } else { + if (_nested == null) { + String attrString = (_attr == null) ? "" : (" " + _attr.toString()); + lineln(out, n, "<" + _name + attrString + "/>"); + } else { + String attrString = (_attr == null) ? "" : (" " + _attr.toString()); + lineln(out, n, "<" + _name + attrString + ">"); + _nested.printToStream(out, n + 1); + if (_nested.isElement()) { + lineln(out, n, "</" + _name + ">"); + } else { + out.print("</" + _name + ">"); + } + } + } + if (_next != null) { + _next.printToStream(out, n); + } + } + + private static void lineln(PrintWriter out, int indent, String s) { + out.println(""); + for(int i = 0; i < indent; i++) { + out.print(" "); + } + out.print(s); + } + + public String toString() { + StringWriter sw = new StringWriter(1000); + PrintWriter pw = new PrintWriter(sw); + printToStream(pw); + pw.close(); + return sw.toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLParsing.java b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLParsing.java new file mode 100755 index 0000000000000000000000000000000000000000..94571efda15f3877e97a434863e282a112d2e6d8 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/XMLParsing.java @@ -0,0 +1,189 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.servlet; + +import javax.xml.parsers.*; +import java.util.ArrayList; +import java.util.List; +import org.xml.sax.*; +import org.w3c.dom.*; + +/** Contains handy methods for looking up information + * stored in XMLNodes. + */ +public class XMLParsing { + + public static XMLNode convert(Node n) { + if (n == null) { + return null; + } else if (n instanceof Text) { + Text tn = (Text)n; + return new XMLNode(tn.getNodeValue()); + } else if (n instanceof Element) { + Element en = (Element)n; + + XMLAttribute xmlatts = null; + NamedNodeMap attributes = en.getAttributes(); + for(int i = attributes.getLength() - 1; i >= 0; i--) { + Attr ar = (Attr)attributes.item(i); + xmlatts = new XMLAttribute(ar.getName(), ar.getValue(), xmlatts); + } + + // Convert childern + XMLNode thisNode = new XMLNode(en.getNodeName(), xmlatts, null, null);; + XMLNode last = null; + Node nn = en.getFirstChild(); + while(nn != null) { + if (thisNode.getNested() == null) { + last = convert(nn); + thisNode.setNested(last); + } else { + XMLNode nnode = convert(nn); + last.setNext(nnode); + last = nnode; + } + last.setParent(thisNode); + nn = nn.getNextSibling(); + } + + return thisNode; + } + return null; + } + + /** Returns true if the path exists in the document, otherwise false */ + static public boolean isElementPath(XMLNode root, String path) { + return findElementPath(root, path) != null; + } + + + /** Returns a string describing the current location in the DOM */ + static public String getPathString(XMLNode e) { + return (e == null || !(e.isElement())) ? "" : getPathString(e.getParent()) + "<" + e.getName() + ">"; + } + + + /** Like getElementContents(...) but with a defaultValue of null */ + static public String getElementContent(XMLNode root, String path) { + return getElementContent(root, path, null); + } + + /** Like getElementContents(...) but with a defaultValue of null */ + static public String[] getMultiElementContent(XMLNode root, String path) { + final List list = new ArrayList(); + visitElements(root, path, new ElementVisitor() { + public void visitElement(XMLNode n) { + String value = getElementContent(n, ""); + if (value != null) list.add(value); + } + }); + if (list.size() == 0) return null; + return (String[])list.toArray(new String[list.size()]); + } + + /** Returns the value of the last element tag in the path, e.g., <..><tag>value</tag>. The DOM is assumes + * to be normalized. If no value is found, the defaultvalue is returned + */ + static public String getElementContent(XMLNode root, String path, String defaultvalue) { + XMLNode e = findElementPath(root, path); + if (e == null) return defaultvalue; + XMLNode n = e.getNested(); + if (n != null && !n.isElement()) return n.getName(); + return defaultvalue; + } + + /** Parses a path string of the form <tag1><tag2><tag3> and returns the specific Element + * node for that tag, or null if it does not exist. If multiple elements exists with same + * path the first is returned + */ + static public XMLNode findElementPath(XMLNode elem, String path) { + // End condition. Root null -> path does not exist + if (elem == null) return null; + // End condition. String empty, return current root + if (path == null || path.length() == 0) return elem; + + // Strip of first tag + int idx = path.indexOf('>'); + String head = path.substring(1, idx); + String tail = path.substring(idx + 1); + return findElementPath(findChildElement(elem, head), tail); + } + + /** Returns an child element with the current tag name or null. */ + static public XMLNode findChildElement(XMLNode elem, String tag) { + XMLNode n = elem.getNested(); + while(n != null) { + if (n.isElement() && n.getName().equals(tag)) return n; + n = n.getNext(); + } + return null; + } + + /** Iterator class */ + public abstract static class ElementVisitor { + abstract public void visitElement(XMLNode e); + } + + /** Visits all elements which matches the <path>. The iteration is only + * done on the last elment in the path. + */ + static public void visitElements(XMLNode root, String path, ElementVisitor ev) { + // Get last element in path + int idx = path.lastIndexOf('<'); + String head = path.substring(0, idx); + String tag = path.substring(idx + 1, path.length() - 1); + + XMLNode elem = findElementPath(root, head); + if (elem == null) return; + + // Iterate through all child nodes + XMLNode n = elem.getNested(); + while(n != null) { + if (n.isElement() && n.getName().equals(tag)) { + ev.visitElement(n); + } + n = n.getNext(); + } + } + + static public void visitChildrenElements(XMLNode elem, ElementVisitor ev) { + // Iterate through all child nodes + XMLNode n = elem.getNested(); + while(n != null) { + if (n.isElement()) ev.visitElement(n); + n = n.getNext(); + } + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/resources/strings.properties b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/resources/strings.properties new file mode 100755 index 0000000000000000000000000000000000000000..7b4f30d8086f29effa41e969aafebe7aec0212ed --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/servlet/resources/strings.properties @@ -0,0 +1,67 @@ +# +# Copyright (c) 2005, 2010, Oracle and/or its affiliates. All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# -Redistribution of source code must retain the above copyright notice, this +# list of conditions and the following disclaimer. +# +# -Redistribution in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# Neither the name of Oracle nor the names of contributors may +# be used to endorse or promote products derived from this software without +# specific prior written permission. +# +# This software is provided "AS IS," without a warranty of any kind. ALL +# EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING +# ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE +# OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") +# AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE +# AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS +# DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST +# REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, +# INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY +# OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, +# EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. +# +# You acknowledge that this software is not designed, licensed or intended +# for use in the design, construction, operation or maintenance of any +# nuclear facility. +# + +# Fatals +servlet.log.fatal.internalerror=Internal error: + +# Warnings +servlet.log.warning.nolastmodified=Last-modified read as 0 for {0} +servlet.log.warning.notimestamp=Timestamp read as 0 for {0}. Using Last-modified instead +servlet.log.warning.missing-file=Reference to non-existing file ({0}) in {1} +servlet.log.warning.xml.parsing=Error parsing {0} at line {1}: {2} +servlet.log.warning.xml.reading=Unexpected error reading {0}: +servlet.log.warning.xml.missing-jnlp=Missing <jnlp-versions> element in {0} +servlet.log.warning.xml.missing-pattern=Missing <pattern> element in {0} +servlet.log.warning.xml.missing-elems=Missing <version-id> or <file> attribute in {0} +servlet.log.warning.xml.missing-elems2=Missing <version-id>, <file>, or <product-version-id> attribute in {0} +servlet.log.warning.jardiff.failed=Failed to generate JarDiff for {0} {1}->{2} + +# Informational +servlet.log.info.request=Request: {0} +servlet.log.info.useragent=User-Agent: {0} +servlet.log.info.goodrequest=Resource returned: {0} +servlet.log.info.badrequest=Error code returned for request: {0} +servlet.log.scandir=Rescanning directory: {0} +servlet.log.info.jardiff.response=JarDiff returned for request +servlet.log.info.jardiff.gen=Generating JarDiff for {0} {1}->{2} + +# JNLP Error strings +servlet.jnlp.err.10 = Could not locate resource +servlet.jnlp.err.11 = Could not locate requested version +servlet.jnlp.err.20 = Unsupported operating system +servlet.jnlp.err.21 = Unsupported architecture +servlet.jnlp.err.22 = Unsupported locale +servlet.jnlp.err.23 = Unsupported JRE version +servlet.jnlp.err.99 = Unknown error + diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionID.java b/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionID.java new file mode 100755 index 0000000000000000000000000000000000000000..e52c6778b3eab52c3b226cfe6aa9feb8e80849e8 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionID.java @@ -0,0 +1,260 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.util; + +import java.util.ArrayList; +import java.util.Arrays; + +/** + * VersionID contains a JNLP version ID. + * + * The VersionID also contains a prefix indicator that can + * be used when stored with a VersionString + * + */ +public class VersionID implements Comparable { + private String[] _tuple; // Array of Integer or String objects + private boolean _usePrefixMatch; // star (*) prefix + private boolean _useGreaterThan; // plus (+) greather-than + private boolean _isCompound; // and (&) operator + private VersionID _rest; // remaining part after the & + + /** Creates a VersionID object */ + public VersionID(String str) { + _usePrefixMatch = false; + _useGreaterThan = false; + _isCompound = false; + if (str == null && str.length() == 0) { + _tuple = new String[0]; + return; + } + + // Check for compound + int amp = str.indexOf("&"); + if (amp >= 0) { + _isCompound = true; + VersionID firstPart = new VersionID(str.substring(0, amp)); + _rest = new VersionID(str.substring(amp+1)); + _tuple = firstPart._tuple; + _usePrefixMatch = firstPart._usePrefixMatch; + _useGreaterThan = firstPart._useGreaterThan; + } else { + // Check for postfix + if (str.endsWith("+")) { + _useGreaterThan = true; + str = str.substring(0, str.length() - 1); + } else if (str.endsWith("*")) { + _usePrefixMatch = true; + str = str.substring(0, str.length() - 1); + } + + ArrayList list = new ArrayList(); + int start = 0; + for(int i = 0; i < str.length(); i++) { + // Split at each separator character + if (".-_".indexOf(str.charAt(i)) != -1) { + if (start < i) { + String value = str.substring(start, i); + list.add(value); + } + start = i + 1; + } + } + if (start < str.length()) { + list.add(str.substring(start, str.length())); + } + _tuple = new String[list.size()]; + _tuple = (String[])list.toArray(_tuple); + } + } + + /** Returns true if no flags are set */ + public boolean isSimpleVersion() { + return !_useGreaterThan && !_usePrefixMatch && !_isCompound; + } + + /** Match 'this' versionID against vid. + * The _usePrefixMatch/_useGreaterThan flag is used to determine if a + * prefix match of an exact match should be performed + * if _isCompound, must match _rest also. + */ + public boolean match(VersionID vid) { + if (_isCompound) { + if (!_rest.match(vid)) { + return false; + } + } + return (_usePrefixMatch) ? this.isPrefixMatch(vid) : + (_useGreaterThan) ? vid.isGreaterThanOrEqual(this) : + matchTuple(vid); + } + + /** Compares if two version IDs are equal */ + public boolean equals(Object o) { + if (matchTuple(o)) { + VersionID ov = (VersionID) o; + if (_rest == null || _rest.equals(ov._rest)) { + if ((_useGreaterThan == ov._useGreaterThan) && + (_usePrefixMatch == ov._usePrefixMatch)) { + return true; + } + } + } + return false; + } + + /** Compares if two version IDs are equal */ + private boolean matchTuple(Object o) { + // Check for null and type + if (o == null || !(o instanceof VersionID)) return false; + VersionID vid = (VersionID)o; + + // Normalize arrays + String[] t1 = normalize(_tuple, vid._tuple.length); + String[] t2 = normalize(vid._tuple, _tuple.length); + + // Check contents + for(int i = 0; i < t1.length; i++) { + Object o1 = getValueAsObject(t1[i]); + Object o2 = getValueAsObject(t2[i]); + if (!o1.equals(o2)) return false; + } + return true; + } + + private Object getValueAsObject(String value) { + if (value.length() > 0 && value.charAt(0) != '-') { + try { return Integer.valueOf(value); + } catch(NumberFormatException nfe) { /* fall through */ } + } + return value; + } + + public boolean isGreaterThan(VersionID vid) { + return isGreaterThanOrEqualHelper(vid, false); + } + + public boolean isGreaterThanOrEqual(VersionID vid) { + return isGreaterThanOrEqualHelper(vid, true); + } + + /** Compares if 'this' is greater than vid */ + private boolean isGreaterThanOrEqualHelper(VersionID vid, + boolean allowEqual) { + + if (_isCompound) { + if (!_rest.isGreaterThanOrEqualHelper(vid, allowEqual)) { + return false; + } + } + // Normalize the two strings + String[] t1 = normalize(_tuple, vid._tuple.length); + String[] t2 = normalize(vid._tuple, _tuple.length); + + for(int i = 0; i < t1.length; i++) { + // Compare current element + Object e1 = getValueAsObject(t1[i]); + Object e2 = getValueAsObject(t2[i]); + if (e1.equals(e2)) { + // So far so good + } else { + if (e1 instanceof Integer && e2 instanceof Integer) { + return ((Integer)e1).intValue() > ((Integer)e2).intValue(); + } else { + String s1 = t1[i].toString(); + String s2 = t2[i].toString(); + return s1.compareTo(s2) > 0; + } + + } + } + // If we get here, they are equal + return allowEqual; + } + + /** Checks if 'this' is a prefix of vid */ + public boolean isPrefixMatch(VersionID vid) { + + if (_isCompound) { + if (!_rest.isPrefixMatch(vid)) { + return false; + } + } + // Make sure that vid is at least as long as the prefix + String[] t2 = normalize(vid._tuple, _tuple.length); + + for(int i = 0; i < _tuple.length; i++) { + Object e1 = _tuple[i]; + Object e2 = t2[i]; + if (e1.equals(e2)) { + // So far so good + } else { + // Not a prefix + return false; + } + } + return true; + } + + /** Normalize an array to a certain lengh */ + private String[] normalize(String[] list, int minlength) { + if (list.length < minlength) { + // Need to do padding + String[] newlist = new String[minlength]; + System.arraycopy(list, 0, newlist, 0, list.length); + Arrays.fill(newlist, list.length, newlist.length, "0"); + return newlist; + } else { + return list; + } + } + + public int compareTo(Object o) { + if (o == null || !(o instanceof VersionID)) return -1; + VersionID vid = (VersionID)o; + return equals(vid) ? 0 : (isGreaterThanOrEqual(vid) ? 1 : -1); + } + /** Show it as a string */ + public String toString() { + StringBuffer sb = new StringBuffer(); + for(int i = 0; i < _tuple.length -1; i++) { + sb.append(_tuple[i]); + sb.append('.'); + } + if (_tuple.length > 0 ) sb.append(_tuple[_tuple.length - 1]); + if (_usePrefixMatch) sb.append('+'); + return sb.toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionString.java b/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionString.java new file mode 100755 index 0000000000000000000000000000000000000000..296093c4376a1acfadb812a4913b9c33231665fb --- /dev/null +++ b/bbb-screenshare/app/src/main/java/jnlp/sample/util/VersionString.java @@ -0,0 +1,105 @@ +/* + * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * -Redistribution of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * -Redistribution in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * Neither the name of Oracle nor the names of contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * This software is provided "AS IS," without a warranty of any kind. ALL + * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING + * ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE + * OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN MICROSYSTEMS, INC. ("SUN") + * AND ITS LICENSORS SHALL NOT BE LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE + * AS A RESULT OF USING, MODIFYING OR DISTRIBUTING THIS SOFTWARE OR ITS + * DERIVATIVES. IN NO EVENT WILL SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST + * REVENUE, PROFIT OR DATA, OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, + * INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY + * OF LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE, + * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. + * + * You acknowledge that this software is not designed, licensed or intended + * for use in the design, construction, operation or maintenance of any + * nuclear facility. + */ + +package jnlp.sample.util; +import java.util.ArrayList; +import java.util.StringTokenizer; + +/* + * Utility class that knows to handle version strings + * A version string is of the form: + * + * (version-id ('+'?) ' ') * + * + */ +public class VersionString { + private ArrayList _versionIds; + + /** Constructs a VersionString object from string */ + public VersionString(String vs) { + _versionIds = new ArrayList(); + if (vs != null) { + StringTokenizer st = new StringTokenizer(vs, " ", false); + while(st.hasMoreElements()) { + // Note: The VersionID class takes care of a postfixed '+' + _versionIds.add(new VersionID(st.nextToken())); + } + } + } + + /** Check if this VersionString object contains the VersionID m */ + public boolean contains(VersionID m) { + for(int i = 0; i < _versionIds.size(); i++) { + VersionID vi = (VersionID)_versionIds.get(i); + boolean check = vi.match(m); + if (check) return true; + } + return false; + } + + /** Check if this VersionString object contains the VersionID m, given as a string */ + public boolean contains(String versionid) { + return contains(new VersionID(versionid)); + } + + /** Check if this VersionString object contains anything greater than m */ + public boolean containsGreaterThan(VersionID m) { + for(int i = 0; i < _versionIds.size(); i++) { + VersionID vi = (VersionID)_versionIds.get(i); + boolean check = vi.isGreaterThan(m); + if (check) return true; + } + return false; + } + + /** Check if this VersionString object contains anything greater than the VersionID m, given as a string */ + public boolean containsGreaterThan(String versionid) { + return containsGreaterThan(new VersionID(versionid)); + } + + /** Check if the versionString 'vs' contains the VersionID 'vi' */ + static public boolean contains(String vs, String vi) { + return (new VersionString(vs)).contains(vi); + } + + /** Pretty-print object */ + public String toString() { + StringBuffer sb = new StringBuffer(); + for(int i = 0; i < _versionIds.size(); i++) { + sb.append(_versionIds.get(i).toString()); + sb.append(' '); + } + return sb.toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/Error.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/Error.java new file mode 100755 index 0000000000000000000000000000000000000000..31ac1b587ec4ce1f200efd45af98d7e9ea3daa5e --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/Error.java @@ -0,0 +1,10 @@ +package org.bigbluebutton.app.screenshare; + +public class Error { + + public final String reason; + + public Error(String reason) { + this.reason = reason; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/EventRecordingService.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/EventRecordingService.java new file mode 100755 index 0000000000000000000000000000000000000000..d534387e2e99c3a2a0c0b38bdb7511a69bf0a0d3 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/EventRecordingService.java @@ -0,0 +1,43 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare; + + +import java.util.Map; + +import redis.clients.jedis.Jedis; + +public class EventRecordingService { + private static final String COLON = ":"; + + private final String host; + private final int port; + + public EventRecordingService(String host, int port) { + this.host = host; + this.port = port; + } + + public void record(String meetingId, Map<String, String> event) { + Jedis jedis = new Jedis(host, port); + Long msgid = jedis.incr("global:nextRecordedMsgId"); + jedis.hmset("recording:" + meetingId + COLON + msgid, event); + jedis.rpush("meeting:" + meetingId + COLON + "recordings", msgid.toString()); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IScreenShareApplication.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IScreenShareApplication.java new file mode 100755 index 0000000000000000000000000000000000000000..1ad412e2f860394783c3df9f1c6b4146b15c5511 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IScreenShareApplication.java @@ -0,0 +1,17 @@ +package org.bigbluebutton.app.screenshare; + +public interface IScreenShareApplication { + + IsScreenSharingResponse isScreenSharing(String meetingId); + ScreenShareInfoResponse getScreenShareInfo(String meetingId, String token); + StartShareRequestResponse startShareRequest(String meetingId, String userId, Boolean record); + void stopShareRequest(String meetingId, String streamId); + void streamStarted(String meetingId, String streamId, String url); + void streamStopped(String meetingId, String streamId); + void sharingStarted(String meetingId, String streamId, Integer width, Integer height); + void sharingStopped(String meetingId, String streamId); + void updateShareStatus(String meetingId, String streamId, Integer seqNum); + Boolean isSharingStopped(String meetingId, String streamId); + Boolean recordStream(String meetingId, String streamId); + void userDisconnected(String meetingId, String userId); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IsScreenSharingResponse.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IsScreenSharingResponse.java new file mode 100755 index 0000000000000000000000000000000000000000..2cab8dea76454735b89183d102ca432746c6d837 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/IsScreenSharingResponse.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare; + +public class IsScreenSharingResponse { + + public final StreamInfo info; + public final Error error; + + public IsScreenSharingResponse(StreamInfo info, Error error) { + this.info = info; + this.error = error; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfo.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfo.java new file mode 100755 index 0000000000000000000000000000000000000000..294706eb766b9305861728b13d2afcf935e30d17 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfo.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare; + +public class ScreenShareInfo { + + public final String streamId; + public final String publishUrl; + + public ScreenShareInfo(String publishUrl, String streamId) { + this.streamId = streamId; + this.publishUrl = publishUrl; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfoResponse.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfoResponse.java new file mode 100755 index 0000000000000000000000000000000000000000..5f97d545beb2bf9a1e4d7ca5ef8993f4aa0f2658 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenShareInfoResponse.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare; + +public class ScreenShareInfoResponse { + + public final ScreenShareInfo info; + public final Error error; + + public ScreenShareInfoResponse(ScreenShareInfo info, Error error) { + this.info = info; + this.error = error; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenshareStreamListener.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenshareStreamListener.java new file mode 100755 index 0000000000000000000000000000000000000000..378e6dbfbf111ea130aaa34de530001e320f5663 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/ScreenshareStreamListener.java @@ -0,0 +1,99 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare; + +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.TimeUnit; + +import org.apache.mina.core.buffer.IoBuffer; +import org.red5.server.api.IConnection; +import org.red5.server.api.Red5; +import org.red5.server.api.stream.IBroadcastStream; +import org.red5.server.api.stream.IStreamListener; +import org.red5.server.api.stream.IStreamPacket; +import org.red5.server.net.rtmp.event.VideoData; + +/** + * Class to listen for the first video packet of the webcam. + * We need to listen for the first packet and send a startWebcamEvent. + * The reason is that when starting the webcam, sometimes Flash Player + * needs to prompt the user for permission to access the webcam. However, + * while waiting for the user to click OK to the prompt, Red5 has already + * called the startBroadcast method which we take as the start of the recording. + * When the user finally clicks OK, the packets then start to flow through. + * This introduces a delay of when we assume the start of the recording and + * the webcam actually publishes video packets. When we do the ingest and + * processing of the video and multiplex the audio, the video and audio will + * be un-synched by at least this amount of delay. + * @author Richard Alam + * + */ +public class ScreenshareStreamListener implements IStreamListener { + private EventRecordingService recordingService; + private volatile boolean firstPacketReceived = false; + private String recordingDir; + + public ScreenshareStreamListener(EventRecordingService s, String recordingDir) { + this.recordingService = s; + this.recordingDir = recordingDir; + } + + private Long genTimestamp() { + return TimeUnit.NANOSECONDS.toMillis(System.nanoTime()); + } + + @Override + public void packetReceived(IBroadcastStream stream, IStreamPacket packet) { + IoBuffer buf = packet.getData(); + if (buf != null) + buf.rewind(); + + if (buf == null || buf.remaining() == 0){ + return; + } + + if (packet instanceof VideoData) { + if (! firstPacketReceived) { + firstPacketReceived = true; + IConnection conn = Red5.getConnectionLocal(); + + String meetingId = conn.getScope().getName(); + + String filename = recordingDir; + if (!filename.endsWith("/")) { + filename.concat("/"); + } + + filename = filename.concat(meetingId).concat("/").concat(stream.getPublishedName()).concat(".flv"); + + Map<String, String> event = new HashMap<String, String>(); + event.put("module", "Deskshare"); + event.put("timestamp", new Long(System.currentTimeMillis()).toString()); + event.put("meetingId", meetingId); + event.put("file", filename); + event.put("stream", stream.getPublishedName()); + event.put("eventName", "DeskshareStartedEvent"); + + recordingService.record(conn.getScope().getName(), event); + } + } + } + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StartShareRequestResponse.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StartShareRequestResponse.java new file mode 100755 index 0000000000000000000000000000000000000000..bd1e58e5dc5b3264f30fa70bad84eb51bcbea5c4 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StartShareRequestResponse.java @@ -0,0 +1,15 @@ +package org.bigbluebutton.app.screenshare; + +public class StartShareRequestResponse { + + public final String token; + public final String jnlp; + public final Error error; + + public StartShareRequestResponse(String token, String jnlp, Error error) { + this.token = token; + this.jnlp = jnlp; + + this.error = error; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StreamInfo.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StreamInfo.java new file mode 100755 index 0000000000000000000000000000000000000000..1c9334a96d7ba352024720798505024a364280fc --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/StreamInfo.java @@ -0,0 +1,19 @@ +package org.bigbluebutton.app.screenshare; + +public class StreamInfo { + + public final String streamId; + public final Boolean sharing; + public final int width; + public final int height; + public final String url; + + public StreamInfo(Boolean sharing, String streamId, + int width, int height, String url) { + this.sharing = sharing; + this.streamId = streamId; + this.width = width; + this.height = height; + this.url = url; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/EventMessageBusImp.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/EventMessageBusImp.java new file mode 100755 index 0000000000000000000000000000000000000000..9a53a2b2915e713249ab63ac4d6d866e55584199 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/EventMessageBusImp.java @@ -0,0 +1,67 @@ +package org.bigbluebutton.app.screenshare.events; + +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; +import java.util.Set; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + +public class EventMessageBusImp implements IEventsMessageBus { + private static Logger log = Red5LoggerFactory.getLogger(EventMessageBusImp.class, "screenshare"); + + private BlockingQueue<IEvent> receivedMessages = new LinkedBlockingQueue<IEvent>(); + private volatile boolean processMessage = false; + private final Executor msgProcessorExec = Executors.newSingleThreadExecutor(); + private int maxThreshold = 1024; + private Set<IEventListener> listeners; + + public void send(IEvent msg) { + if (receivedMessages.size() > maxThreshold) { + log.warn("Queued number of events [{}] is greater than threshold [{}]", receivedMessages.size(), maxThreshold); + } + receivedMessages.add(msg); + } + + public void stop() { + processMessage = false; + } + + public void start() { + try { + processMessage = true; + + Runnable messageProcessor = new Runnable() { + public void run() { + while (processMessage) { + try { + IEvent msg = receivedMessages.take(); + processMessage(msg); + } catch (InterruptedException e) { + log.warn("Error while taking received message from queue."); + } + } + } + }; + msgProcessorExec.execute(messageProcessor); + } catch (Exception e) { + log.error("Error processing event: " + e.getMessage()); + } + } + + private void processMessage(final IEvent msg) { + for (IEventListener listener : listeners) { + listener.handleMessage(msg); + } + } + + public void setListeners(Set<IEventListener> listeners) { + this.listeners = listeners; + } + + public void setMaxThreshold(int threshold) { + maxThreshold = threshold; + } + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..b6bc34cbc2e4e18f9daa079a90484810b2dbbfb4 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEvent.java @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.events; + +public interface IEvent { + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventListener.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventListener.java new file mode 100755 index 0000000000000000000000000000000000000000..5036e43ea7e67c268f4b5927c74ba63e2f5cb4d7 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventListener.java @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.events; + +public interface IEventListener { + void handleMessage(IEvent msg); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventsMessageBus.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventsMessageBus.java new file mode 100755 index 0000000000000000000000000000000000000000..eea2cab91f8146c8ddd742660e9863777477687a --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/IEventsMessageBus.java @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.events; + +public interface IEventsMessageBus { + void send(IEvent msg); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStartedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStartedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..530e8bfea2afcc8180d44ff3f05a29bb88935147 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStartedEvent.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare.events; + +public class ShareStartedEvent implements IEvent { + + public final String meetingId; + public final String streamId; + + public ShareStartedEvent(String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStoppedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStoppedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..929ae16d8b30cd6aa433fc36c38e30e633ce3db9 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/ShareStoppedEvent.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare.events; + +public class ShareStoppedEvent implements IEvent { + + public final String meetingId; + public final String streamId; + + public ShareStoppedEvent(String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStartedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStartedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..086178a8f29b806a97d6f243b64cde376f585c8c --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStartedEvent.java @@ -0,0 +1,19 @@ +package org.bigbluebutton.app.screenshare.events; + +public class StreamStartedEvent implements IEvent { + + public final String meetingId; + public final String streamId; + public final int width; + public final int height; + public final String url; + + public StreamStartedEvent(String meetingId, String streamId, + int width, int height, String url) { + this.meetingId = meetingId; + this.streamId = streamId; + this.width = width; + this.height = height; + this.url = url; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStoppedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStoppedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..78869cfc0a64e7576cc7b79485659f17ed7e97cb --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamStoppedEvent.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.app.screenshare.events; + +public class StreamStoppedEvent implements IEvent { + + public final String meetingId; + public final String streamId; + + public StreamStoppedEvent(String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamUpdateEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamUpdateEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..75373b163c6e33a56a8d455ad0469acf7112224a --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/events/StreamUpdateEvent.java @@ -0,0 +1,14 @@ +package org.bigbluebutton.app.screenshare.events; + +public class StreamUpdateEvent implements IEvent { + + public final String meetingId; + public final String streamId; + public final Long date; + + public StreamUpdateEvent(String meetingId, String streamId, Long date) { + this.meetingId = meetingId; + this.streamId = streamId; + this.date = date; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MeetingMessageHandler.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MeetingMessageHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..e4bf6fd309e761a3ea532cd608f57ebb587432ed --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MeetingMessageHandler.java @@ -0,0 +1,35 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +import java.util.HashMap; +import java.util.Map; + +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import com.google.gson.Gson; + + +public class MeetingMessageHandler implements MessageHandler { + private static Logger log = Red5LoggerFactory.getLogger(MeetingMessageHandler.class, "screenshare"); + + + @Override + public void handleMessage(String pattern, String channel, String message) { + + if (channel.equalsIgnoreCase(MessagingConstants.TO_MEETING_CHANNEL)) { + +// IMessage msg = MessageFromJsonConverter.convert(message); +// if (msg != null) { + +// } + } else if (channel.equalsIgnoreCase(MessagingConstants.TO_SYSTEM_CHANNEL)) { +// IMessage msg = MessageFromJsonConverter.convert(message); + +// if (msg != null) { +// +// } + } + } + + + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageDistributor.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageDistributor.java new file mode 100755 index 0000000000000000000000000000000000000000..794704bec8156ff8b50bf697df60fd8dea12eddb --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageDistributor.java @@ -0,0 +1,25 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +import java.util.Set; + +public class MessageDistributor { + private ReceivedMessageHandler handler; + private Set<MessageHandler> listeners; + + public void setMessageListeners(Set<MessageHandler> listeners) { + this.listeners = listeners; + } + + public void setMessageHandler(ReceivedMessageHandler handler) { + this.handler = handler; + if (handler != null) { + handler.setMessageDistributor(this); + } + } + + public void notifyListeners(String pattern, String channel, String message) { + for (MessageHandler listener : listeners) { + listener.handleMessage(pattern, channel, message); + } + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageHandler.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..9b4cabcac78bbc5e4d79a9c58885e6ccf46498bf --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageHandler.java @@ -0,0 +1,23 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.messaging.redis; + +public interface MessageHandler { + void handleMessage(String pattern, String channel, String message); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageReceiver.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageReceiver.java new file mode 100755 index 0000000000000000000000000000000000000000..a59df4c096472a43d3737d45c436b4282b286861 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageReceiver.java @@ -0,0 +1,88 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import redis.clients.jedis.Jedis; +import redis.clients.jedis.JedisPool; +import redis.clients.jedis.JedisPubSub; + +public class MessageReceiver { + private static Logger log = Red5LoggerFactory.getLogger(MessageReceiver.class, "bigbluebutton"); + + private ReceivedMessageHandler handler; + + private JedisPool redisPool; + private volatile boolean receiveMessage = false; + + private final Executor msgReceiverExec = Executors.newSingleThreadExecutor(); + + public void stop() { + receiveMessage = false; + } + + public void start() { + log.info("Ready to receive messages from Redis pubsub."); + try { + receiveMessage = true; + final Jedis jedis = redisPool.getResource(); + + Runnable messageReceiver = new Runnable() { + public void run() { + if (receiveMessage) { + jedis.psubscribe(new PubSubListener(), MessagingConstants.TO_BBB_APPS_PATTERN); + } + } + }; + msgReceiverExec.execute(messageReceiver); + } catch (Exception e) { + log.error("Error subscribing to channels: " + e.getMessage()); + } + } + + public void setRedisPool(JedisPool redisPool){ + this.redisPool = redisPool; + } + + public void setMessageHandler(ReceivedMessageHandler handler) { + this.handler = handler; + } + + private class PubSubListener extends JedisPubSub { + + public PubSubListener() { + super(); + } + + @Override + public void onMessage(String channel, String message) { + // Not used. + } + + @Override + public void onPMessage(String pattern, String channel, String message) { + handler.handleMessage(pattern, channel, message); + } + + @Override + public void onPSubscribe(String pattern, int subscribedChannels) { + log.debug("Subscribed to the pattern: " + pattern); + } + + @Override + public void onPUnsubscribe(String pattern, int subscribedChannels) { + // Not used. + } + + @Override + public void onSubscribe(String channel, int subscribedChannels) { + // Not used. + } + + @Override + public void onUnsubscribe(String channel, int subscribedChannels) { + // Not used. + } + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageSender.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageSender.java new file mode 100755 index 0000000000000000000000000000000000000000..18067a0a986a34ae5b7440e0c90decafeb8ef410 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageSender.java @@ -0,0 +1,74 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import redis.clients.jedis.Jedis; +import redis.clients.jedis.JedisPool; + +public class MessageSender { + private static Logger log = Red5LoggerFactory.getLogger(MessageSender.class, "bigbluebutton"); + + private JedisPool redisPool; + private volatile boolean sendMessage = false; + + private final Executor msgSenderExec = Executors.newSingleThreadExecutor(); + private final Executor runExec = Executors.newSingleThreadExecutor(); + private BlockingQueue<MessageToSend> messages = new LinkedBlockingQueue<MessageToSend>(); + + public void stop() { + sendMessage = false; + } + + public void start() { + log.info("Redis message publisher starting!"); + try { + sendMessage = true; + + Runnable messageSender = new Runnable() { + public void run() { + while (sendMessage) { + try { + MessageToSend msg = messages.take(); + publish(msg.getChannel(), msg.getMessage()); + } catch (InterruptedException e) { + log.warn("Failed to get message from queue."); + } + } + } + }; + msgSenderExec.execute(messageSender); + } catch (Exception e) { + log.error("Error subscribing to channels: " + e.getMessage()); + } + } + + public void send(String channel, String message) { + MessageToSend msg = new MessageToSend(channel, message); + messages.add(msg); + } + + private void publish(final String channel, final String message) { + Runnable task = new Runnable() { + public void run() { + Jedis jedis = redisPool.getResource(); + try { + jedis.publish(channel, message); + } catch(Exception e){ + log.warn("Cannot publish the message to redis", e); + } finally { + redisPool.returnResource(jedis); + } + } + }; + + runExec.execute(task); + } + + public void setRedisPool(JedisPool redisPool){ + this.redisPool = redisPool; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageToSend.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageToSend.java new file mode 100755 index 0000000000000000000000000000000000000000..45951b4dccdf449b9fc386b35faf5dac32dde813 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessageToSend.java @@ -0,0 +1,19 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +public class MessageToSend { + private final String channel; + private final String message; + + public MessageToSend(String channel, String message) { + this.channel = channel; + this.message = message; + } + + public String getChannel() { + return channel; + } + + public String getMessage() { + return message; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessagingConstants.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessagingConstants.java new file mode 100755 index 0000000000000000000000000000000000000000..0a787e84d3788d7d562b198694078f545ce84e74 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/MessagingConstants.java @@ -0,0 +1,40 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2014 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.messaging.redis; + +public class MessagingConstants { + + public static final String FROM_BBB_APPS_CHANNEL = "bigbluebutton:from-bbb-apps"; + public static final String FROM_BBB_APPS_PATTERN = FROM_BBB_APPS_CHANNEL + ":*"; + public static final String FROM_SYSTEM_CHANNEL = FROM_BBB_APPS_CHANNEL + ":system"; + public static final String FROM_MEETING_CHANNEL = FROM_BBB_APPS_CHANNEL + ":meeting"; + + public static final String TO_BBB_APPS_CHANNEL = "bigbluebutton:to-bbb-apps"; + public static final String TO_BBB_APPS_PATTERN = TO_BBB_APPS_CHANNEL + ":*"; + public static final String TO_MEETING_CHANNEL = TO_BBB_APPS_CHANNEL + ":meeting"; + public static final String TO_SYSTEM_CHANNEL = TO_BBB_APPS_CHANNEL + ":system"; + + public static final String DESTROY_MEETING_REQUEST_EVENT = "DestroyMeetingRequestEvent"; + public static final String CREATE_MEETING_REQUEST_EVENT = "CreateMeetingRequestEvent"; + public static final String END_MEETING_REQUEST_EVENT = "EndMeetingRequestEvent"; + public static final String MEETING_STARTED_EVENT = "meeting_created_message"; + public static final String MEETING_ENDED_EVENT = "meeting_ended_event"; + public static final String MEETING_DESTROYED_EVENT = "meeting_destroyed_event"; + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..08d4b3996a6c06e70e4ad630cf8b10dd800604d7 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessage.java @@ -0,0 +1,28 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +public class ReceivedMessage { + + private final String pattern; + private final String channel; + private final String message; + + public ReceivedMessage(String pattern, String channel, String message) { + this.pattern = pattern; + this.channel = channel; + this.message = message; + } + + public String getPattern() { + return pattern; + } + + public String getChannel() { + return channel; + } + + public String getMessage() { + return message; + } + + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessageHandler.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessageHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..b713dd3c3f6096da4725640d2219f84811fb20c5 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/messaging/redis/ReceivedMessageHandler.java @@ -0,0 +1,72 @@ +package org.bigbluebutton.app.screenshare.messaging.redis; + +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; + +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + +public class ReceivedMessageHandler { + private static Logger log = Red5LoggerFactory.getLogger(ReceivedMessageHandler.class, "bigbluebutton"); + + private BlockingQueue<ReceivedMessage> receivedMessages = new LinkedBlockingQueue<ReceivedMessage>(); + + private volatile boolean processMessage = false; + + private final Executor msgProcessorExec = Executors.newSingleThreadExecutor(); + private final Executor runExec = Executors.newSingleThreadExecutor(); + + private MessageDistributor handler; + + public void stop() { + processMessage = false; + } + + public void start() { + log.info("Ready to handle messages from Redis pubsub!"); + + try { + processMessage = true; + + Runnable messageProcessor = new Runnable() { + public void run() { + while (processMessage) { + try { + ReceivedMessage msg = receivedMessages.take(); + processMessage(msg); + } catch (InterruptedException e) { + log.warn("Error while taking received message from queue."); + } + } + } + }; + msgProcessorExec.execute(messageProcessor); + } catch (Exception e) { + log.error("Error subscribing to channels: " + e.getMessage()); + } + } + + private void processMessage(final ReceivedMessage msg) { + Runnable task = new Runnable() { + public void run() { + if (handler != null) { + handler.notifyListeners(msg.getPattern(), msg.getChannel(), msg.getMessage()); + } else { + log.info("No listeners interested in messages from Redis!"); + } + } + }; + runExec.execute(task); + } + + public void handleMessage(String pattern, String channel, String message) { + ReceivedMessage rm = new ReceivedMessage(pattern, channel, message); + receivedMessages.add(rm); + } + + public void setMessageDistributor(MessageDistributor h) { + this.handler = h; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/BroadcastClientMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/BroadcastClientMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..0800b43b004f96de6b2cd8da08bff024cc78b666 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/BroadcastClientMessage.java @@ -0,0 +1,46 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.Map; + +public class BroadcastClientMessage implements ClientMessage { + + private String meetingID; + private Map<String, Object> message; + private String messageName; + + public BroadcastClientMessage(String meetingID, String messageName, Map<String, Object> message) { + this.meetingID = meetingID; + this.message = message; + this.messageName = messageName; + } + + public String getMeetingID() { + return meetingID; + } + + public String getMessageName() { + return messageName; + } + + public Map<String, Object> getMessage() { + return message; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ClientMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ClientMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..501256a21dcccd22d633a78e8457a1b862a8b0fc --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ClientMessage.java @@ -0,0 +1,24 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.red5; + + +public interface ClientMessage { + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ConnectionInvokerService.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ConnectionInvokerService.java new file mode 100755 index 0000000000000000000000000000000000000000..12f811f039ce2c3c4f907791de82e4917e444e92 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/ConnectionInvokerService.java @@ -0,0 +1,220 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.Set; +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; +import org.red5.logging.Red5LoggerFactory; +import org.red5.server.api.IConnection; +import org.red5.server.api.scope.IScope; +import org.red5.server.api.scope.ScopeType; +import org.red5.server.api.service.ServiceUtils; +import org.red5.server.api.so.ISharedObject; +import org.red5.server.api.so.ISharedObjectService; +import org.red5.server.so.SharedObjectService; +import org.red5.server.util.ScopeUtils; +import org.slf4j.Logger; + +public class ConnectionInvokerService { + private static Logger log = Red5LoggerFactory.getLogger(ConnectionInvokerService.class, "screenshare"); + + private static final int NTHREADS = 1; + private static final Executor exec = Executors.newFixedThreadPool(NTHREADS); + private static final Executor runExec = Executors.newFixedThreadPool(NTHREADS); + + private BlockingQueue<ClientMessage> messages; + + private volatile boolean sendMessages = false; + private IScope bbbAppScope; + + public ConnectionInvokerService() { + messages = new LinkedBlockingQueue<ClientMessage>(); + } + + public void setAppScope(IScope scope) { + bbbAppScope = scope; + } + + public void start() { + sendMessages = true; + Runnable sender = new Runnable() { + public void run() { + while (sendMessages) { + ClientMessage message; + try { + message = messages.take(); + sendMessageToClient(message); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + + } + } + }; + exec.execute(sender); + } + + public void stop() { + sendMessages = false; + } + + public void sendMessage(final ClientMessage message) { + messages.offer(message); + } + + private void sendMessageToClient(ClientMessage message) { + if (message instanceof BroadcastClientMessage) { + sendBroadcastMessage((BroadcastClientMessage) message); + } else if (message instanceof DirectClientMessage) { + sendDirectMessage((DirectClientMessage) message); + } else if (message instanceof SharedObjectClientMessage) { + sendSharedObjectMessage((SharedObjectClientMessage) message); + } else if (message instanceof DisconnectClientMessage) { + handlDisconnectClientMessage((DisconnectClientMessage) message); + } else if (message instanceof DisconnectAllClientsMessage) { + handleDisconnectAllClientsMessage((DisconnectAllClientsMessage) message); + } + } + + private void handleDisconnectAllClientsMessage(DisconnectAllClientsMessage msg) { + IScope meetingScope = getScope(msg.getMeetingId()); + if (meetingScope != null) { + Set<IConnection> conns = meetingScope.getClientConnections(); + + for (IConnection conn : conns) { + if (conn.isConnected()) { + String connId = (String) conn.getAttribute("INTERNAL_USER_ID"); + log.info("Disconnecting client=[{}] from meeting=[{}]", connId, msg.getMeetingId()); + conn.close(); + } + } + } + } + + private void handlDisconnectClientMessage(DisconnectClientMessage msg) { + IScope meetingScope = getScope(msg.getMeetingId()); + if (meetingScope != null) { + IConnection conn = getConnection(meetingScope, msg.getUserId()); + if (conn != null) { + if (conn.isConnected()) { + log.info("Disconnecting user=[{}] from meeting=[{}]", msg.getUserId(), msg.getMeetingId()); + conn.close(); + } + } + } + } + + private void sendSharedObjectMessage(SharedObjectClientMessage msg) { + System.out.println("*********** Request to send [" + msg.getMessageName() + "] using shared object."); + + IScope meetingScope = getScope(msg.getMeetingID()); + if (meetingScope != null) { + if (meetingScope.hasChildScope(ScopeType.SHARED_OBJECT, msg.getSharedObjectName())) { + ISharedObject so = getSharedObject(meetingScope, msg.getSharedObjectName()); + if (so != null) { + System.out.println("*********** Sending [" + msg.getMessageName() + "] using shared object."); + so.sendMessage(msg.getMessageName(), msg.getMessage()); + } else { + System.out.println("**** Cannot get SO for [" + msg.getSharedObjectName() + "]"); + } + } else { + System.out.println("**** No SO scope for [" + msg.getSharedObjectName() + "]"); + } + } else { + System.out.println("**** No Meeting scope for [" + msg.getMeetingID() + "]"); + } + } + + private void sendDirectMessage(final DirectClientMessage msg) { + Runnable sender = new Runnable() { + public void run() { + IScope meetingScope = getScope(msg.getMeetingID()); + if (meetingScope != null) { + log.debug("Found scope =[{}] for meeting=[{}]", meetingScope.getName(), msg.getMeetingID()); + IConnection conn = getConnection(meetingScope, msg.getUserID()); + if (conn != null) { + if (conn.isConnected()) { + List<Object> params = new ArrayList<Object>(); + params.add(msg.getMessageName()); + params.add(msg.getMessage()); + log.debug("Sending message=[{}] to meeting=[{}]", msg.getMessageName(), msg.getMeetingID()); + ServiceUtils.invokeOnConnection(conn, "onMessageFromServer", params.toArray()); + } else { + log.warn("Connection not connected for userid=[{}] in meeting=[{}]", msg.getUserID(), msg.getMeetingID()); + } + } else { + log.warn("No connection for userid=[{}] in meeting=[{}]", msg.getUserID(), msg.getMeetingID()); + } + } else { + log.error("Failed to find scope for meeting=[{}]", msg.getMeetingID()); + } + } + }; + runExec.execute(sender); + } + + private void sendBroadcastMessage(final BroadcastClientMessage msg) { + Runnable sender = new Runnable() { + public void run() { + IScope meetingScope = getScope(msg.getMeetingID()); + if (meetingScope != null) { + List<Object> params = new ArrayList<Object>(); + params.add(msg.getMessageName()); + params.add(msg.getMessage()); + ServiceUtils.invokeOnAllScopeConnections(meetingScope, "onMessageFromServer", params.toArray(), null); + } + } + }; + runExec.execute(sender); + } + + private IConnection getConnection(IScope scope, String userID) { + Set<IConnection> conns = scope.getClientConnections(); + for (IConnection conn : conns) { + String connID = (String) conn.getAttribute("USERID"); + if (connID != null && connID.equals(userID)) { + return conn; + } + } + + return null; + } + + public IScope getScope(String meetingID) { + if (bbbAppScope != null) { + return bbbAppScope.getContext().resolveScope("screenshare/" + meetingID); + } else { + log.error("BigBlueButton Scope not initialized. No messages are going to the Flash client!"); + } + + return null; + } + + private ISharedObject getSharedObject(IScope scope, String name) { + ISharedObjectService service = (ISharedObjectService) ScopeUtils.getScopeService(scope, ISharedObjectService.class, SharedObjectService.class, false); + return service.getSharedObject(scope, name); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DirectClientMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DirectClientMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..c84773194ef564dab2347ca2d36414ec35d344c2 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DirectClientMessage.java @@ -0,0 +1,61 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.Map; + +public class DirectClientMessage implements ClientMessage { + + private String meetingID; + private String userID; + private Map<String, Object> message; + private String messageName; + private String sharedObjectName; + + public DirectClientMessage(String meetingID, String userID, String messageName, Map<String, Object> message) { + this.meetingID = meetingID; + this.userID = userID; + this.message = message; + this.messageName = messageName; + } + + public void setSharedObjectName(String name) { + sharedObjectName = name; + } + + public String getSharedObjectName() { + return sharedObjectName; + } + + public String getMeetingID() { + return meetingID; + } + + public String getUserID() { + return userID; + } + + public String getMessageName() { + return messageName; + } + + public Map<String, Object> getMessage() { + return message; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectAllClientsMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectAllClientsMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..cabe2d3141e80f55644230bdf614aad721539f40 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectAllClientsMessage.java @@ -0,0 +1,14 @@ +package org.bigbluebutton.app.screenshare.red5; + +public class DisconnectAllClientsMessage implements ClientMessage { + + private final String meetingId; + + public DisconnectAllClientsMessage(String meetingId) { + this.meetingId = meetingId; + } + + public String getMeetingId() { + return meetingId; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectClientMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectClientMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..64352951631fd7b04de70ccf1a58667d8ca320e7 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/DisconnectClientMessage.java @@ -0,0 +1,20 @@ +package org.bigbluebutton.app.screenshare.red5; + +public class DisconnectClientMessage implements ClientMessage { + + private final String meetingId; + private final String userId; + + public DisconnectClientMessage(String meetingId, String userId) { + this.meetingId = meetingId; + this.userId = userId; + } + + public String getMeetingId() { + return meetingId; + } + + public String getUserId() { + return userId; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/EventListenerImp.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/EventListenerImp.java new file mode 100755 index 0000000000000000000000000000000000000000..08f675428da7cd8fb76dd9f96ecf4e7c25588ae9 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/EventListenerImp.java @@ -0,0 +1,91 @@ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.HashMap; +import java.util.Map; + +import org.bigbluebutton.app.screenshare.events.IEvent; +import org.bigbluebutton.app.screenshare.events.IEventListener; +import org.bigbluebutton.app.screenshare.events.ShareStartedEvent; +import org.bigbluebutton.app.screenshare.events.ShareStoppedEvent; +import org.bigbluebutton.app.screenshare.events.StreamStartedEvent; +import org.bigbluebutton.app.screenshare.events.StreamStoppedEvent; + +import com.google.gson.Gson; + +public class EventListenerImp implements IEventListener { + private ConnectionInvokerService sender; + + @Override + public void handleMessage(IEvent event) { + if (event instanceof ShareStartedEvent) { + sendShareStartedEvent((ShareStartedEvent) event); + } else if (event instanceof ShareStoppedEvent) { + sendShareStoppedEvent((ShareStoppedEvent) event); + } else if (event instanceof StreamStartedEvent) { + sendStreamStartedEvent((StreamStartedEvent) event); + } else if (event instanceof StreamStoppedEvent) { + sendStreamStoppedEvent((StreamStoppedEvent) event); + } + + } + + private void sendShareStartedEvent(ShareStartedEvent event) { + Map<String, Object> data = new HashMap<String, Object>(); + data.put("meetingId", event.meetingId); + data.put("streamId", event.streamId); + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + BroadcastClientMessage msg = new BroadcastClientMessage(event.meetingId, "screenShareStartedMessage", message); + sender.sendMessage(msg); + } + + private void sendShareStoppedEvent(ShareStoppedEvent event) { + Map<String, Object> data = new HashMap<String, Object>(); + data.put("meetingId", event.meetingId); + data.put("streamId", event.streamId); + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + BroadcastClientMessage msg = new BroadcastClientMessage(event.meetingId, "screenShareStoppedMessage", message); + sender.sendMessage(msg); + } + + private void sendStreamStartedEvent(StreamStartedEvent event) { + Map<String, Object> data = new HashMap<String, Object>(); + data.put("meetingId", event.meetingId); + data.put("streamId", event.streamId); + data.put("width", event.width); + data.put("height", event.height); + data.put("url", event.url); + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + BroadcastClientMessage msg = new BroadcastClientMessage(event.meetingId, "screenStreamStartedMessage", message); + sender.sendMessage(msg); + } + + private void sendStreamStoppedEvent(StreamStoppedEvent event) { + Map<String, Object> data = new HashMap<String, Object>(); + data.put("meetingId", event.meetingId); + data.put("streamId", event.streamId); + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + BroadcastClientMessage msg = new BroadcastClientMessage(event.meetingId, "screenStreamStoppedMessage", message); + sender.sendMessage(msg); + } + + public void setMessageSender(ConnectionInvokerService sender) { + this.sender = sender; + } + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppAdapter.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppAdapter.java new file mode 100755 index 0000000000000000000000000000000000000000..ccbf050f2055d43979de51143577874676ed4c98 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppAdapter.java @@ -0,0 +1,274 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.TimeUnit; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +import org.red5.logging.Red5LoggerFactory; +import org.red5.server.adapter.MultiThreadedApplicationAdapter; +import org.red5.server.api.IConnection; +import org.red5.server.api.Red5; +import org.red5.server.api.scope.IScope; +import org.red5.server.api.stream.IBroadcastStream; +import org.red5.server.api.stream.IServerStream; +import org.red5.server.api.stream.IStreamListener; +import org.red5.server.stream.ClientBroadcastStream; +import org.slf4j.Logger; + +import com.google.gson.Gson; + +import org.bigbluebutton.app.screenshare.EventRecordingService; +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.bigbluebutton.app.screenshare.ScreenshareStreamListener; + +public class Red5AppAdapter extends MultiThreadedApplicationAdapter { + private static Logger log = Red5LoggerFactory.getLogger(Red5AppAdapter.class, "screenshare"); + + private EventRecordingService recordingService; + private final Map<String, IStreamListener> streamListeners = new HashMap<String, IStreamListener>(); + + private IScreenShareApplication app; + private String streamBaseUrl; + private ConnectionInvokerService sender; + private String recordingDirectory; + + @Override + public boolean appStart(IScope app) { + super.appStart(app); + log.info("BBB Screenshare appStart"); + sender.setAppScope(app); + return true; + } + + @Override + public boolean appConnect(IConnection conn, Object[] params) { + log.info("BBB Screenshare appConnect"); + return super.appConnect(conn, params); + } + + @Override + public boolean roomConnect(IConnection conn, Object[] params) { + log.info("BBB Screenshare roomConnect"); + return super.roomConnect(conn, params); + } + + private String getConnectionType(String connType) { + if ("persistent".equals(connType.toLowerCase())) { + return "RTMP"; + } else if("polling".equals(connType.toLowerCase())) { + return "RTMPT"; + } else { + return connType.toUpperCase(); + } + } + + private String getUserId() { + String userid = (String) Red5.getConnectionLocal().getAttribute("USERID"); + if ((userid == null) || ("".equals(userid))) userid = "unknown-userid"; + return userid; + } + + private String getMeetingId() { + String meetingId = (String) Red5.getConnectionLocal().getAttribute("MEETING_ID"); + if ((meetingId == null) || ("".equals(meetingId))) meetingId = "unknown-meetingid"; + return meetingId; + } + + @Override + public void appDisconnect(IConnection conn) { + log.info("BBB Screenshare appDisconnect"); + + String connType = getConnectionType(Red5.getConnectionLocal().getType()); + String connId = Red5.getConnectionLocal().getSessionId(); + + Map<String, Object> logData = new HashMap<String, Object>(); + logData.put("meetingId", getMeetingId()); + logData.put("userId", getUserId()); + logData.put("connType", connType); + logData.put("connId", connId); + logData.put("event", "user_leaving_bbb_screenshare"); + logData.put("description", "User leaving BBB Screenshare."); + + Gson gson = new Gson(); + String logStr = gson.toJson(logData); + + log.info("User leaving bbb-screenshare: data={}", logStr); + + super.appDisconnect(conn); + } + + @Override + public void roomDisconnect(IConnection conn) { + log.info("BBB Screenshare roomDisconnect"); + + String connType = getConnectionType(Red5.getConnectionLocal().getType()); + String connId = Red5.getConnectionLocal().getSessionId(); + + String meetingId = conn.getScope().getName(); + String userId = getUserId(); + + app.userDisconnected(meetingId, userId); + + Map<String, Object> logData = new HashMap<String, Object>(); + logData.put("meetingId", getMeetingId()); + logData.put("userId", userId); + logData.put("connType", connType); + logData.put("connId", connId); + logData.put("event", "user_leaving_bbb_screenshare"); + logData.put("description", "User leaving BBB Screenshare."); + + Gson gson = new Gson(); + String logStr = gson.toJson(logData); + + log.info("User leaving bbb-screenshare: data={}", logStr); + + super.roomDisconnect(conn); + } + + @Override + public void streamPublishStart(IBroadcastStream stream) { + super.streamPublishStart(stream); + } + + @Override + public void streamBroadcastStart(IBroadcastStream stream) { + IConnection conn = Red5.getConnectionLocal(); + super.streamBroadcastStart(stream); + + log.info("streamBroadcastStart " + stream.getPublishedName() + "]"); + String streamId = stream.getPublishedName(); + Matcher matcher = STREAM_ID_PATTERN.matcher(stream.getPublishedName()); + if (matcher.matches()) { + String meetingId = matcher.group(1).trim(); + String url = streamBaseUrl + "/" + meetingId + "/" + streamId; + app.streamStarted(meetingId, streamId, url); + + boolean recordVideoStream = app.recordStream(meetingId, streamId); + if (recordVideoStream) { + recordStream(stream); + ScreenshareStreamListener listener = new ScreenshareStreamListener(recordingService, recordingDirectory); + stream.addStreamListener(listener); + streamListeners.put(conn.getScope().getName() + "-" + stream.getPublishedName(), listener); + } + } else { + log.error("Invalid streamid format [{}]", streamId); + } + } + + private Long genTimestamp() { + return TimeUnit.NANOSECONDS.toMillis(System.nanoTime()); + } + + private final Pattern STREAM_ID_PATTERN = Pattern.compile("(.*)-(.*)$"); + + @Override + public void streamBroadcastClose(IBroadcastStream stream) { + super.streamBroadcastClose(stream); + + log.info("streamBroadcastStop " + stream.getPublishedName() + "]"); + String streamId = stream.getPublishedName(); + Matcher matcher = STREAM_ID_PATTERN.matcher(stream.getPublishedName()); + if (matcher.matches()) { + String meetingId = matcher.group(1).trim(); + app.streamStopped(meetingId, streamId); + + boolean recordVideoStream = app.recordStream(meetingId, streamId); + if (recordVideoStream) { + IConnection conn = Red5.getConnectionLocal(); + String scopeName; + if (conn != null) { + scopeName = conn.getScope().getName(); + } else { + log.info("Connection local was null, using scope name from the stream: {}", stream); + scopeName = stream.getScope().getName(); + } + IStreamListener listener = streamListeners.remove(scopeName + "-" + stream.getPublishedName()); + if (listener != null) { + stream.removeStreamListener(listener); + } + + String filename = recordingDirectory; + if (!filename.endsWith("/")) { + filename.concat("/"); + } + + filename = filename.concat(meetingId).concat("/").concat(stream.getPublishedName()).concat(".flv"); + + long publishDuration = (System.currentTimeMillis() - stream.getCreationTime()) / 1000; + log.info("streamBroadcastClose " + stream.getPublishedName() + " " + System.currentTimeMillis() + " " + scopeName); + Map<String, String> event = new HashMap<String, String>(); + event.put("module", "Deskshare"); + event.put("timestamp", new Long(System.currentTimeMillis()).toString()); + event.put("meetingId", scopeName); + event.put("stream", stream.getPublishedName()); + event.put("file", filename); + event.put("duration", new Long(publishDuration).toString()); + event.put("eventName", "DeskshareStoppedEvent"); + recordingService.record(scopeName, event); + } + } else { + log.error("Invalid streamid format [{}]", streamId); + } + } + + /** + * A hook to record a stream. A file is written in webapps/video/streams/ + * @param stream + */ + private void recordStream(IBroadcastStream stream) { + IConnection conn = Red5.getConnectionLocal(); + long now = System.currentTimeMillis(); + String recordingStreamName = stream.getPublishedName(); // + "-" + now; /** Comment out for now...forgot why I added this - ralam */ + + try { + log.info("Recording stream " + recordingStreamName ); + ClientBroadcastStream cstream = (ClientBroadcastStream) this.getBroadcastStream(conn.getScope(), stream.getPublishedName()); + cstream.saveAs(recordingStreamName, false); + } catch(Exception e) { + log.error("ERROR while recording stream " + e.getMessage()); + e.printStackTrace(); + } + } + + + + public void setEventRecordingService(EventRecordingService s) { + recordingService = s; + } + + public void setStreamBaseUrl(String baseUrl) { + streamBaseUrl = baseUrl; + } + + public void setRecordingDirectory(String dir) { + recordingDirectory = dir; + } + + public void setApplication(IScreenShareApplication app) { + this.app = app; + } + + public void setMessageSender(ConnectionInvokerService sender) { + this.sender = sender; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppHandler.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..b25fd74529a0971a69712862892dd0a664f3279b --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppHandler.java @@ -0,0 +1,93 @@ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.HashMap; +import java.util.Map; +import java.util.regex.Matcher; +import java.util.regex.Pattern; +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.bigbluebutton.app.screenshare.IsScreenSharingResponse; +import org.bigbluebutton.app.screenshare.StartShareRequestResponse; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import com.google.gson.Gson; + +public class Red5AppHandler { + private static Logger log = Red5LoggerFactory.getLogger(Red5AppHandler.class, "screenshare"); + + private IScreenShareApplication app; + private ConnectionInvokerService sender; + + private final Pattern STREAM_ID_PATTERN = Pattern.compile("(.*)-(.*)$"); + + public void isScreenSharing(String meetingId, String userId) { + IsScreenSharingResponse resp = app.isScreenSharing(meetingId); + + Map<String, Object> data = new HashMap<String, Object>(); + data.put("sharing", resp.info.sharing); + + if (resp.info.sharing) { + data.put("streamId", resp.info.streamId); + data.put("width", resp.info.width); + data.put("height", resp.info.height); + data.put("url", resp.info.url); + } + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + log.info("Sending isSharingScreenRequestResponse to client, meetingId=" + meetingId + " userid=" + userId); + DirectClientMessage msg = new DirectClientMessage(meetingId, userId, "isSharingScreenRequestResponse", message); + sender.sendMessage(msg); + } + + public void startShareRequest(String meetingId, String userId, Boolean record) { + StartShareRequestResponse resp = app.startShareRequest(meetingId, userId, record); + + Map<String, Object> data = new HashMap<String, Object>(); + + if (resp.error != null) { + data.put("error", resp.error.reason); + } else { + data.put("authToken", resp.token); + data.put("jnlp", resp.jnlp); + } + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + log.info("Sending startShareRequestResponse to client, meetingId=" + meetingId + " userid=" + userId); + DirectClientMessage msg = new DirectClientMessage(meetingId, userId, "startShareRequestResponse", message); + sender.sendMessage(msg); + } + + public void stopShareRequest(String meetingId, String streamId) { + Matcher matcher = STREAM_ID_PATTERN.matcher(streamId); + if (matcher.matches()) { + app.stopShareRequest(meetingId, streamId); + } + + Map<String, Object> data = new HashMap<String, Object>(); + data.put("meetingId", meetingId); + data.put("streamId", streamId); + + Map<String, Object> message = new HashMap<String, Object>(); + Gson gson = new Gson(); + message.put("msg", gson.toJson(data)); + + log.info("Sending stopShareRequest to client, meetingId=" + meetingId + " streamId=" + streamId); + BroadcastClientMessage msg = new BroadcastClientMessage(meetingId, "stopViewingStream", message); + sender.sendMessage(msg); + } + + public void setApplication(IScreenShareApplication app) { + this.app = app; + } + + public void setMessageSender(ConnectionInvokerService sender) { + this.sender = sender; + } + + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppService.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppService.java new file mode 100755 index 0000000000000000000000000000000000000000..2cde72d2701c8199f685a29f5c779d13648f101c --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/Red5AppService.java @@ -0,0 +1,87 @@ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.HashMap; +import java.util.Map; + +import org.red5.logging.Red5LoggerFactory; +import org.red5.server.api.Red5; +import org.slf4j.Logger; + +import com.google.gson.Gson; + + +public class Red5AppService { + private static Logger log = Red5LoggerFactory.getLogger(Red5AppService.class, "screenshare"); + + private Red5AppHandler handler; + + /** + * Called from the client to pass us the userId. + * + * We need to to this as we can't have params on the connect call + * as FFMeeg won't be able to connect. + * @param userId + */ + public void setUserId(Map<String, Object> msg) { + String meetingId = Red5.getConnectionLocal().getScope().getName(); + String userId = (String) msg.get("userId"); + Red5.getConnectionLocal().setAttribute("MEETING_ID", meetingId); + Red5.getConnectionLocal().setAttribute("USERID", userId); + + String connType = getConnectionType(Red5.getConnectionLocal().getType()); + String connId = Red5.getConnectionLocal().getSessionId(); + + Map<String, Object> logData = new HashMap<String, Object>(); + logData.put("meetingId", meetingId); + logData.put("userId", userId); + logData.put("connType", connType); + logData.put("connId", connId); + logData.put("event", "user_joining_bbb_screenshare"); + logData.put("description", "User joining BBB Screenshare."); + + Gson gson = new Gson(); + String logStr = gson.toJson(logData); + + log.info("User joining bbb-screenshare: data={}", logStr); + } + + private String getConnectionType(String connType) { + if ("persistent".equals(connType.toLowerCase())) { + return "RTMP"; + } else if("polling".equals(connType.toLowerCase())) { + return "RTMPT"; + } else { + return connType.toUpperCase(); + } + } + + public void isScreenSharing(Map<String, Object> msg) { + String meetingId = Red5.getConnectionLocal().getScope().getName(); + log.debug("Received check if publishing for meeting=[{}]", meetingId); + String userId = (String) Red5.getConnectionLocal().getAttribute("USERID"); + + handler.isScreenSharing(meetingId, userId); + } + + public void startShareRequest(Map<String, Object> msg) { + Boolean record = (Boolean) msg.get("record"); + String meetingId = Red5.getConnectionLocal().getScope().getName(); + log.debug("Received startShareRequest for meeting=[{}]", meetingId); + String userId = (String) Red5.getConnectionLocal().getAttribute("USERID"); + + handler.startShareRequest(meetingId, userId, record); + } + + public void stopShareRequest(Map<String, Object> msg) { + String meetingId = Red5.getConnectionLocal().getScope().getName(); + String streamId = (String) msg.get("streamId"); + log.debug("Received stopShareRequest for meeting=[{}]", meetingId); + + handler.stopShareRequest(meetingId, streamId); + } + + + public void setAppHandler(Red5AppHandler handler) { + this.handler = handler; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/SharedObjectClientMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/SharedObjectClientMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..054cfdbaf506f2c8c0318cfe67875b7b3c5a454c --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/red5/SharedObjectClientMessage.java @@ -0,0 +1,59 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.red5; + +import java.util.ArrayList; + +public class SharedObjectClientMessage implements ClientMessage { + public static final String BROADCAST = "broadcast"; + public static final String DIRECT = "direct"; + public static final String SHAREDOBJECT = "sharedobject"; + + private String meetingID; + private String sharedObjectName; + private ArrayList<Object> message; + private String messageName; + + public SharedObjectClientMessage(String meetingID, String sharedObjectName, String messageName, ArrayList<Object> message) { + this.meetingID = meetingID; + this.message = message; + this.sharedObjectName = sharedObjectName; + this.messageName = messageName; + } + + public void setSharedObjectName(String name) { + sharedObjectName = name; + } + + public String getSharedObjectName() { + return sharedObjectName; + } + + public String getMeetingID() { + return meetingID; + } + + public String getMessageName() { + return messageName; + } + + public ArrayList<Object> getMessage() { + return message; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureEndMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureEndMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..46b0b8ad9793ed25f5d903a247c6f2ca647a35aa --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureEndMessage.java @@ -0,0 +1,38 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.messages; + +public class CaptureEndMessage { + + private final String room; + private final int sequenceNum; + + public CaptureEndMessage(String room, int sequenceNum) { + this.room = room; + this.sequenceNum = sequenceNum; + } + + public String getRoom() { + return room; + } + + public int getSequenceNum() { + return sequenceNum; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureStartMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureStartMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..592c9b6eb350e1c13c4ce76c8851b099b3ccb74e --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureStartMessage.java @@ -0,0 +1,58 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.messages; + +import org.bigbluebutton.app.screenshare.server.session.Dimension; + +public class CaptureStartMessage { + + private final String room; + private final Dimension screenDim; + private final Dimension blockDim; + private final int sequenceNum; + private final boolean useSVC2; + + public CaptureStartMessage(String room, Dimension screen, Dimension block, int sequenceNum, boolean useSVC2) { + this.room = room; + screenDim = screen; + blockDim = block; + this.sequenceNum = sequenceNum; + this.useSVC2 = useSVC2; + } + + public Dimension getScreenDimension() { + return screenDim; + } + + public Dimension getBlockDimension() { + return blockDim; + } + + public String getRoom() { + return room; + } + + public int getSequenceNum() { + return sequenceNum; + } + + public boolean isUseSVC2() { + return useSVC2; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureUpdateMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureUpdateMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..1e4c704a28872eb9e0ede0ec0a2b41aa278fb3a8 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/CaptureUpdateMessage.java @@ -0,0 +1,55 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.messages; + +public class CaptureUpdateMessage { + private final String room; + private final int position; + private final byte[] videoData; + private final boolean isKeyFrame; + private final int sequenceNum; + + public CaptureUpdateMessage(String room, int position, byte[] videoData, boolean isKeyFrame, int sequenceNum) { + this.room = room; + this.position = position; + this.videoData = videoData; + this.isKeyFrame = isKeyFrame; + this.sequenceNum = sequenceNum; + } + + public String getRoom() { + return room; + } + + public int getPosition() { + return position; + } + + public byte[] getVideoData() { + return videoData; + } + + public boolean isKeyFrame() { + return isKeyFrame; + } + + public int getSequenceNum() { + return sequenceNum; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/MouseLocationMessage.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/MouseLocationMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..3a394d072eadbf9e5e8e2314def2c9873f2a6003 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/messages/MouseLocationMessage.java @@ -0,0 +1,46 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.messages; + +import java.awt.Point; + +public class MouseLocationMessage { + + private String room; + private Point loc; + private final int sequenceNum; + + public MouseLocationMessage(String room, Point loc, int sequenceNum) { + this.room = room; + this.loc = loc; + this.sequenceNum = sequenceNum; + } + + public String getRoom() { + return room; + } + + public Point getLoc() { + return loc; + } + + public int getSequenceNum() { + return sequenceNum; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/EventRecorder.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/EventRecorder.java new file mode 100755 index 0000000000000000000000000000000000000000..ec4452c66ea6600f878cfd91355bd3283aed6aa2 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/EventRecorder.java @@ -0,0 +1,59 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder; + +import java.util.concurrent.TimeUnit; + +import org.bigbluebutton.app.screenshare.server.recorder.event.AbstractDeskshareRecordEvent; +import org.bigbluebutton.app.screenshare.server.recorder.event.RecordEvent; +import org.bigbluebutton.app.screenshare.server.recorder.event.RecordStartedEvent; +import org.bigbluebutton.app.screenshare.server.recorder.event.RecordStoppedEvent; + +import redis.clients.jedis.Jedis; + +public class EventRecorder implements RecordStatusListener { + private static final String COLON=":"; + private String host; + private int port; + + public EventRecorder(String host, int port){ + this.host = host; + this.port = port; + } + + private Long genTimestamp() { + return TimeUnit.NANOSECONDS.toMillis(System.nanoTime()); + } + + private void record(String session, RecordEvent message) { + Jedis jedis = new Jedis(host, port); + Long msgid = jedis.incr("global:nextRecordedMsgId"); + jedis.hmset("recording" + COLON + session + COLON + msgid, message.toMap()); + jedis.rpush("meeting" + COLON + session + COLON + "recordings", msgid.toString()); + } + + @Override + public void notify(RecordEvent event) { + if ((event instanceof RecordStoppedEvent) || (event instanceof RecordStartedEvent)) { + event.setTimestamp(genTimestamp()); + event.setMeetingId(((AbstractDeskshareRecordEvent)event).getSession()); + record(((AbstractDeskshareRecordEvent)event).getSession(), event); + } + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListener.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListener.java new file mode 100755 index 0000000000000000000000000000000000000000..517426e07c65e8fd646cd8aa46b4ff92b443de84 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListener.java @@ -0,0 +1,25 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder; + +import org.bigbluebutton.app.screenshare.server.recorder.event.RecordEvent; + +public interface RecordStatusListener { + void notify(RecordEvent event); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListeners.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListeners.java new file mode 100755 index 0000000000000000000000000000000000000000..f2f5b15c70ba88661f5af5f561d91f59be8542ca --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordStatusListeners.java @@ -0,0 +1,42 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder; + +import java.util.HashSet; +import java.util.Set; + +import org.bigbluebutton.app.screenshare.server.recorder.event.RecordEvent; + +public class RecordStatusListeners { + private final Set<RecordStatusListener> listeners = new HashSet<RecordStatusListener>(); + + public void addListener(RecordStatusListener l) { + listeners.add(l); + } + + public void removeListener(RecordStatusListener l) { + listeners.remove(l); + } + + public void notifyListeners(RecordEvent event) { + for (RecordStatusListener listener: listeners) { + listener.notify(event); + } + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/Recorder.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/Recorder.java new file mode 100755 index 0000000000000000000000000000000000000000..b98edc86517f2ccad2b7a2fb519709e9e25ba1b9 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/Recorder.java @@ -0,0 +1,29 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder; + +import org.apache.mina.core.buffer.IoBuffer; + +public interface Recorder { + public void record(IoBuffer frame); + public void start(); + public void stop(); + public void addListener(RecordStatusListener l); + public void removeListener(RecordStatusListener l); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordingService.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordingService.java new file mode 100755 index 0000000000000000000000000000000000000000..3f464f4a0211e0b176d9931be93d1bedf404e69d --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/RecordingService.java @@ -0,0 +1,28 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder; + +public interface RecordingService { + /** + * Get a recorder for a particular stream + * @param name the name of the stream + * @return the recorder for the stream + */ + Recorder getRecorderFor(String name); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/AbstractDeskshareRecordEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/AbstractDeskshareRecordEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..74ff65fd82d6838133b99c23a08e11fb3b5a5cd2 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/AbstractDeskshareRecordEvent.java @@ -0,0 +1,33 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +public class AbstractDeskshareRecordEvent extends RecordEvent { + + private String session; + + public AbstractDeskshareRecordEvent(String session) { + setModule("Deskshare"); + this.session = session; + } + + public String getSession() { + return session; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordErrorEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordErrorEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..9fcdf4e09405eb495a49cc89f3a35aeae74f8f11 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordErrorEvent.java @@ -0,0 +1,37 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +public class RecordErrorEvent extends AbstractDeskshareRecordEvent { + + private String reason; + + public RecordErrorEvent(String session) { + super(session); + } + + public void setReason(String reason) { + this.reason = reason; + } + + public String getReason() { + return reason; + } + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..653552f51ac1a82c5a9341bb21ce244093ca2b82 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordEvent.java @@ -0,0 +1,81 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +import java.util.HashMap; + +/** + * Abstract class for all events that need to be recorded. + * @author Richard Alam + * + */ +public abstract class RecordEvent { + protected final HashMap<String, String> eventMap = new HashMap<String, String>(); + + protected final static String MODULE = "module"; + protected final static String TIMESTAMP = "timestamp"; + protected final static String MEETING = "meetingId"; + protected final static String EVENT = "eventName"; + + /** + * Set the module that generated the event. + * @param module + */ + public final void setModule(String module) { + eventMap.put(MODULE, module); + } + + /** + * Set the timestamp of the event. + * @param timestamp + */ + public final void setTimestamp(long timestamp) { + eventMap.put(TIMESTAMP, Long.toString(timestamp)); + } + + /** + * Set the meetingId for this particular event. + * @param meetingId + */ + public final void setMeetingId(String meetingId) { + eventMap.put(MEETING, meetingId); + } + + /** + * Set the name of the event. + * @param event + */ + public final void setEvent(String event) { + eventMap.put(EVENT, event); + } + + + /** + * Convert the event into a Map to be recorded. + * @return + */ + public final HashMap<String, String> toMap() { + return eventMap; + } + + @Override + public String toString() { + return eventMap.get(MODULE) + " " + eventMap.get(EVENT); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStartedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStartedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..41d6c26a44b727761872a2c89afec3be6015dab9 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStartedEvent.java @@ -0,0 +1,31 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +public class RecordStartedEvent extends AbstractDeskshareRecordEvent { + + public RecordStartedEvent(String session) { + super(session); + setEvent("DeskshareStartedEvent"); + } + + public void setFile(String path) { + eventMap.put("file", path); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStoppedEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStoppedEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..18a856c2e5791a18cfaf1d87270ce42554dc2eec --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordStoppedEvent.java @@ -0,0 +1,31 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +public class RecordStoppedEvent extends AbstractDeskshareRecordEvent { + + public RecordStoppedEvent(String session) { + super(session); + setEvent("DeskshareStoppedEvent"); + } + + public void setFile(String path) { + eventMap.put("file", path); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordUpdateEvent.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordUpdateEvent.java new file mode 100755 index 0000000000000000000000000000000000000000..855af46cfa5c590a501c878d752fefe231cd905c --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/recorder/event/RecordUpdateEvent.java @@ -0,0 +1,25 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.recorder.event; + +public class RecordUpdateEvent extends AbstractDeskshareRecordEvent { + public RecordUpdateEvent(String session) { + super(session); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/HttpTunnelStreamController.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/HttpTunnelStreamController.java new file mode 100755 index 0000000000000000000000000000000000000000..0f6655b8bd1c12121a409992c0ab679b0d6dbed4 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/HttpTunnelStreamController.java @@ -0,0 +1,135 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.servlet; + +import java.util.*; +import java.awt.Point; +import javax.servlet.ServletContext; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; + +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.bigbluebutton.app.screenshare.server.session.Dimension; +import org.bigbluebutton.app.screenshare.server.session.ISessionManagerGateway; +import org.bigbluebutton.app.screenshare.server.socket.BlockStreamEventMessageHandler; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import org.springframework.context.ApplicationContext; +import org.springframework.http.HttpStatus; +import org.springframework.web.context.WebApplicationContext; +import org.springframework.web.multipart.MultipartFile; +import org.springframework.web.multipart.MultipartHttpServletRequest; +import org.springframework.web.servlet.ModelAndView; +import org.springframework.web.servlet.mvc.multiaction.MultiActionController; + +public class HttpTunnelStreamController extends MultiActionController { + final private Logger log = Red5LoggerFactory.getLogger(HttpTunnelStreamController.class, "screenshare"); + + private boolean hasSessionManager = false; + private IScreenShareApplication screenShareApplication; + + public ModelAndView screenCaptureHandler(HttpServletRequest request, HttpServletResponse response) throws Exception { + String event = request.getParameterValues("event")[0]; + int captureRequest = Integer.parseInt(event); + + if (0 == captureRequest) { + handleCaptureStartRequest(request, response); + response.setStatus(HttpServletResponse.SC_OK); + } else if (1 == captureRequest) { + handleCaptureUpdateRequest(request, response); + response.setStatus(HttpServletResponse.SC_OK); + if (isSharingStopped(request, response)) { + response.setStatus(HttpServletResponse.SC_NOT_FOUND); + } + } else if (2 == captureRequest) { + handleCaptureEndRequest(request, response); + response.setStatus(HttpServletResponse.SC_OK); + } else { + log.warn("Cannot handle screen capture event " + captureRequest); + response.setStatus(HttpServletResponse.SC_OK); + } + return null; + } + + private Boolean isSharingStopped(HttpServletRequest request, HttpServletResponse response) throws Exception { + String meetingId = request.getParameterValues("meetingId")[0]; + String streamId = request.getParameterValues("streamId")[0]; + boolean stopped = screenShareApplication.isSharingStopped(meetingId, streamId); + if (stopped) { + log.info("Screensharing for stream={} has stopped.", streamId); + } + + return stopped; + } + + private void handleCaptureStartRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { + String meetingId = request.getParameterValues("meetingId")[0]; + String streamId = request.getParameterValues("streamId")[0]; + String screenInfo = request.getParameterValues("screenInfo")[0]; + + String[] screen = screenInfo.split("x"); + + if (! hasSessionManager) { + screenShareApplication = getScreenShareApplication(); + hasSessionManager = true; + } + screenShareApplication.sharingStarted(meetingId, streamId, Integer.parseInt(screen[0]), Integer.parseInt(screen[1])); + } + + private void handleCaptureUpdateRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { + String meetingId = request.getParameterValues("meetingId")[0]; + String streamId = request.getParameterValues("streamId")[0]; + + log.debug("Received stream update message for meetingId={} streamId={}", meetingId, streamId); + + if (! hasSessionManager) { + screenShareApplication = getScreenShareApplication(); + hasSessionManager = true; + } + + screenShareApplication.updateShareStatus(meetingId, streamId, 0); + + } + + private void handleCaptureEndRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { + String meetingId = request.getParameterValues("meetingId")[0]; + String streamId = request.getParameterValues("streamId")[0]; + + if (! hasSessionManager) { + screenShareApplication = getScreenShareApplication(); + hasSessionManager = true; + } + System.out.println("HttpTunnel: Received Capture Enfd Event."); + screenShareApplication.sharingStopped(meetingId, streamId); + } + + private IScreenShareApplication getScreenShareApplication() { + //Get the servlet context + ServletContext ctx = getServletContext(); + //Grab a reference to the application context + ApplicationContext appCtx = (ApplicationContext) ctx.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE); + + //Get the bean holding the parameter + IScreenShareApplication manager = (IScreenShareApplication) appCtx.getBean("screenShareApplication"); + if (manager != null) { + log.debug("Got the IScreenShareApplication context: *****"); + } + return manager; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/JnlpConfigurator.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/JnlpConfigurator.java new file mode 100755 index 0000000000000000000000000000000000000000..755ba59cdc99638fb14fae5bcdf771dc190ccd9f --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/servlet/JnlpConfigurator.java @@ -0,0 +1,48 @@ +package org.bigbluebutton.app.screenshare.server.servlet; + +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.bigbluebutton.app.screenshare.ScreenShareInfo; +import org.bigbluebutton.app.screenshare.ScreenShareInfoResponse; + +public class JnlpConfigurator { + + private String jnlpUrl; + private IScreenShareApplication screenShareApplication; + private String streamBaseUrl; + private String codecOptions; + + + public String getJnlpUrl() { + return jnlpUrl; + } + + public void setJnlpUrl(String url) { + this.jnlpUrl = url; + } + + public void setStreamBaseUrl(String baseUrl) { + streamBaseUrl = baseUrl; + } + + public String getStreamBaseUrl() { + return streamBaseUrl; + } + + public void setCodecOptions(String codeOptions) { + this.codecOptions = codeOptions; + } + + public String getCodecOptions() { + return codecOptions; + } + + public ScreenShareInfo getScreenShareInfo(String meetingId, String token) { + ScreenShareInfoResponse resp = screenShareApplication.getScreenShareInfo(meetingId, token); + if (resp.error != null) return null; + else return resp.info; + } + + public void setApplication(IScreenShareApplication screenShareApplication) { + this.screenShareApplication = screenShareApplication; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/Dimension.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/Dimension.java new file mode 100755 index 0000000000000000000000000000000000000000..38e9236cb2fb74bcc7285969debabc3db360d690 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/Dimension.java @@ -0,0 +1,41 @@ +/** +* +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2010 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 2.1 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +**/ +package org.bigbluebutton.app.screenshare.server.session; + +public final class Dimension { + + private final int width; + private final int height; + + public Dimension(int width, int height) { + this.width = width; + this.height = height; + } + + public int getWidth() { + return width; + } + + public int getHeight() { + return height; + } + + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/ISessionManagerGateway.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/ISessionManagerGateway.java new file mode 100755 index 0000000000000000000000000000000000000000..e33b21fe9cf442bfa5577c4e5d980b93bdd47d6b --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/session/ISessionManagerGateway.java @@ -0,0 +1,35 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.session; + + +/** + * Interface between Java -> Scala + * @author Richard Alam + * + */ +public interface ISessionManagerGateway { + public void createSession(String streamId); + + public void removeSession(String streamId, int seqNum); + + public void updateSession(String streamId, int seqNum); + + public boolean isSharingStopped(String meetingId); +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamEventMessageHandler.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamEventMessageHandler.java new file mode 100755 index 0000000000000000000000000000000000000000..464c94c63f1e4e73c49050d1439cb89109f7d7c7 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamEventMessageHandler.java @@ -0,0 +1,118 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.socket; + +import org.apache.mina.core.future.CloseFuture; +import org.bigbluebutton.app.screenshare.IScreenShareApplication; +import org.bigbluebutton.app.screenshare.server.messages.CaptureEndMessage; +import org.bigbluebutton.app.screenshare.server.messages.CaptureStartMessage; +import org.bigbluebutton.app.screenshare.server.messages.CaptureUpdateMessage; +import org.bigbluebutton.app.screenshare.server.messages.MouseLocationMessage; +import org.apache.mina.core.service.IoHandlerAdapter; +import org.apache.mina.core.session.IdleStatus; +import org.apache.mina.core.session.IoSession; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + +public class BlockStreamEventMessageHandler extends IoHandlerAdapter { + final private Logger log = Red5LoggerFactory.getLogger(BlockStreamEventMessageHandler.class, "screenshare"); + + private IScreenShareApplication app; + private static final String ROOM = "ROOM"; + + @Override + public void exceptionCaught( IoSession session, Throwable cause ) throws Exception { + log.warn(cause.toString() + " \n " + cause.getMessage()); + cause.printStackTrace(); + closeSession(session); + } + + private void closeSession(IoSession session) { + String room = (String)session.getAttribute(ROOM, null); + if (room != null) { + log.info("Closing session [" + room + "]. "); + } else { + log.info("Cannot determine session to close."); + } + CloseFuture future = session.close(true); + } + + @Override + public void messageReceived( IoSession session, Object message ) throws Exception + { + if (message instanceof CaptureStartMessage) { + System.out.println("Got CaptureStartBlockEvent"); + CaptureStartMessage event = (CaptureStartMessage) message; +// sessionManager.createSession(event.getRoom()); + } else if (message instanceof CaptureUpdateMessage) { +// System.out.println("Got CaptureUpdateBlockEvent"); + CaptureUpdateMessage event = (CaptureUpdateMessage) message; +// sessionManager.updateBlock(event.getRoom(), event.getSequenceNum()); + if (app.isSharingStopped(event.getRoom(), event.getRoom())) { + // The flash client told us to stop sharing. Force stopping by closing connection from applet. + // We're changing how to tell the applet to stop sharing as AS ExternalInterface to JS to Applet calls + // generates a popup dialog that users may or may not see causing the browser to hang. (ralam aug 24, 2014) + log.info("Sharing has stopped for meeting [" + event.getRoom() + "]. Closing connection."); + session.close(true); + } + } else if (message instanceof CaptureEndMessage) { + CaptureEndMessage event = (CaptureEndMessage) message; +// sessionManager.removeSession(event.getRoom(), event.getSequenceNum()); + } else if (message instanceof MouseLocationMessage) { + MouseLocationMessage event = (MouseLocationMessage) message; +// sessionManager.updateMouseLocation(event.getRoom(), event.getLoc(), event.getSequenceNum()); + } + } + + @Override + public void sessionIdle( IoSession session, IdleStatus status ) throws Exception + { + log.debug( "IDLE " + session.getIdleCount( status )); + super.sessionIdle(session, status); + } + + @Override + public void sessionCreated(IoSession session) throws Exception { + log.debug("Session Created"); + super.sessionCreated(session); + } + + @Override + public void sessionOpened(IoSession session) throws Exception { + log.debug("Session Opened."); + super.sessionOpened(session); + } + + @Override + public void sessionClosed(IoSession session) throws Exception { + log.debug("Session Closed."); + + String room = (String) session.getAttribute("ROOM"); + if (room != null) { + log.debug("Session Closed for room " + room); + app.sharingStopped(room, room); + } else { + log.warn("Closing session for a NULL room"); + } + } + + public void setApplication(IScreenShareApplication app) { + this.app = app; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamProtocolDecoder.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamProtocolDecoder.java new file mode 100755 index 0000000000000000000000000000000000000000..9b3e0d2bcd5e9ace1fe119cefda55148b46cafeb --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/BlockStreamProtocolDecoder.java @@ -0,0 +1,283 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.socket; + +import java.awt.Point; +import java.nio.charset.CharacterCodingException; +import java.nio.charset.Charset; +import java.util.Arrays; + +import org.apache.mina.core.buffer.IoBuffer; +import org.apache.mina.core.session.IoSession; +import org.apache.mina.filter.codec.CumulativeProtocolDecoder; +import org.apache.mina.filter.codec.ProtocolDecoderOutput; +import org.bigbluebutton.app.screenshare.server.messages.CaptureEndMessage; +import org.bigbluebutton.app.screenshare.server.messages.CaptureStartMessage; +import org.bigbluebutton.app.screenshare.server.messages.CaptureUpdateMessage; +import org.bigbluebutton.app.screenshare.server.messages.MouseLocationMessage; +import org.bigbluebutton.app.screenshare.server.session.Dimension; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + +public class BlockStreamProtocolDecoder extends CumulativeProtocolDecoder { + final private Logger log = Red5LoggerFactory.getLogger(BlockStreamProtocolDecoder.class, "screenshare"); + + private static final String ROOM = "ROOM"; + private static final byte[] POLICY_REQUEST = new byte[] {'<','p','o','l','i','c','y','-','f','i','l','e','-','r','e','q','u','e','s','t','/','>',0}; + private static final byte[] END_FRAME = new byte[] {'S', 'S', '-', 'E', 'N', 'D'}; + private static final byte[] HEADER = new byte[] {'B', 'B', 'B', '-', 'S', 'S'}; + private static final byte CAPTURE_START_EVENT = 0; + private static final byte CAPTURE_UPDATE_EVENT = 1; + private static final byte CAPTURE_END_EVENT = 2; + private static final byte MOUSE_LOCATION_EVENT = 3; + + protected boolean doDecode(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + // Remember the initial position. + int start = in.position(); + byte[] endFrame = new byte[END_FRAME.length]; + + // Now find the END FRAME delimeter in the buffer. + int curpos = 0; + while (in.remaining() >= END_FRAME.length) { + curpos = in.position(); + in.get(endFrame); + + if (Arrays.equals(endFrame, END_FRAME)) { + //log.debug("***** END FRAME {} = {}", endFrame, END_FRAME); + // Remember the current position and limit. + int position = in.position(); + int limit = in.limit(); + try { + in.position(start); + in.limit(position); + // The bytes between in.position() and in.limit() + // now contain a full frame. + parseFrame(session, in.slice(), out); + } finally { + // Set the position to point right after the + // detected END FRAME and set the limit to the old + // one. + in.position(position); + in.limit(limit); + } + return true; + } + + in.position(curpos+1); + } + + // Try to find a policy request, used by the BigBlueButton Client Checker + in.position(start); + if (tryToParsePolicyRequest(session, in, out)) { + return true; + } + + // Could not find END FRAME in the buffer. Reset the initial + // position to the one we recorded above. + in.position(start); + + return false; + } + + private boolean tryToParsePolicyRequest(IoSession session, IoBuffer in, ProtocolDecoderOutput out) { + byte[] message = new byte[POLICY_REQUEST.length]; + if (in.remaining() >= POLICY_REQUEST.length) { + in.get(message, 0, message.length); + if (Arrays.equals(message, POLICY_REQUEST)) { + log.debug("Sending cross domain policy to the user"); + IoBuffer buffer = IoBuffer.allocate(8); + buffer.setAutoExpand(true); + try { + buffer.putString("<cross-domain-policy><allow-access-from domain=\"*\" to-ports=\"*\" /></cross-domain-policy>", Charset.forName("UTF-8").newEncoder()); + buffer.put((byte) 0); + } catch (CharacterCodingException e) { + e.printStackTrace(); + return false; + } + buffer.flip(); + session.write(buffer); + return true; + } + } + + return false; + } + + private void parseFrame(IoSession session, IoBuffer in, ProtocolDecoderOutput out) { + //log.debug("Frame = {}", in.toString()); + try { + byte[] header = new byte[HEADER.length]; + + in.get(header, 0, HEADER.length); + + if (! Arrays.equals(header, HEADER)) { + log.info("Invalid header. Discarding. {}", header); + return; + } + + int messageLength = in.getInt(); + + if (in.remaining() < messageLength) { + log.info("Invalid length. Discarding. [{} < {}]", in.remaining(), messageLength); + return; + } + + decodeMessage(session, in, out); + + return; + } catch (Exception e) { + log.warn("Failed to parse frame. Discarding."); + } + } + + private void decodeMessage(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + byte event = in.get(); + switch (event) { + case CAPTURE_START_EVENT: + log.info("Decoding CAPTURE_START_EVENT"); + decodeCaptureStartEvent(session, in, out); + break; + case CAPTURE_UPDATE_EVENT: + //log.info("Decoding CAPTURE_UPDATE_EVENT"); + decodeCaptureUpdateEvent(session, in, out); + break; + case CAPTURE_END_EVENT: + log.info("Got CAPTURE_END_EVENT event: " + event); + decodeCaptureEndEvent(session, in, out); + break; + case MOUSE_LOCATION_EVENT: + decodeMouseLocationEvent(session, in, out); + break; + default: + log.error("Unknown event: " + event); + throw new Exception("Unknown event: " + event); + } + } + + private void decodeMouseLocationEvent(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + String room = decodeRoom(session, in); + if ("".equals(room)) { + log.warn("Empty meeting name in decoding mouse location."); + throw new Exception("Empty meeting name in decoding mouse location."); + } + + int seqNum = in.getInt(); + int mouseX = in.getInt(); + int mouseY = in.getInt(); + + /** Swallow end frame **/ + in.get(new byte[END_FRAME.length]); + + MouseLocationMessage event = new MouseLocationMessage(room, new Point(mouseX, mouseY), seqNum); + out.write(event); + } + + private void decodeCaptureEndEvent(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + String room = decodeRoom(session, in); + if ("".equals(room)) { + log.warn("Empty meeting name in decoding capture end event."); + throw new Exception("Empty meeting name in decoding capture end event."); + } + + log.info("CaptureEndEvent for " + room); + int seqNum = in.getInt(); + + /** Swallow end frame **/ + in.get(new byte[END_FRAME.length]); + + CaptureEndMessage event = new CaptureEndMessage(room, seqNum); + out.write(event); + } + + private void decodeCaptureStartEvent(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + String room = decodeRoom(session, in); + if ("".equals(room)) { + log.warn("Empty meeting name in decoding capture start event."); + throw new Exception("Empty meeting name in decoding capture start event."); + } + + session.setAttribute(ROOM, room); + int seqNum = in.getInt(); + + Dimension blockDim = decodeDimension(in); + Dimension screenDim = decodeDimension(in); + + boolean useSVC2 = (in.get() == 1); + + /** Swallow end frame **/ + in.get(new byte[END_FRAME.length]); + + log.info("CaptureStartEvent for " + room); + CaptureStartMessage event = new CaptureStartMessage(room, screenDim, blockDim, seqNum, useSVC2); + out.write(event); + } + + private Dimension decodeDimension(IoBuffer in) { + int width = in.getInt(); + int height = in.getInt(); + return new Dimension(width, height); + } + + private String decodeRoom(IoSession session, IoBuffer in) { + int roomLength = in.get(); +// System.out.println("Room length = " + roomLength); + String room = ""; + try { + room = in.getString(roomLength, Charset.forName( "UTF-8" ).newDecoder()); + if (session.containsAttribute(ROOM)) { + String attRoom = (String) session.getAttribute(ROOM); + if (!attRoom.equals(room)) { + log.warn(room + " is not the same as room in attribute [" + attRoom + "]"); + } + } + } catch (CharacterCodingException e) { + log.error(e.getMessage()); + } + + return room; + } + + private void decodeCaptureUpdateEvent(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { + String room = decodeRoom(session, in); + if ("".equals(room)) { + log.warn("Empty meeting name in decoding capture start event."); + throw new Exception("Empty meeting name in decoding capture start event."); + } + + int seqNum = in.getInt(); + int numBlocks = in.getShort(); + + String blocksStr = "Blocks changed "; + + for (int i = 0; i < numBlocks; i++) { + int position = in.getShort(); + blocksStr += " " + position; + + boolean isKeyFrame = (in.get() == 1) ? true : false; + int length = in.getInt(); + byte[] data = new byte[length]; + in.get(data, 0, length); + CaptureUpdateMessage event = new CaptureUpdateMessage(room, position, data, isKeyFrame, seqNum); + out.write(event); + } + + /** Swallow end frame **/ + in.get(new byte[END_FRAME.length]); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/DeskShareServer.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/DeskShareServer.java new file mode 100755 index 0000000000000000000000000000000000000000..ead02c9075fe1f7cecbbdd9c896fc94005c47ce5 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/DeskShareServer.java @@ -0,0 +1,67 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.socket; + +import java.io.IOException; +import java.net.InetSocketAddress; + +import org.apache.mina.core.service.IoHandlerAdapter; +import org.apache.mina.core.session.IdleStatus; +import org.apache.mina.filter.codec.ProtocolCodecFilter; +import org.apache.mina.transport.socket.nio.NioSocketAcceptor; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; + + +public class DeskShareServer { + final private Logger log = Red5LoggerFactory.getLogger(DeskShareServer.class, "deskshare"); + + private int port = 1270; + + private IoHandlerAdapter screenCaptureHandler; + private NioSocketAcceptor acceptor; + + public void start() + { + acceptor = new NioSocketAcceptor(); + acceptor.getFilterChain().addLast( "codec", new ProtocolCodecFilter(new ScreenCaptureProtocolCodecFactory())); + + acceptor.setHandler( screenCaptureHandler); + acceptor.getSessionConfig().setIdleTime( IdleStatus.BOTH_IDLE, 10 ); + acceptor.setReuseAddress(true); + try { + acceptor.bind( new InetSocketAddress(port) ); + } catch (IOException e) { + log.error("IOException while binding to port {}", port); + } + } + + public void setScreenCaptureHandler(IoHandlerAdapter screenCaptureHandler) { + this.screenCaptureHandler = screenCaptureHandler; + } + + public void stop() { + acceptor.unbind(); + acceptor.dispose(); + } + + public void setPort(int port) { + this.port = port; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/NullProtocolEncoder.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/NullProtocolEncoder.java new file mode 100755 index 0000000000000000000000000000000000000000..0d5c81271745407c9818cc72c3be0dc8c0f4c1bc --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/NullProtocolEncoder.java @@ -0,0 +1,38 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.socket; + +import org.apache.mina.core.session.IoSession; +import org.apache.mina.filter.codec.ProtocolEncoder; +import org.apache.mina.filter.codec.ProtocolEncoderOutput; + +public class NullProtocolEncoder implements ProtocolEncoder { + + public void dispose(IoSession in) throws Exception { + // TODO Auto-generated method stub + + } + + public void encode(IoSession session, Object message, ProtocolEncoderOutput out) + throws Exception { + // TODO Auto-generated method stub + + } + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/ScreenCaptureProtocolCodecFactory.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/ScreenCaptureProtocolCodecFactory.java new file mode 100755 index 0000000000000000000000000000000000000000..a8badc8b686ecda75acc605c1bb1cd060473b527 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/socket/ScreenCaptureProtocolCodecFactory.java @@ -0,0 +1,42 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.socket; + +import org.apache.mina.core.session.IoSession; +import org.apache.mina.filter.codec.ProtocolCodecFactory; +import org.apache.mina.filter.codec.ProtocolDecoder; +import org.apache.mina.filter.codec.ProtocolEncoder; + +public class ScreenCaptureProtocolCodecFactory implements ProtocolCodecFactory { + private ProtocolEncoder encoder; + private ProtocolDecoder decoder; + + public ScreenCaptureProtocolCodecFactory() { + encoder = new NullProtocolEncoder(); + decoder = new BlockStreamProtocolDecoder(); + } + + public ProtocolEncoder getEncoder(IoSession ioSession) throws Exception { + return encoder; + } + + public ProtocolDecoder getDecoder(IoSession ioSession) throws Exception { + return decoder; + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/util/StackTraceUtil.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/util/StackTraceUtil.java new file mode 100755 index 0000000000000000000000000000000000000000..deccbd60730c669ae4e4729fd8973356765318a0 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/server/util/StackTraceUtil.java @@ -0,0 +1,32 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.util; + +import java.io.PrintWriter; +import java.io.StringWriter; +import java.io.Writer; + +public final class StackTraceUtil { + public static String getStackTrace(Throwable aThrowable) { + final Writer result = new StringWriter(); + final PrintWriter printWriter = new PrintWriter(result); + aThrowable.printStackTrace(printWriter); + return result.toString(); + } +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/IDataStore.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/IDataStore.java new file mode 100755 index 0000000000000000000000000000000000000000..0d2bd61f9f5008e91e36eb453a83c52829e6a557 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/IDataStore.java @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.store; + +public interface IDataStore { + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/IScreenShareData.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/IScreenShareData.java new file mode 100755 index 0000000000000000000000000000000000000000..da147d0a76400a50777f255d9c5eff15eaf2a987 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/IScreenShareData.java @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.store.redis; + +public interface IScreenShareData { + +} diff --git a/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/RedisDataStore.java b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/RedisDataStore.java new file mode 100755 index 0000000000000000000000000000000000000000..e1900231febfe63631debd14a2c6a61702f4e220 --- /dev/null +++ b/bbb-screenshare/app/src/main/java/org/bigbluebutton/app/screenshare/store/redis/RedisDataStore.java @@ -0,0 +1,79 @@ +package org.bigbluebutton.app.screenshare.store.redis; + +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; +import org.red5.logging.Red5LoggerFactory; +import org.slf4j.Logger; +import redis.clients.jedis.Jedis; +import redis.clients.jedis.JedisPool; + +public class RedisDataStore { + private static Logger log = Red5LoggerFactory.getLogger(RedisDataStore.class, "screenshare"); + + private JedisPool redisPool; + private volatile boolean sendMessage = false; + private int maxThreshold = 1024; + private final Executor msgSenderExec = Executors.newSingleThreadExecutor(); + private BlockingQueue<IScreenShareData> dataToStore = new LinkedBlockingQueue<IScreenShareData>(); + private final Executor runExec = Executors.newSingleThreadExecutor(); + + public void stop() { + sendMessage = false; + } + + public void start() { + try { + sendMessage = true; + + Runnable messageSender = new Runnable() { + public void run() { + while (sendMessage) { + try { + IScreenShareData data = dataToStore.take(); + storeData(data); + } catch (InterruptedException e) { + log.warn("Failed to get data from queue."); + } + } + } + }; + msgSenderExec.execute(messageSender); + } catch (Exception e) { + log.error("Error storing data into redis: " + e.getMessage()); + } + } + + public void store(IScreenShareData data) { + if (dataToStore.size() > maxThreshold) { + log.warn("Queued number of data [{}] is greater than threshold [{}]", dataToStore.size(), maxThreshold); + } + dataToStore.add(data); + } + + private void storeData(IScreenShareData data) { + Runnable task = new Runnable() { + public void run() { + Jedis jedis = redisPool.getResource(); + try { + // jedis.publish(channel, message); + } catch(Exception e){ + log.warn("Cannot publish the message to redis", e); + } finally { + redisPool.returnResource(jedis); + } + } + }; + + runExec.execute(task); + } + + public void setRedisPool(JedisPool redisPool){ + this.redisPool = redisPool; + } + + public void setMaxThreshold(int threshold) { + maxThreshold = threshold; + } +} diff --git a/bbb-screenshare/app/src/main/resources/logback-screenshare.xml b/bbb-screenshare/app/src/main/resources/logback-screenshare.xml new file mode 100755 index 0000000000000000000000000000000000000000..87637bd81f6a557d47751483b7ce829a9cb7465e --- /dev/null +++ b/bbb-screenshare/app/src/main/resources/logback-screenshare.xml @@ -0,0 +1,25 @@ +<?xml version="1.0" encoding="UTF-8"?> +<configuration> + <appender name="screenshare" class="ch.qos.logback.core.rolling.RollingFileAppender"> + <File>log/screenshare-slf.log</File> + + <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> + <FileNamePattern>log/screenshare-slf.%d{yyyy-MM-dd}.log</FileNamePattern> + <!-- keep 30 days worth of history --> + <MaxHistory>30</MaxHistory> + </rollingPolicy> + + <encoder> + <charset>UTF-8</charset> + <pattern>%d{ISO8601} [%thread] %-5level %logger{35} - %msg%n</pattern> + </encoder> + </appender> + + <root> + <level value="DEBUG" /> + <appender-ref ref="screenshare" /> + </root> + + <!-- LEVEL CAN NOT BE DEBUG --> + <logger name="org.apache" level="INFO"></logger> +</configuration> diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/ScreenShareApplication.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/ScreenShareApplication.scala new file mode 100755 index 0000000000000000000000000000000000000000..a53495e3f71499c2c7b90c6376cad02881bfefb1 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/ScreenShareApplication.scala @@ -0,0 +1,183 @@ +package org.bigbluebutton.app.screenshare + +import org.bigbluebutton.app.screenshare.events.IEventsMessageBus +import org.bigbluebutton.app.screenshare.server.sessions.ScreenshareSessionManager +import org.bigbluebutton.app.screenshare.server.sessions.messages._ +import org.bigbluebutton.app.screenshare.server.util.LogHelper + +class ScreenShareApplication(val bus: IEventsMessageBus, val jnlpFile: String, + val streamBaseUrl: String) + extends IScreenShareApplication with LogHelper { + + val sessionManager: ScreenshareSessionManager = new ScreenshareSessionManager(bus) + sessionManager.start + + val initError: Error = new Error("Uninitialized error.") + + def userDisconnected(meetingId: String, userId: String) { + if (logger.isDebugEnabled()) { + logger.debug("Received user disconnected on meeting=" + meetingId + + "] userid=[" + userId + "]") + } + + sessionManager ! new UserDisconnected(meetingId, userId) + } + + + def isScreenSharing(meetingId: String):IsScreenSharingResponse = { + if (logger.isDebugEnabled()) { + logger.debug("Received is screen sharing on meeting=" + meetingId + + "]") + } + + var response: IsScreenSharingResponse = new IsScreenSharingResponse(null, initError) + sessionManager !? (3000, IsScreenSharing(meetingId)) match { + case None => { + logger.info("Failed to get response to is screen sharing request on meeting=" + meetingId + "]") + val info = new StreamInfo(false, "none", 0, 0, "none") + response = new IsScreenSharingResponse(info, new Error("Timedout waiting for response")) + } + case Some(rep) => { + val reply = rep.asInstanceOf[IsScreenSharingReply] + val info = new StreamInfo(true, reply.streamId, reply.width, reply.height, reply.url) + response = new IsScreenSharingResponse(info, null) + } + } + + response + } + + def getScreenShareInfo(meetingId: String, token: String):ScreenShareInfoResponse = { + if (logger.isDebugEnabled()) { + logger.debug("Received get screen sharing info on token=" + token + + "]") + } + + var response: ScreenShareInfoResponse = new ScreenShareInfoResponse(null, initError) + sessionManager !? (3000, ScreenShareInfoRequest(meetingId, token)) match { + case None => { + logger.info("Failed to get response to get screen sharing info request on token=" + token + "]") + response = new ScreenShareInfoResponse(null, new Error("Timedout waiting for response.")) + } + case Some(rep) => { + val reply = rep.asInstanceOf[ScreenShareInfoRequestReply] + val publishUrl = streamBaseUrl + "/" + meetingId + "/" + reply.streamId + val info = new ScreenShareInfo(publishUrl, reply.streamId) + response = new ScreenShareInfoResponse(info, null) + } + } + + response + } + + def recordStream(meetingId: String, streamId: String):java.lang.Boolean = { + if (logger.isDebugEnabled()) { + logger.debug("Received record stream request on stream=" + streamId + "]") + } + + var record = false + + sessionManager !? (3000, IsStreamRecorded(meetingId, streamId)) match { + case None => { + logger.info("Failed to get response to record stream request on streamId=" + + streamId + "]") + record = false + } + case Some(rep) => { + val reply = rep.asInstanceOf[IsStreamRecordedReply] + record = reply.record + } + } + + record + } + + def startShareRequest(meetingId: String, userId: String, record: java.lang.Boolean): StartShareRequestResponse = { + if (logger.isDebugEnabled()) { + logger.debug("Received start share request on meeting=" + meetingId + + "for user=" + userId + "]") + } + + var response: StartShareRequestResponse = new StartShareRequestResponse(null, null, initError) + + sessionManager !? (3000, StartShareRequestMessage(meetingId, userId, record)) match { + case None => { + logger.info("Failed to get response to start share request on meeting=" + + meetingId + " for user=" + userId + "]") + response = new StartShareRequestResponse(null, null, new Error("Timedout waiting for response")) + } + case Some(rep) => { + val reply = rep.asInstanceOf[StartShareRequestReplyMessage] + response = new StartShareRequestResponse(reply.token, jnlpFile, null) + } + } + + response + } + + def stopShareRequest(meetingId: String, streamId: String) { + if (logger.isDebugEnabled()) { + logger.debug("Received stop share request on meeting=[" + meetingId + + "] for stream=[" + streamId + "]") + } + sessionManager ! new StopShareRequestMessage(meetingId, streamId) + } + + def streamStarted(meetingId: String, streamId: String, url: String) { + if (logger.isDebugEnabled()) { + logger.debug("Received stream started on meeting=[" + meetingId + + "] for stream=[" + streamId + "]") + } + sessionManager ! new StreamStartedMessage(meetingId, streamId, url) + } + + def streamStopped(meetingId: String, streamId: String) { + if (logger.isDebugEnabled()) { + logger.debug("Received stream stopped on meeting=[" + meetingId + + "] for stream=[" + streamId + "]") + } + sessionManager ! new StreamStoppedMessage(meetingId, streamId) + } + + def sharingStarted(meetingId: String, streamId: String, width: java.lang.Integer, height: java.lang.Integer) { + if (logger.isDebugEnabled()) { + logger.debug("Received share started on meeting=[" + meetingId + + "] for stream=[" + streamId + "] with region=[" + width + "x" + height + "]") + } + sessionManager ! new SharingStartedMessage(meetingId, streamId, width, height) + } + + def sharingStopped(meetingId: String, streamId: String) { + if (logger.isDebugEnabled()) { + logger.debug("Received sharing stopped on meeting=" + meetingId + + "for stream=" + streamId + "]") + } + sessionManager ! new SharingStoppedMessage(meetingId, streamId) + } + + def updateShareStatus(meetingId: String, streamId : String, seqNum: java.lang.Integer) { + if (logger.isDebugEnabled()) { + logger.debug("Received sharing status on meeting=" + meetingId + + "for stream=" + streamId + "]") + } + sessionManager ! new UpdateShareStatus(meetingId, streamId, seqNum) + } + + def isSharingStopped(meetingId: String, streamId: String): java.lang.Boolean = { + if (logger.isDebugEnabled()) { + logger.debug("Received sharing status on meeting=" + meetingId + + "for stream=" + streamId + "]") + } + + var stopped = false + sessionManager !? (3000, IsSharingStopped(meetingId, streamId)) match { + case None => stopped = true + case Some(rep) => { + val reply = rep.asInstanceOf[IsSharingStoppedReply] + stopped = reply.stopped + } + } + + stopped + } +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/MeetingActor.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/MeetingActor.scala new file mode 100755 index 0000000000000000000000000000000000000000..bedcb3e5f4ff6a73f85d3a27afc0cb9dee7f45b8 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/MeetingActor.scala @@ -0,0 +1,252 @@ +package org.bigbluebutton.app.screenshare.server.sessions + +import scala.actors.Actor +import scala.actors.Actor._ +import scala.collection.mutable.HashMap +import org.bigbluebutton.app.screenshare.events.IEventsMessageBus +import org.bigbluebutton.app.screenshare.server.util._ +import org.bigbluebutton.app.screenshare.server.sessions.messages._ + + +class MeetingActor(val sessionManager: ScreenshareSessionManager, + val bus: IEventsMessageBus, + val meetingId: String) extends Actor with LogHelper { + + private val sessions = new HashMap[String, ScreenshareSession] + + private var lastHasSessionCheck:Long = TimeUtil.getCurrentMonoTime + + private var activeSession:Option[ScreenshareSession] = None + private var stopped = false + + private val IS_MEETING_RUNNING = "IsMeetingRunning" + + def scheduleIsMeetingRunningCheck() { + val mainActor = self + actor { + Thread.sleep(60000) + mainActor ! IS_MEETING_RUNNING + } + } + + def act() = { + loop { + react { + case msg: StartShareRequestMessage => handleStartShareRequestMessage(msg) + case msg: StopShareRequestMessage => handleStopShareRequestMessage(msg) + case msg: StreamStartedMessage => handleStreamStartedMessage(msg) + case msg: StreamStoppedMessage => handleStreamStoppedMessage(msg) + case msg: SharingStartedMessage => handleSharingStartedMessage(msg) + case msg: SharingStoppedMessage => handleSharingStoppedMessage(msg) + case msg: IsSharingStopped => handleIsSharingStopped(msg) + case msg: IsScreenSharing => handleIsScreenSharing(msg) + case msg: IsStreamRecorded => handleIsStreamRecorded(msg) + case msg: UpdateShareStatus => handleUpdateShareStatus(msg) + case msg: UserDisconnected => handleUserDisconnected(msg) + case msg: ScreenShareInfoRequest => handleScreenShareInfoRequest(msg) + case IS_MEETING_RUNNING => handleIsMeetingRunning() + case msg: KeepAliveTimeout => handleKeepAliveTimeout(msg) + case m: Any => logger.warn("Session: Unknown message [{}]", m) + } + } + } + + private def findSessionByUser(userId: String):Option[ScreenshareSession] = { + sessions.values find (su => su.userId == userId) + } + + private def findSessionWithToken(token: String):Option[ScreenshareSession] = { + sessions.values find (su => su.token == token) + } + + private def handleUserDisconnected(msg: UserDisconnected) { + if (logger.isDebugEnabled()) { + logger.debug("Received UserDisconnected for meetingId=[" + msg.meetingId + "]") + } + + findSessionByUser(msg.userId) foreach (s => s forward msg) + } + + private def handleIsScreenSharing(msg: IsScreenSharing) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsScreenSharing for meetingId=[" + msg.meetingId + "]") + } + + activeSession foreach (s => s forward msg) + } + + private def handleScreenShareInfoRequest(msg: ScreenShareInfoRequest) { + if (logger.isDebugEnabled()) { + logger.debug("Received ScreenShareInfoRequest for token=[" + msg.token + "]") + } + + findSessionWithToken(msg.token) foreach (s => s forward msg) + } + + private def handleIsStreamRecorded(msg: IsStreamRecorded) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsStreamRecorded for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + } + case None => { + logger.info("IsStreamRecorded on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleUpdateShareStatus(msg: UpdateShareStatus) { + if (logger.isDebugEnabled()) { + logger.debug("Received UpdateShareStatus for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + } + case None => { + logger.info("Sharing stopped on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleSharingStoppedMessage(msg: SharingStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received SharingStoppedMessage for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + + } + case None => { + logger.info("Sharing stopped on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleSharingStartedMessage(msg: SharingStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received SharingStartedMessage for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + } + case None => { + logger.info("Sharing started on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleStreamStoppedMessage(msg: StreamStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StreamStoppedMessage for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + activeSession = None + } + case None => { + logger.info("Stream stopped on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleStreamStartedMessage(msg: StreamStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StreamStartedMessage for streamId=[" + msg.streamId + "]") + } + + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + activeSession = Some(session) + } + case None => { + logger.info("Stream started on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleStopShareRequestMessage(msg: StopShareRequestMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StopShareRequestMessage for streamId=[" + msg.streamId + "]") + } + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + } + case None => { + logger.info("Stop share request on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleStartShareRequestMessage(msg: StartShareRequestMessage) { + val token = RandomStringGenerator.randomAlphanumericString(16) + val streamId = msg.meetingId + "-" + System.currentTimeMillis(); + + val session: ScreenshareSession = new ScreenshareSession(this, bus, + meetingId, streamId, token, + msg.record, msg.userId) + sessions += streamId -> session + session.start + + session forward msg + + } + + private def handleIsSharingStopped(msg: IsSharingStopped) { + sessions.get(msg.streamId) match { + case Some(session) => { + session forward msg + } + case None => { + logger.info("Stream stopped on a non-existing session=[" + msg.streamId + "]") + } + } + } + + private def handleStopSession() { + stopped = true + } + + private def handleStartSession() { + stopped = false + scheduleIsMeetingRunningCheck + } + + private def handleIsMeetingRunning() { + // If not sessions in the last 5 minutes, then assume meeting has ended. + if (sessions.isEmpty) { + if (TimeUtil.getCurrentMonoTime - lastHasSessionCheck > 300000) { + sessionManager ! MeetingHasEnded(meetingId) + } else { + scheduleIsMeetingRunningCheck + } + } else { + lastHasSessionCheck = TimeUtil.getCurrentMonoTime + scheduleIsMeetingRunningCheck + } + } + + private def handleKeepAliveTimeout(msg: KeepAliveTimeout) { + sessions.remove(msg.streamId) foreach { s => + if (activeSession != None) { + activeSession foreach { as => + if (as.streamId == s.streamId) activeSession = None + } + } + } + } + + +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSession.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSession.scala new file mode 100755 index 0000000000000000000000000000000000000000..78c68e915ee8706c936b5976d7ee0e65233fbd3a --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSession.scala @@ -0,0 +1,205 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.app.screenshare.server.sessions + + +import scala.actors.Actor +import scala.actors.Actor._ +import net.lag.logging.Logger +import org.bigbluebutton.app.screenshare.server.util.LogHelper +import org.bigbluebutton.app.screenshare.server.util.TimeUtil +import org.bigbluebutton.app.screenshare.server.sessions.messages._ +import org.bigbluebutton.app.screenshare.events.IEventsMessageBus +import org.bigbluebutton.app.screenshare.events.ShareStartedEvent +import org.bigbluebutton.app.screenshare.events.ShareStoppedEvent +import org.bigbluebutton.app.screenshare.events.StreamStoppedEvent +import org.bigbluebutton.app.screenshare.events.StreamStartedEvent + + +case object StartSession +case object StopSession +case class KeepAliveTimeout(streamId: String) + +class ScreenshareSession(parent: MeetingActor, + bus: IEventsMessageBus, + val meetingId: String, + val streamId: String, + val token: String, + val recorded: Boolean, + val userId: String) extends Actor with LogHelper { + + private var timeOfLastKeepAliveUpdate:Long = TimeUtil.getCurrentMonoTime + private val KEEP_ALIVE_TIMEOUT = 60000 + + // if ffmpeg is still broadcasting + private var streamStopped = true + // if jws is still running + private var shareStopped = true + + // if the user has requested to stop sharing + private var stopShareRequested = false + + private var width: Int = 0 + private var height: Int = 0 + + private var streamUrl: String = "" + + private var timestamp = 0L; + private var lastUpdate:Long = System.currentTimeMillis(); + + private val IS_STREAM_ALIVE = "IsStreamAlive" + + def scheduleKeepAliveCheck() { + val mainActor = self + actor { + Thread.sleep(5000) + mainActor ! IS_STREAM_ALIVE + } + } + + def act() = { + loop { + react { + case msg: StartShareRequestMessage => handleStartShareRequestMessage(msg) + case msg: StopShareRequestMessage => handleStopShareRequestMessage(msg) + case msg: StreamStartedMessage => handleStreamStartedMessage(msg) + case msg: StreamStoppedMessage => handleStreamStoppedMessage(msg) + case msg: SharingStartedMessage => handleSharingStartedMessage(msg) + case msg: SharingStoppedMessage => handleSharingStoppedMessage(msg) + case msg: IsSharingStopped => handleIsSharingStopped(msg) + case msg: IsScreenSharing => handleIsScreenSharing(msg) + case msg: IsStreamRecorded => handleIsStreamRecorded(msg) + case msg: UpdateShareStatus => handleUpdateShareStatus(msg) + case msg: UserDisconnected => handleUserDisconnected(msg) + case msg: ScreenShareInfoRequest => handleScreenShareInfoRequest(msg) + case IS_STREAM_ALIVE => checkIfStreamIsAlive() + case m: Any => logger.warn("Session: Unknown message [%s]", m) + } + } + } + + private def handleUserDisconnected(msg: UserDisconnected) { + if (logger.isDebugEnabled()) { + logger.debug("Received UserDisconnected for streamId=[" + streamId + "]") + } + + stopShareRequested = true + } + + private def handleIsStreamRecorded(msg: IsStreamRecorded) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsStreamRecorded for streamId=[" + msg.streamId + "]") + } + + reply(new IsStreamRecordedReply(recorded)) + } + + private def handleIsScreenSharing(msg: IsScreenSharing) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsScreenSharing for meetingId=[" + msg.meetingId + "]") + } + + reply(new IsScreenSharingReply(true, streamId, width, height, streamUrl)) + } + + private def handleScreenShareInfoRequest(msg: ScreenShareInfoRequest) { + if (logger.isDebugEnabled()) { + logger.debug("Received ScreenShareInfoRequest for token=" + msg.token + " streamId=[" + streamId + "]") + } + + reply(new ScreenShareInfoRequestReply(msg.meetingId, streamId)) + } + + private def handleSharingStoppedMessage(msg: SharingStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received SharingStoppedMessage for streamId=[" + msg.streamId + "]") + } + + shareStopped = true + width = 0 + height = 0 + bus.send(new ShareStoppedEvent(meetingId, streamId)) + } + + private def handleSharingStartedMessage(msg: SharingStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received SharingStartedMessagefor streamId=[" + msg.streamId + "]") + } + + stopShareRequested = false + shareStopped = false + width = msg.width + height = msg.height + bus.send(new ShareStartedEvent(meetingId, streamId)) + } + + private def handleStreamStoppedMessage(msg: StreamStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StreamStoppedMessage streamId=[" + msg.streamId + "]") + } + + streamStopped = true + bus.send(new StreamStoppedEvent(meetingId, streamId)) + } + + private def handleStreamStartedMessage(msg: StreamStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StreamStartedMessage for streamId=[" + msg.streamId + "]") + } + + streamStopped = false + streamUrl = msg.url + bus.send(new StreamStartedEvent(meetingId, streamId, width, height, msg.url)) + } + + private def handleStopShareRequestMessage(msg: StopShareRequestMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StopShareRequestMessage for streamId=[" + msg.streamId + "]") + } + + stopShareRequested = true + bus.send(new ShareStoppedEvent(meetingId, streamId)) + } + + private def handleStartShareRequestMessage(msg: StartShareRequestMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received StartShareRequestMessage for streamId=[" + msg.meetingId + "]") + } + + scheduleKeepAliveCheck() + reply(new StartShareRequestReplyMessage(token)) + } + + private def handleIsSharingStopped(msg: IsSharingStopped) { + reply(new IsSharingStoppedReply(stopShareRequested)) + } + + private def handleUpdateShareStatus(msg: UpdateShareStatus): Unit = { + timeOfLastKeepAliveUpdate = TimeUtil.getCurrentMonoTime + } + + private def checkIfStreamIsAlive() { + if (TimeUtil.getCurrentMonoTime - timeOfLastKeepAliveUpdate > KEEP_ALIVE_TIMEOUT) { + logger.warn("Did not received updates for more than 1 minute. Removing stream {}", streamId) + parent ! new KeepAliveTimeout(streamId) + } else { + scheduleKeepAliveCheck() + } + } +} diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSessionManager.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSessionManager.scala new file mode 100755 index 0000000000000000000000000000000000000000..aabca4089f58bbd7ecbc116fff7397bbe3f34ce9 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/ScreenshareSessionManager.scala @@ -0,0 +1,207 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.app.screenshare.server.sessions + +import scala.actors.Actor +import scala.actors.Actor._ +import net.lag.logging.Logger +import scala.collection.mutable.HashMap +import org.bigbluebutton.app.screenshare.events.IEventsMessageBus +import org.bigbluebutton.app.screenshare.server.sessions.messages._ +import org.bigbluebutton.app.screenshare.server.util.LogHelper + + +case class HasScreenShareSession(meetingId: String) +case class HasScreenShareSessionReply(meetingId: String, sharing: Boolean, streamId:Option[String]) +case class MeetingHasEnded(meetingId: String) + +class ScreenshareSessionManager(val bus: IEventsMessageBus) + extends Actor with LogHelper { + + private val meetings = new HashMap[String, MeetingActor] + + def act() = { + loop { + react { + case msg: StartShareRequestMessage => handleStartShareRequestMessage(msg) + case msg: StopShareRequestMessage => handleStopShareRequestMessage(msg) + case msg: StreamStartedMessage => handleStreamStartedMessage(msg) + case msg: StreamStoppedMessage => handleStreamStoppedMessage(msg) + case msg: SharingStartedMessage => handleSharingStartedMessage(msg) + case msg: SharingStoppedMessage => handleSharingStoppedMessage(msg) + case msg: IsStreamRecorded => handleIsStreamRecorded(msg) + case msg: IsSharingStopped => handleIsSharingStopped(msg) + case msg: IsScreenSharing => handleIsScreenSharing(msg) + case msg: ScreenShareInfoRequest => handleScreenShareInfoRequest(msg) + case msg: UpdateShareStatus => handleUpdateShareStatus(msg) + case msg: UserDisconnected => handleUserDisconnected(msg) + case msg: MeetingHasEnded => handleMeetingHasEnded(msg) + + case msg: Any => logger.warn("Unknown message " + msg) + } + } + } + + + private def handleUserDisconnected(msg: UserDisconnected) { + if (logger.isDebugEnabled()) { + logger.debug("Received UserDisconnected message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleIsStreamRecorded(msg: IsStreamRecorded) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsStreamRecorded message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleIsScreenSharing(msg: IsScreenSharing) { + if (logger.isDebugEnabled()) { + logger.debug("Received IsScreenSharing message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleMeetingHasEnded(msg: MeetingHasEnded) { + logger.info("Removing meeting [" + msg.meetingId + "]") + meetings -= msg.meetingId + } + + private def handleScreenShareInfoRequest(msg: ScreenShareInfoRequest) { + if (logger.isDebugEnabled()) { + logger.debug("Received ScreenShareInfoRequest message for meetingId=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleUpdateShareStatus(msg: UpdateShareStatus) { + if (logger.isDebugEnabled()) { + logger.debug("Received update share message for meeting=[" + msg.streamId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleSharingStoppedMessage(msg: SharingStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received sharing stopped message for meeting=[" + msg.streamId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleSharingStartedMessage(msg: SharingStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received sharing started message for meeting=[" + msg.streamId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + + private def handleIsSharingStopped(msg: IsSharingStopped) { + meetings.get(msg.meetingId) foreach { s => s forward msg } + } + + private def handleStreamStoppedMessage(msg: StreamStoppedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received stream stopped message for meeting=[" + msg.streamId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleStreamStartedMessage(msg: StreamStartedMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received stream started message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleStopShareRequestMessage(msg: StopShareRequestMessage) { + if (logger.isDebugEnabled()) { + logger.debug("Received stop share request message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) foreach { meeting => + meeting forward msg + } + } + + private def handleStartShareRequestMessage(msg: StartShareRequestMessage): Unit = { + if (logger.isDebugEnabled()) { + logger.debug("Received start share request message for meeting=[" + msg.meetingId + "]") + } + + meetings.get(msg.meetingId) match { + case None => { + if (logger.isDebugEnabled()) { + logger.debug("Creating meeting=[" + msg.meetingId + "]") + } + + val meeting: MeetingActor = new MeetingActor(this, bus, msg.meetingId) + meetings += msg.meetingId -> meeting + meeting.start + meeting forward msg + } + case Some(meeting) => { + if (logger.isDebugEnabled()) { + logger.debug("Meeting already exists. meeting=[" + msg.meetingId + "]") + } + meeting forward msg + } + } + } + + private def removeSession(meetingId: String): Unit = { + logger.debug("SessionManager: Removing session " + meetingId); + meetings.get(meetingId) foreach { s => + s ! StopSession + val old:Int = meetings.size + meetings -= meetingId; + logger.debug("RemoveSession: Session length [%d,%d]", old, meetings.size) + } + } + +} diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/IMessage.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/IMessage.scala new file mode 100755 index 0000000000000000000000000000000000000000..0f90d01fb8c991cb80942c65add11f240b9e4cab --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/IMessage.scala @@ -0,0 +1,36 @@ +package org.bigbluebutton.app.screenshare.server.sessions.messages + +case class StartShareRequestMessage(meetingId: String, userId: String, record: Boolean) + +case class StartShareRequestReplyMessage(token: String) + +case class StopShareRequestMessage(meetingId: String, streamId: String) + +case class StreamStartedMessage(meetingId: String, streamId: String, url: String) + +case class StreamStoppedMessage(meetingId: String, streamId: String) + +case class SharingStartedMessage(meetingId: String, streamId: String, width: Int, height: Int) + +case class SharingStoppedMessage(meetingId: String, streamId: String) + +case class IsStreamRecorded(meetingId: String, streamId: String) + +case class IsStreamRecordedReply(record: Boolean) + +case class IsSharingStopped(meetingId: String, streamId: String) + +case class IsSharingStoppedReply(stopped: Boolean) + +case class UpdateShareStatus(meetingId: String, streamId: String, sequence: Int) + +case class IsScreenSharing(meetingId: String) + +case class IsScreenSharingReply(sharing: Boolean, streamId: String, + width: Int, height: Int, url: String) + +case class ScreenShareInfoRequest(meetingId: String, token: String) + +case class ScreenShareInfoRequestReply(meetingId: String, streamId: String) + +case class UserDisconnected(meetingId: String, userId: String) \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/ShareScreenResponse.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/ShareScreenResponse.scala new file mode 100755 index 0000000000000000000000000000000000000000..f81ae8589562a3d9b2cb43a90c2e5252c00c2245 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/sessions/messages/ShareScreenResponse.scala @@ -0,0 +1,5 @@ +package org.bigbluebutton.app.screenshare.server.sessions.messages + +class ShareScreenResponse { + +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/LogHelper.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/LogHelper.scala new file mode 100755 index 0000000000000000000000000000000000000000..41a602aeee11ed8cb1d7dac675b5cfaef5352b4e --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/LogHelper.scala @@ -0,0 +1,13 @@ +package org.bigbluebutton.app.screenshare.server.util + +import org.slf4j.Logger +import org.red5.logging.Red5LoggerFactory; + +/** + * LogHelper is a trait you can mix in to provide easy log4j logging + * for your scala classes. + **/ +trait LogHelper { + val loggerName = this.getClass.getName + lazy val logger = Red5LoggerFactory.getLogger(this.getClass, "screenshare") +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/RandomStringGenerator.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/RandomStringGenerator.scala new file mode 100755 index 0000000000000000000000000000000000000000..cef85f6a9722136249342f14b6ad8e5fef50fad9 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/RandomStringGenerator.scala @@ -0,0 +1,17 @@ +package org.bigbluebutton.app.screenshare.server.util + +object RandomStringGenerator { +// From: http://www.bindschaedler.com/2012/04/07/elegant-random-string-generation-in-scala/ + + // Random generator + val random = new scala.util.Random + + // Generate a random string of length n from the given alphabet + def randomString(alphabet: String)(n: Int): String = + Stream.continually(random.nextInt(alphabet.size)).map(alphabet).take(n).mkString + + // Generate a random alphabnumeric string of length n + def randomAlphanumericString(n: Int) = + randomString("abcdefghijklmnopqrstuvwxyz0123456789")(n) +} + diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/TimeUtil.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/TimeUtil.scala new file mode 100755 index 0000000000000000000000000000000000000000..14d08ff51a44cb4f69d4fee2caa9714086821233 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/server/util/TimeUtil.scala @@ -0,0 +1,18 @@ +package org.bigbluebutton.app.screenshare.server.util + +import java.util.concurrent.TimeUnit + +object TimeUtil { + + def generateTimestamp():Long = { + TimeUnit.NANOSECONDS.toMillis(System.nanoTime()) + } + + def getCurrentMonoTime():Long = { + TimeUnit.NANOSECONDS.toMillis(System.nanoTime()) + } + + def getCurrentTime():Long = { + System.currentTimeMillis(); + } +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/store/redis/EventListenerImp.scala b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/store/redis/EventListenerImp.scala new file mode 100755 index 0000000000000000000000000000000000000000..3517942bbe23a365586f41c86ab230b3ecf67293 --- /dev/null +++ b/bbb-screenshare/app/src/main/scala/org/bigbluebutton/app/screenshare/store/redis/EventListenerImp.scala @@ -0,0 +1,11 @@ +package org.bigbluebutton.app.screenshare.store.redis + +import org.bigbluebutton.app.screenshare.events.IEventListener +import org.bigbluebutton.app.screenshare.events.IEvent + +class EventListenerImp extends IEventListener { + + def handleMessage(event: IEvent) { + + } +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.properties b/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.properties new file mode 100755 index 0000000000000000000000000000000000000000..71fa8ffbaae52a11f520db7bd40fac02cad3656d --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.properties @@ -0,0 +1,2 @@ +webapp.contextPath=/screenshare +webapp.virtualHosts=*, localhost, localhost:8088, 127.0.0.1:8088 diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.xml b/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.xml new file mode 100755 index 0000000000000000000000000000000000000000..e5774867d0fd41262cfcdb0aa952ec7c7320c1ae --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/red5-web.xml @@ -0,0 +1,110 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!-- + +BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + +Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + +This program is free software; you can redistribute it and/or modify it under the +terms of the GNU Lesser General Public License as published by the Free Software +Foundation; either version 3.0 of the License, or (at your option) any later +version. + +BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public License along +with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + +--> +<beans xmlns="http://www.springframework.org/schema/beans" + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns:lang="http://www.springframework.org/schema/lang" + xsi:schemaLocation="http://www.springframework.org/schema/beans + http://www.springframework.org/schema/beans/spring-beans-2.0.xsd + http://www.springframework.org/schema/lang + http://www.springframework.org/schema/lang/spring-lang-2.0.xsd"> + + <bean id="placeholderConfig" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> + <property name="locations"> + <list> + <value>/WEB-INF/red5-web.properties</value> + <value>/WEB-INF/screenshare.properties</value> + </list> + </property> + </bean> + + <bean id="web.context" class="org.red5.server.Context" autowire="byType"/> + + <bean id="web.scope" class="org.red5.server.scope.WebScope" init-method="register"> + <property name="server" ref="red5.server"/> + <property name="parent" ref="global.scope"/> + <property name="context" ref="web.context"/> + <property name="handler" ref="web.handler"/> + <property name="contextPath" value="${webapp.contextPath}"/> + <property name="virtualHosts" value="${webapp.virtualHosts}"/> + </bean> + + <bean id="web.handler" class="org.bigbluebutton.app.screenshare.red5.Red5AppAdapter"> + <property name="streamBaseUrl" value="${streamBaseUrl}"/> + <property name="eventRecordingService" ref="eventRecordingService"/> + <property name="recordingDirectory" value="${recordingDirectory}"/> + <property name="application" ref="screenShareApplication"/> + <property name="messageSender" ref="connectionInvokerService"/> + </bean> + + <bean id="screenshare.service" class="org.bigbluebutton.app.screenshare.red5.Red5AppService"> + <property name="appHandler" ref="red5AppHandler"/> + </bean> + + <bean id="red5AppHandler" class="org.bigbluebutton.app.screenshare.red5.Red5AppHandler"> + <property name="application" ref="screenShareApplication"/> + <property name="messageSender" ref="connectionInvokerService"/> + </bean> + + <!-- The IoHandler implementation --> + <bean id="screenCaptureHandler" class="org.bigbluebutton.app.screenshare.server.socket.BlockStreamEventMessageHandler"> + <property name="application" ref="screenShareApplication"/> + </bean> + + <bean id="screenShareApplication" class="org.bigbluebutton.app.screenshare.ScreenShareApplication"> + <constructor-arg index="0" ref="messageBus"/> + <constructor-arg index="1" value="${jnlpFile}"/> + <constructor-arg index="2" value="${streamBaseUrl}"/> + </bean> + + <bean id="eventListenerImp" class="org.bigbluebutton.app.screenshare.red5.EventListenerImp"> + <property name="messageSender" ref="connectionInvokerService"/> + </bean> + + <bean id="jnlpConfigurator" class="org.bigbluebutton.app.screenshare.server.servlet.JnlpConfigurator"> + <property name="jnlpUrl" value="${jnlpUrl}"/> + <property name="streamBaseUrl" value="${streamBaseUrl}"/> + <property name="codecOptions" value="${codecOptions}"/> + <property name="application" ref="screenShareApplication"/> + </bean> + + <bean id="messageBus" class="org.bigbluebutton.app.screenshare.events.EventMessageBusImp" + init-method="start" destroy-method="stop"> + <property name="listeners"> + <set> + <ref bean="eventListenerImp" /> + </set> + </property> + </bean> + + <bean id="connectionInvokerService" class="org.bigbluebutton.app.screenshare.red5.ConnectionInvokerService" + init-method="start" destroy-method="stop"> + </bean> + + <bean id="eventRecordingService" class="org.bigbluebutton.app.screenshare.EventRecordingService"> + <constructor-arg index="0" value="${redis.host}"/> + <constructor-arg index="1" value="${redis.port}"/> + </bean> + + <bean id="redisRecorder" class="org.bigbluebutton.app.screenshare.server.recorder.EventRecorder"> + <constructor-arg index="0" value="${redis.host}"/> + <constructor-arg index="1" value="${redis.port}"/> + </bean> +</beans> diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.conf b/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.conf new file mode 100755 index 0000000000000000000000000000000000000000..f0f0e6c273c36fa8deb90739040bae5bcdc7d8d1 --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.conf @@ -0,0 +1,21 @@ +log { + filename = "/usr/share/red5/log/screenshare.log" + roll = "daily" + level = "info" + use_full_package_names = on + + silence_net_sf_cache { + node = "net.sf.ehcache" + level = "error" + } + + silence_sun_rmi { + node = "sun.rmi" + level = "error" + } + + silence_org_apache { + node = "org.apache" + level = "error" + } +} \ No newline at end of file diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.properties b/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.properties new file mode 100755 index 0000000000000000000000000000000000000000..a61a86e3feeb19bb885431218a7b838d53b700ae --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/screenshare.properties @@ -0,0 +1,29 @@ +# +# NOTE: default properties. +# +# NOTE!!!! NOTE!!!! NOTE!!!! NOTE!!!! NOTE!!!! +# When making changes that you don't want checked-in, do +# git update-index --assume-unchanged <file> +# +# To have git track the changes again +# git update-index --no-assume-unchanged <file> +# + +recordingDirectory=/usr/share/red5/webapps/screenshare/streams + +redis.host=127.0.0.1 +redis.port=6379 + + +streamBaseUrl=rtmp://192.168.23.36/screenshare +jnlpUrl=http://192.168.23.36/screenshare +jnlpFile=http://192.168.23.36/screenshare/screenshare.jnlp + +# NOTES: +# 1. GOP (group of pictures) is calculated as frameRate * keyFrameInterval +# 2. intra-refresh=1 doesn't work in Chrome. Late comers can't view the stream as +# the user missed the key frame +# 3. keyFrameInterval is in seconds +# 4. Make sure you encode & into & as it will break the JNLP XML +#codecOptions=crf=36&preset=veryfast&tune=animation,zerolatency&frameRate=12.0&keyFrameInterval=6 +codecOptions=crf=38&preset=veryfast&tune=zerolatency&frameRate=12.0&keyFrameInterval=6&intra-refresh=1 diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/tunnel-servlet.xml b/bbb-screenshare/app/src/main/webapp/WEB-INF/tunnel-servlet.xml new file mode 100755 index 0000000000000000000000000000000000000000..0760dc00afb2c0458ab7bd3fc405c9ecb4ae7d18 --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/tunnel-servlet.xml @@ -0,0 +1,44 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!-- + +BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + +Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + +This program is free software; you can redistribute it and/or modify it under the +terms of the GNU Lesser General Public License as published by the Free Software +Foundation; either version 3.0 of the License, or (at your option) any later +version. + +BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public License along +with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + +--> +<beans xmlns="http://www.springframework.org/schema/beans" + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns:lang="http://www.springframework.org/schema/lang" + xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd + http://www.springframework.org/schema/lang http://www.springframework.org/schema/lang/spring-lang-2.0.xsd"> + + <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"/> + + <bean id="handlerMapping" class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"> + <property name="defaultHandler" ref="httpTunnelController"/> + </bean> + + <bean id="httpTunnelController" class="org.bigbluebutton.app.screenshare.server.servlet.HttpTunnelStreamController"> + <property name="methodNameResolver"> + <bean class="org.springframework.web.servlet.mvc.multiaction.PropertiesMethodNameResolver"> + <property name="mappings"> + <props> + <prop key="/screenCapture">screenCaptureHandler</prop> + </props> + </property> + </bean> + </property> + </bean> +</beans> diff --git a/bbb-screenshare/app/src/main/webapp/WEB-INF/web.xml b/bbb-screenshare/app/src/main/webapp/WEB-INF/web.xml new file mode 100755 index 0000000000000000000000000000000000000000..35d88442837f2c1cecd72b6419095b3addd97e68 --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/WEB-INF/web.xml @@ -0,0 +1,87 @@ +<?xml version="1.0" encoding="ISO-8859-1"?> +<!-- + +BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + +Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + +This program is free software; you can redistribute it and/or modify it under the +terms of the GNU Lesser General Public License as published by the Free Software +Foundation; either version 3.0 of the License, or (at your option) any later +version. + +BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public License along +with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + +--> +<web-app + xmlns="http://java.sun.com/xml/ns/j2ee" + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd" + version="2.4"> + + <display-name>deskShare</display-name> + + <context-param> + <param-name>webAppRootKey</param-name> + <param-value>/screenshare</param-value> + </context-param> + + <listener> + <listener-class>org.red5.logging.ContextLoggingListener</listener-class> + </listener> + + <filter> + <filter-name>LoggerContextFilter</filter-name> + <filter-class>org.red5.logging.LoggerContextFilter</filter-class> + </filter> + + <filter-mapping> + <filter-name>LoggerContextFilter</filter-name> + <url-pattern>/*</url-pattern> + </filter-mapping> + + <!--listener> + <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> + </listener--> + + <servlet> + <servlet-name>tunnel</servlet-name> + <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> + <load-on-startup>1</load-on-startup> + </servlet> + + <!-- maps the sample dispatcher to *.form --> + <servlet-mapping> + <servlet-name>tunnel</servlet-name> + <url-pattern>/tunnel/*</url-pattern> + </servlet-mapping> + + <servlet> + <servlet-name>JnlpDownloadServlet</servlet-name> + <servlet-class>jnlp.sample.servlet.JnlpDownloadServlet</servlet-class> + </servlet> + + <servlet-mapping> + <servlet-name>JnlpDownloadServlet</servlet-name> + <url-pattern>*.jnlp</url-pattern> + </servlet-mapping> + + <servlet-mapping> + <servlet-name>JnlpDownloadServlet</servlet-name> + <url-pattern>*.jar</url-pattern> + </servlet-mapping> + + <security-constraint> + <web-resource-collection> + <web-resource-name>Forbidden</web-resource-name> + <url-pattern>/streams/*</url-pattern> + </web-resource-collection> + <auth-constraint/> + </security-constraint> + +</web-app> diff --git a/bbb-screenshare/app/src/main/webapp/streams/README b/bbb-screenshare/app/src/main/webapp/streams/README new file mode 100755 index 0000000000000000000000000000000000000000..7e41f04ba5c7ceb44ecd1f94474580e6f2c98e8a --- /dev/null +++ b/bbb-screenshare/app/src/main/webapp/streams/README @@ -0,0 +1 @@ +Do not delete the streams directory. This is where the screenshare recordings will be saved. \ No newline at end of file diff --git a/bbb-screenshare/app/src/test/java/README b/bbb-screenshare/app/src/test/java/README new file mode 100755 index 0000000000000000000000000000000000000000..28b05c6394926aa5dd12ccf8e7100907300e0261 --- /dev/null +++ b/bbb-screenshare/app/src/test/java/README @@ -0,0 +1 @@ + placeholder for tests diff --git a/bbb-screenshare/app/src/test/java/org/bigbluebutton/deskshare/server/recorder/FileRecorderTest.java b/bbb-screenshare/app/src/test/java/org/bigbluebutton/deskshare/server/recorder/FileRecorderTest.java new file mode 100755 index 0000000000000000000000000000000000000000..f5491111f761d49201584918ff900cdd51a8c3d3 --- /dev/null +++ b/bbb-screenshare/app/src/test/java/org/bigbluebutton/deskshare/server/recorder/FileRecorderTest.java @@ -0,0 +1,12 @@ +package org.bigbluebutton.deskshare.server.recorder; + +import org.testng.Assert; +import org.testng.annotations.Test; + +public class FileRecorderTest { + + @Test + public void testHello() { + Assert.assertEquals(true, true); + } +} diff --git a/bbb-screenshare/app/src/test/resources/testng.xml b/bbb-screenshare/app/src/test/resources/testng.xml new file mode 100755 index 0000000000000000000000000000000000000000..41fcfdd9e180c627aa928cb4502451d08050c6a8 --- /dev/null +++ b/bbb-screenshare/app/src/test/resources/testng.xml @@ -0,0 +1,14 @@ +<!DOCTYPE suite SYSTEM "http://beust.com/testng/testng-1.0.dtd" > +<suite name="BigBlueButton Desktop Sharing Test Suite"> + <test name="Conference tests"> + <!--groups> + <run> + <exclude name="broken"/> + </run> + </groups--> + + <packages> + <package name="org.bigbluebutton.deskshare.server.recorder"/> + </packages> + </test> +</suite> \ No newline at end of file diff --git a/bbb-screenshare/jws/README.md b/bbb-screenshare/jws/README.md new file mode 100755 index 0000000000000000000000000000000000000000..298cd0f33315127b396a1b0d79ed35383fcfa62e --- /dev/null +++ b/bbb-screenshare/jws/README.md @@ -0,0 +1 @@ +# bigbluebutton-screenshare diff --git a/bbb-screenshare/jws/native-libs/README.md b/bbb-screenshare/jws/native-libs/README.md new file mode 100755 index 0000000000000000000000000000000000000000..1023ffc06c3e26150d0e9825ad4ae3d7114fffad --- /dev/null +++ b/bbb-screenshare/jws/native-libs/README.md @@ -0,0 +1,15 @@ + +This directory contains the difference JavaCV jar files that we need for our web start application. + +To build each native library and sign each ffmpeg native libraries: + +1. cd to the platform directory you want to build (cd ffmpeg-win-x86) +2. type ```gradle jar``` to build the jar file +3. type ```ant sign-jar``` to sign the jar file with your certificate +4. copy the signed-jar to ```bbb-screenshare/app/jws/lib``` to be included when + deploying to red5 + +We have included unsigned jars for ```ffmpeg.jar```, ```javacpp.jar```, and ```javacv.jar``` in the +unsigned-jars directory. You can sign the jar files there with your certificate. + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/.gitignore b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/.gitignore new file mode 100755 index 0000000000000000000000000000000000000000..24e6582bbcf2f11d21b3c52ed6bd4a5e29a8c303 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/.gitignore @@ -0,0 +1,5 @@ +build/ +src/ +workdir/ +*.p12 + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.gradle b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..7e20f10a8c068f27843e42380ed1327c8b6b64c0 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.gradle @@ -0,0 +1,10 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'ffmpeg-linux-x86' + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.xml b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..3aa62722a7f196f483d288dffb98f092c21d036d --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/build.xml @@ -0,0 +1,38 @@ +<?xml version="1.0" ?> +<project name="ffmpeg-linux-x86-signing" basedir="."> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/ffmpeg-linux-x86-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + + +</project> diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/ffmpeg-linux-x86.jar b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/ffmpeg-linux-x86.jar new file mode 100755 index 0000000000000000000000000000000000000000..668fc4bcd41f11495b3bc659e009939d9629c681 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/ffmpeg-linux-x86.jar differ diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/sign-jar.sh b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/sign-jar.sh new file mode 100755 index 0000000000000000000000000000000000000000..291781c99a16d0dbd2672ba84e9476dd91194800 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86/sign-jar.sh @@ -0,0 +1,12 @@ +mkdir workdir +cp ffmpeg-linux-x86.jar workdir +rm -rf src +mkdir -p src/main/resources +cd workdir +jar xvf ffmpeg-linux-x86.jar +cp org/bytedeco/javacpp/linux-x86/* ../src/main/resources +cd .. +rm -rf workdir +gradle jar +ant sign-jar +cp build/libs/ffmpeg-linux-x86-0.0.1.jar ../../../app/jws/lib/ffmpeg-linux-x86.jar diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/.gitignore b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/.gitignore new file mode 100755 index 0000000000000000000000000000000000000000..24e6582bbcf2f11d21b3c52ed6bd4a5e29a8c303 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/.gitignore @@ -0,0 +1,5 @@ +build/ +src/ +workdir/ +*.p12 + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.gradle b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..6cc065a7dfd20e1ce798aba3ce3c45b4a40e49a1 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.gradle @@ -0,0 +1,10 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'ffmpeg-linux-x86_64' + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.xml b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..ac9a22aaef132842657d420270db75b4a96c9c5a --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/build.xml @@ -0,0 +1,38 @@ +<?xml version="1.0" ?> +<project name="ffmpeg-linux-x86_64-signing" basedir="."> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/ffmpeg-linux-x86_64-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + + +</project> diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/ffmpeg-linux-x86_64.jar b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/ffmpeg-linux-x86_64.jar new file mode 100755 index 0000000000000000000000000000000000000000..6241ebbcfb2c1f8cd55dc3c06a1ebaa7472a33f4 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/ffmpeg-linux-x86_64.jar differ diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/sign-jar.sh b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/sign-jar.sh new file mode 100755 index 0000000000000000000000000000000000000000..cf31f4f5151eeb1f22199e5ba4236e145948823c --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-linux-x86_64/sign-jar.sh @@ -0,0 +1,12 @@ +mkdir workdir +cp ffmpeg-linux-x86_64.jar workdir +rm -rf src +mkdir -p src/main/resources +cd workdir +jar xvf ffmpeg-linux-x86_64.jar +cp org/bytedeco/javacpp/linux-x86_64/* ../src/main/resources +cd .. +rm -rf workdir +gradle jar +ant sign-jar +cp build/libs/ffmpeg-linux-x86_64-0.0.1.jar ../../../app/jws/lib/ffmpeg-linux-x86_64.jar diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/.gitignore b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/.gitignore new file mode 100755 index 0000000000000000000000000000000000000000..24e6582bbcf2f11d21b3c52ed6bd4a5e29a8c303 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/.gitignore @@ -0,0 +1,5 @@ +build/ +src/ +workdir/ +*.p12 + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.gradle b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..2243f4c691a847d3ddc6c50f9f7397e79243601e --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.gradle @@ -0,0 +1,10 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'ffmpeg-macosx-x86_64' + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.xml b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..3b06a54dce2c3ea75bb4a8d8ed2b4f3ce863f6af --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/build.xml @@ -0,0 +1,38 @@ +<?xml version="1.0" ?> +<project name="ffmpeg-macosx-x86_64-signing" basedir="."> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/ffmpeg-macosx-x86_64-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + + +</project> diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/ffmpeg-macosx-x86_64.jar b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/ffmpeg-macosx-x86_64.jar new file mode 100755 index 0000000000000000000000000000000000000000..334a1cac0ec23678ec55b37ee92c9e05064aaf4a Binary files /dev/null and b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/ffmpeg-macosx-x86_64.jar differ diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/sign-jar.sh b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/sign-jar.sh new file mode 100755 index 0000000000000000000000000000000000000000..70d3c5613ce92ff932453812b8b76c06df8d7178 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-macosx-x86_64/sign-jar.sh @@ -0,0 +1,12 @@ +mkdir workdir +cp ffmpeg-macosx-x86_64.jar workdir +rm -rf src +mkdir -p src/main/resources +cd workdir +jar xvf ffmpeg-macosx-x86_64.jar +cp org/bytedeco/javacpp/macosx-x86_64/* ../src/main/resources +cd .. +rm -rf workdir +gradle jar +ant sign-jar +cp build/libs/ffmpeg-macosx-x86_64-0.0.1.jar ../../../app/jws/lib/ffmpeg-macosx-x86_64.jar diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/.gitignore b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/.gitignore new file mode 100755 index 0000000000000000000000000000000000000000..24e6582bbcf2f11d21b3c52ed6bd4a5e29a8c303 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/.gitignore @@ -0,0 +1,5 @@ +build/ +src/ +workdir/ +*.p12 + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.gradle b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..3ab1f6fe39730f2fe49d536366cdf641e709af55 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.gradle @@ -0,0 +1,10 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'ffmpeg-windows-x86' + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.xml b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..54ca846d98aca44b3e728625d20f0c04d3466aeb --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/build.xml @@ -0,0 +1,38 @@ +<?xml version="1.0" ?> +<project name="ffmpeg-win-x86-signing" basedir="."> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/ffmpeg-windows-x86-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + + +</project> diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/ffmpeg-windows-x86.jar b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/ffmpeg-windows-x86.jar new file mode 100755 index 0000000000000000000000000000000000000000..ca60f25da40209bed487ea8231ae49099ad1d89a Binary files /dev/null and b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/ffmpeg-windows-x86.jar differ diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/sign-jar.sh b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/sign-jar.sh new file mode 100755 index 0000000000000000000000000000000000000000..6cb8f4c9d4ae092d65a5b37cfc3eddf6ff086cc5 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86/sign-jar.sh @@ -0,0 +1,15 @@ +mkdir workdir +cp ffmpeg-windows-x86.jar workdir/ffmpeg-windows-x86.jar +rm -rf src +mkdir -p src/main/resources +mkdir -p src/main/java +cd workdir +jar xvf ffmpeg-windows-x86.jar +cp org/bytedeco/javacpp/windows-x86/*.dll ../src/main/resources +cd .. +rm -rf workdir +gradle jar +ant sign-jar +cp build/libs/ffmpeg-windows-x86-0.0.1.jar ../../../app/jws/lib/ffmpeg-windows-x86.jar +rm -rf src + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/.gitignore b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/.gitignore new file mode 100755 index 0000000000000000000000000000000000000000..24e6582bbcf2f11d21b3c52ed6bd4a5e29a8c303 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/.gitignore @@ -0,0 +1,5 @@ +build/ +src/ +workdir/ +*.p12 + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.gradle b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..1d9791a055d075a6ecdd036a34ebec6119ec718f --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.gradle @@ -0,0 +1,10 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'ffmpeg-windows-x86_64' + + diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.xml b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..769a2130bd2528ef4742781fbe4fcc837d4c7e46 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/build.xml @@ -0,0 +1,38 @@ +<?xml version="1.0" ?> +<project name="ffmpeg-win-x86-signing" basedir="."> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/ffmpeg-windows-x86_64-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + + +</project> diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/ffmpeg-windows-x86_64.jar b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/ffmpeg-windows-x86_64.jar new file mode 100755 index 0000000000000000000000000000000000000000..413567d01e0f02cdd1c959b0cc522cd31a91894d Binary files /dev/null and b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/ffmpeg-windows-x86_64.jar differ diff --git a/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/sign-jar.sh b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/sign-jar.sh new file mode 100755 index 0000000000000000000000000000000000000000..ed16eb567dc37395fbfc2b0eff738dd89d162af2 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/ffmpeg-windows-x86_64/sign-jar.sh @@ -0,0 +1,15 @@ +mkdir workdir +cp ffmpeg-windows-x86_64.jar workdir/ffmpeg-windows-x86_64.jar +rm -rf src +mkdir -p src/main/resources +mkdir -p src/main/java +cd workdir +jar xvf ffmpeg-windows-x86_64.jar +cp org/bytedeco/javacpp/windows-x86_64/*.dll ../src/main/resources +cd .. +rm -rf workdir +gradle jar +ant sign-jar +cp build/libs/ffmpeg-windows-x86_64-0.0.1.jar ../../../app/jws/lib/ffmpeg-windows-x86_64.jar +rm -rf src + diff --git a/bbb-screenshare/jws/native-libs/signed-jars/ffmpeg.jar b/bbb-screenshare/jws/native-libs/signed-jars/ffmpeg.jar new file mode 100755 index 0000000000000000000000000000000000000000..578ebf0614960448557810f671844689995f19c3 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/signed-jars/ffmpeg.jar differ diff --git a/bbb-screenshare/jws/native-libs/signed-jars/javacpp.jar b/bbb-screenshare/jws/native-libs/signed-jars/javacpp.jar new file mode 100755 index 0000000000000000000000000000000000000000..6ec49265efbf0584fa86edc67d3258348302a064 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/signed-jars/javacpp.jar differ diff --git a/bbb-screenshare/jws/native-libs/signed-jars/javacv.jar b/bbb-screenshare/jws/native-libs/signed-jars/javacv.jar new file mode 100755 index 0000000000000000000000000000000000000000..8a560ec97649c69a7c0364d81fc53063cfa0246a Binary files /dev/null and b/bbb-screenshare/jws/native-libs/signed-jars/javacv.jar differ diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/README.md b/bbb-screenshare/jws/native-libs/unsigned-jars/README.md new file mode 100755 index 0000000000000000000000000000000000000000..562a962044c8d3df8fc9bbd904b4924d1d4b8d0b --- /dev/null +++ b/bbb-screenshare/jws/native-libs/unsigned-jars/README.md @@ -0,0 +1,11 @@ + +To sign the ffmpeg.jar and javacpp.jar, copy them to the ```workdir``` directory and +run "```ant sign-ffmpeg-jar```" or "```ant sign-javacpp-jar```". + +The resulting jar files will now be signed. To verify, run + +``` + jarsigner -verify ffmpeg.jar + jarsigner -verify javacpp.jar +``` + diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/ffmpeg-2.8.1-1.2-SNAPSHOT.jar b/bbb-screenshare/jws/native-libs/unsigned-jars/ffmpeg-2.8.1-1.2-SNAPSHOT.jar new file mode 100755 index 0000000000000000000000000000000000000000..756da46cac0837de999394b94a7f588683c7ab27 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/unsigned-jars/ffmpeg-2.8.1-1.2-SNAPSHOT.jar differ diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/javacpp-1.2-SNAPSHOT.jar b/bbb-screenshare/jws/native-libs/unsigned-jars/javacpp-1.2-SNAPSHOT.jar new file mode 100755 index 0000000000000000000000000000000000000000..1d04b5276017c4daa3fcee22fb34e4610f1e069c Binary files /dev/null and b/bbb-screenshare/jws/native-libs/unsigned-jars/javacpp-1.2-SNAPSHOT.jar differ diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/javacv.jar b/bbb-screenshare/jws/native-libs/unsigned-jars/javacv.jar new file mode 100755 index 0000000000000000000000000000000000000000..1135b0afe7c5e43904b5354629ff6b9cfdad9052 Binary files /dev/null and b/bbb-screenshare/jws/native-libs/unsigned-jars/javacv.jar differ diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/sign-ffmpeg.sh b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-ffmpeg.sh new file mode 100755 index 0000000000000000000000000000000000000000..c87144166d2ff90a85e60a661ce71af52c76d16f --- /dev/null +++ b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-ffmpeg.sh @@ -0,0 +1,4 @@ +cp ffmpeg-2.8.1-1.2-SNAPSHOT.jar workdir/ffmpeg.jar +ant sign-ffmpeg-jar +cp workdir/ffmpeg.jar ../../../app/jws/lib/ + diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacpp.sh b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacpp.sh new file mode 100755 index 0000000000000000000000000000000000000000..36759bbf4f90c31df851680eda64afe7d401ccb0 --- /dev/null +++ b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacpp.sh @@ -0,0 +1,4 @@ +cp javacpp-1.2-SNAPSHOT.jar workdir/javacpp.jar +ant sign-javacpp-jar +cp workdir/javacpp.jar ../../../app/jws/lib/ + diff --git a/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacv.sh b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacv.sh new file mode 100755 index 0000000000000000000000000000000000000000..7ab5fe4f3bb0483a1aaafcaf73bf46279f8c4bfd --- /dev/null +++ b/bbb-screenshare/jws/native-libs/unsigned-jars/sign-javacv.sh @@ -0,0 +1,4 @@ +cp javacv.jar workdir +ant sign-javacv-jar +cp workdir/javacv.jar ../../../app/jws/lib/ + diff --git a/bbb-screenshare/jws/player/README.md b/bbb-screenshare/jws/player/README.md new file mode 100755 index 0000000000000000000000000000000000000000..29aafaafb3c18d838e20b9b3bfb9b0e32b44a569 --- /dev/null +++ b/bbb-screenshare/jws/player/README.md @@ -0,0 +1,28 @@ +# bbb-video-stream-html-client + +Some attempts to show a BigBlueButton video stream in an html with a standalone Flash player. + +Just open `index.html` and try it! + + + +## How to + +When you open `index.html` and you will see 3 inputs you need to fill: +* The name of your BBB server: use the name or IP only, without `http`, for example: `myserver.somewhere.com` or `192.168.0.1` +* The internal meeting ID used by Red5, that willbe similar to `183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719999` +* The internal video stream ID used by Red5, that will be similar to `320x2401-1328884730777` + +The hard part is to find the internal IDs. +You need to either check the logs in your BBB server or include this +information in BBB's API. +In Mconf we included it in the API, see +http://code.google.com/p/mconf/wiki/MconfLiveApiChanges. +The changes are in this branch: https://github.com/mconf/bigbluebutton/tree/audio-video-on-api-2 + +## Important! + +In Flash Player's security model, Flash applications and SWF files on a local computer cannot access the network. +So first you need disable this security option for the files in your local folder. + +To do so, go to the [Flash Security Panel](http://www.macromedia.com/support/documentation/en/flashplayer/help/settings_manager04.html) and add your local folder to the list. diff --git a/bbb-screenshare/jws/player/actions.js b/bbb-screenshare/jws/player/actions.js new file mode 100755 index 0000000000000000000000000000000000000000..84aca308bb7628101f9f3c4cab88f5126748cad6 --- /dev/null +++ b/bbb-screenshare/jws/player/actions.js @@ -0,0 +1,84 @@ +$(document).ready(function() { + + setup_clear = function(wrapper, width, height) { + $(wrapper).empty(); + $(wrapper).css("width", width); + $(wrapper).css("height", height); + } + + setup_jwplayer = function(wrapper, ip, meeting, stream) { + var url = "rtmp://" + ip + "/live/" + meeting; + + // TODO: for now assuming we will never have bigger resolutions... + var width = 1920; // stream.substring(0, 3); + var height = 1080; // stream.substring(4, 7); + + setup_clear(wrapper, width, height); + + var el = $('<div></div>'); + el.attr("id", "mediaplayer"); + $(wrapper).append(el); + + jwplayer("mediaplayer").setup({ + flashplayer: 'jw-player/player.swf', + id: 'playerID', + width: width, + height: height, + file: stream, // ex: '320x2401-1328884730718' + streamer: url, // ex: 'rtmp://192.160.0.100/video/183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719358' + autostart: 'true', + provider: 'rtmp', + duration: '0', + bufferlength: '1', // it's not working + //start: '0', + //live: 'true', + //repeat: 'none', + }); + }; + + setup_flowplayer = function(wrapper, ip, meeting, stream) { + var url = "rtmp://" + ip + "/live/" + meeting; + + // TODO: for now assuming we will never have bigger resolutions... + var width = 1920; //stream.substring(0, 3); + var height = 1080; // stream.substring(4, 7); + + setup_clear(wrapper, width, height); + + var el = $('<a></a>'); + el.addClass("flowplayer") + $(wrapper).append(el); + + $f("a.flowplayer", "flowplayer/flowplayer-3.2.7.swf", { + clip: { + url: stream, // ex: '320x2401-1328884730718' + provider: 'rtmp', + live: true, + bufferLength: 0, + autoPlay: true, + }, + plugins: { + rtmp: { + url: 'flowplayer/flowplayer.rtmp-3.2.3.swf', + netConnectionUrl: url // ex: 'rtmp://192.160.0.100/video/183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719358' + } + } + }); + $("a.flowplayer").click(); + }; + + $("#submit").on("click", function(e) { + e.preventDefault(); + + var ip = $("#server_ip").val(); + var meeting = $("#meeting_id").val(); + var stream = $("#stream_id").val(); + + if ($('input:radio[name=player]:checked').val() == "flowplayer") { + setup_flowplayer("#wrapper", ip, meeting, stream); + } else { + setup_jwplayer("#wrapper", ip, meeting, stream); + } + }); + +}); diff --git a/bbb-screenshare/jws/player/flowplayer/LICENSE.txt b/bbb-screenshare/jws/player/flowplayer/LICENSE.txt new file mode 100755 index 0000000000000000000000000000000000000000..2a00962f22935d2719aea860183914ffb62a6486 --- /dev/null +++ b/bbb-screenshare/jws/player/flowplayer/LICENSE.txt @@ -0,0 +1,721 @@ +The Flowplayer Free version is released under the +GNU GENERAL PUBLIC LICENSE Version 3 (GPL). + +The GPL requires that you not remove the Flowplayer copyright notices +from the user interface. See section 5.d below. + +Commercial licenses are available. The commercial player version +does not require any Flowplayer notices or texts and also provides +some additional features. + +======================================================================== + +ADDITIONAL TERM per GPL Section 7 +If you convey this program (or any modifications of it) and assume +contractual liability for the program to recipients of it, you agree +to indemnify Flowplayer, Ltd. for any liability that those contractual +assumptions impose on Flowplayer, Ltd. + +Except as expressly provided herein, no trademark rights are granted in +any trademarks of Flowplayer, Ltd. Licensees are granted a limited, +non-exclusive right to use the mark Flowplayer and the Flowplayer logos +in connection with unmodified copies of the Program and the copyright +notices required by section 5.d of the GPL license. For the purposes +of this limited trademark license grant, customizing the Flowplayer +by skinning, scripting, or including PlugIns provided by Flowplayer, Ltd. +is not considered modifying the Program. + +Licensees that do modify the Program, taking advantage of the open-source +license, may not use the Flowplayer mark or Flowplayer logos and must +change the fullscreen notice (and the non-fullscreen notice, if that +option is enabled), the copyright notice in the dialog box, and the +notice on the Canvas as follows: + +the full screen (and non-fullscreen equivalent, if activated) notice +should read: "Based on Flowplayer source code"; in the context menu +(right-click menu), the link to "About Flowplayer free version #.#.#" +can remain. The copyright notice can remain, but must be supplemented with +an additional notice, stating that the licensee modified the Flowplayer. +A suitable notice might read "Flowplayer Source code modified by ModOrg 2009"; +for the canvas, the notice should read "Based on Flowplayer source code". +In addition, licensees that modify the Program must give the modified +Program a new name that is not confusingly similar to Flowplayer and +may not distribute it under the name Flowplayer. + +======================================================================== + + + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + <one line to give the program's name and a brief idea of what it does.> + Copyright (C) <year> <name of author> + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see <http://www.gnu.org/licenses/>. + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + <program> Copyright (C) <year> <name of author> + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +<http://www.gnu.org/licenses/>. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +<http://www.gnu.org/philosophy/why-not-lgpl.html>. \ No newline at end of file diff --git a/bbb-screenshare/jws/player/flowplayer/README.txt b/bbb-screenshare/jws/player/flowplayer/README.txt new file mode 100755 index 0000000000000000000000000000000000000000..4609c6d7c70cb7774c443a7e81877f16b20b0297 --- /dev/null +++ b/bbb-screenshare/jws/player/flowplayer/README.txt @@ -0,0 +1,350 @@ +Version history: + +3.2.7 +----- + +- Loads the new controlbar plugin version 3.2.5. No other functional changes. + +3.2.6 +----- +- linkUrl should now work better with popup blockers: http://code.google.com/p/flowplayer-core/issues/detail?id=31 +- new linkWindow value "_popup" opens the linked page in a popup browser window +- added new onClipResized event +- Added new onUnload event, can be only listened in Flash and not triggered to JS +- API: Added new url property to plugin objects +Fixes: +- it was not possible to call play() in an onFinish listener +- fix to preserve the infoObject for custom netStream and netConnection clients in cases where the infoObject is a + primitive object without properties +- does not show the error dialog in the debugger player when showErrors: false +- fixed to correctly handle xx.ca subdomains when validating the license key +- a custom logo is now sized correctly according to the configured size +- does not show the buffer animation any more when the player receives the onBufferEmpty message from the netStream. + The animation was unnecessarily shown in some situations. +- fixed #155. added new urlEncoding property to Clip for url ncoding ut8 urls + +3.2.5 +----- +- added new scaling option 'crop' that resizes to fill all available space, cropping on top/bottom or left/right +- improvements to RSS file parsing +- Now displays a hand cursor when a linkUrl is used in clips + +3.2.4 +----- +- new flowplayer.js version, with Apple iDevice fixes + +3.2.3 +----- +- a new 'type' clip property exposed to JS +- changed the clip type property to better work as a read-write property. Now accepts 'video', 'audio', + 'image' and 'api' as configuration values. +- moved parallel rtmp connection mechanism from the RTMP plugin to Core so other plugins can use it (ie: securestreaming) +Fixes: +- fixed #112, wrong URL computation when using clip with relative URL on a page with a / after a # in its url +- fixed #111, wrong behavior of pre/post roll images with duration 0 +- fixed multiple license keys logic +Fixes: +- correct verification of license keys in *.ca domains +- fix to make playback to always reach end of video +- fixed resuming of live streams + +3.2.2 +----- +Fixes: +- Now recognizes following kind of urls as audio clips: 'mp3:audiostreamname' (ulrs with mp3 prefix and no extension) +- Now ignores the duration from metadata if we already got one. Fix required for pseudostreaming +- Fix to reuse buffered data when replaying a clip + +3.2.1 +--------- +- Support for RTMP redirects (tested with Wowza loadbalancing) +- Fixed video size when no size info available in clip metadata + +Fixes: +- Fix to correctly detect if the player SWF name contains a version number and if it does also use the version number +when it automatically loads the controls plugin. + +3.2.0 +----- +- canvas, controlbar and the content plugin backgound color and border color can be now given with rgb() and rgba() CSS style syntax +- Added onMouseOver() and onMouseOut() listener registration methods to the Flowplayer API +- enhancements to RSS playlist. Converted parsing to E4X, yahoo media and flowplayer namespace support. +- added feature to obtain bitrate and dimension information to a new clip custom property "bitrates" for future support for bitrate choosing. +- added getter for playerSwfName config +- if clip.url has the string "mp3:" in it, the clip.type will report 'audio' +- added setKeyboardShortcutsEnabled(), addKeyListener(), removeKeyListener() to FlowplayerBase +Fixes: +- onSeek() was not fired when seeking while paused and when using RTMP. An extra onStart was fired too. +- fireErrorExternal() was not working properly with an error PlayerEvent +- countPlugins() was throwing an error when a plugin was not found +- external swf files were not scaled properly +- the logo was unnecessary shown when going fullscreen if logo.displayTime was being used +- added a loadPluginWithConfig method to FlowplayerBase, accessible from javascript. Fixed double onload callback call. +- now handles cuepoint parameters injected using the Adobe Media Encoder +- showPlugin was not working when config.play was null +- handles 3-part duration values included in FLV metadata, like "500.123.123" +- player wasn't always reaching end of video +- fixed broken buffering: false +- fixed event dispatching when embedding flowplayer without flowplayer.js (=without playlist config field) +- fixed safari crashes when unloading player +- fixed scrubber behaviour with a playlist containing 2 images (or swf) in a row +- fixed errors in logs when using an RSS playlist +- fixed OverlayPlayButton that was showing even if it shouldn't on some cases +- fixed wrong behavior when onBeforeFinish was returning false within playlists +- /!\ Don't use the fadeIn / fadeOut controlbar's API while using autoHide. +- fixed play state button with images +- fixed splash image flickering + +3.1.5 +----- +Fixes: +- The player went to a locked state when resuming playback after a period that was long enought to send the +netConnection to an invalid state. Now when resuming playback on an invalid connection the clip starts again from +the beginning. This is only when using RTMP connections and does not affect progressive download playback. +- Custom netConnect and netStream events did not pass the info object to JS listeners + +3.1.4 +----- +Fixes: +- player did not initialize if the controlbar plugin was disabled and if the play button overlay was disabled with play: null +- works properly without cachebusting on IE +- RSS playlist parsing now respects the isDefault attribute used in mRSS media group items +- Fixed passing of connection arguments + +3.1.3 +----- +- enhancements to RSS playlist parsing: Now skips all media:content that have unsupported types. Now the type attribute +of the media:content element is mandatory and has to be present in the RSS file +- Possibility to pass a RSS file name with playFeed("playlist.rss") and setPlaylist("playlist.rss") calls. +- changes to the ConnectionProvider and URLResolver APIs +- Now automatically uses a plugin that is called 'rtmp' for all clips that have the rtmp-protocol in their URLs. +- Added possibility to specify all clip properties in an RSS playlist + +Fixes: +- the result of URL resolvers in now cached, and the resolvers will not be used again when a clip is replayed +- some style properties like 'backgroundGradient' had no effect in config +- video goes tiny on Firefox: http://flowplayer.org/forum/8/23226 +- RSS playlists: The 'type' attribute value 'audio/mp3' in the media:content element caused an error. +- Dispatches onMetadata() if an URL resolver changes the clip URL (changes to a different file) +- error codes and error message were not properly passed to onEvent JS listeners + +3.1.2 +----- +- The domain of the logo url must the same domain from where the player SWF is loaded from. +- Fullscreen can be toggled by doublclick on the video area. +Fixes: +- Player was not initialized correctly when instream playlists were used and the provider used in the instream clips was defined in the common clip. +- A separator in the Context Menu made the callbacks in the following menu items out of order. Related forum post: http://flowplayer.org/forum/8/22541 +- the width and height settings of a logo were ignored if the logo was a sWF file +- volume control and mute/unmute were not working after an instream clip had been played +- now possible to use RTMP for mp3 files +- Issue 12: cuepointMultiplier was undefined in the clip object set to JS event listeners +- Issue 14: onBeforeStop was unnecessarily fired when calling setPlaylist() and the player was not playing, + additionally onStop was never fired even if onBeforeStop was +- fixed screen vertical placement problems that reappeared with 3.1.1 +- The rotating animation now has the same size and position as it has after initialized + +3.1.1 +----- +- External configuration files +- Instream playback +- Added toggleFullscreen() the API +- Possibility to specify controls configuration in clips +- Seek target position is now sent in the onBeforeSeek event +Fixes: +- The screen size was initially too small on Firefox (Mac) +- Did not persist a zero volume value: http://www.flowplayer.org/forum/8/18413 + +3.1.0 +----- +New features: +- clip's can have urlResolvers and connectionProviders +- Added new configuration options 'connectionCallbacks' and 'streamCallbacks'. Both accept an Array of event names as a value. + When these events get fired on the connection or stream object, corresponding Clip events will be fired by the player. + This can be used for example when firing custom events from RTMP server apps +- Added new clip event types: 'onConnectionEvent' and 'onStreamEvent' these get fired when the predefined events happen on the connection and stream objects. +- Added Security.allowDomain() to allow loaded plugins to script the player +- Added addClip(clip, index) to the API, index is optional +- Possibility to view videos without metadata, using clip.metaData: false +- Now the player's preloader uses the rotating animation instead of a percent text to indicate the progress + of loading the player SWF. You can disable the aninamtion by setting buffering: false +- calling close() now does not send the onStop event +- Clip's custom properties are now present in the root of the clip argument in all clip events that are sent to JS. + +Bug fixes: +- The preloader sometimes failed to initialize the player +- Allow seeking while in buffering state: http://flowplayer.org/forum/8/16505 +- Replay of a RTMP stream was failing after the connection had expired +- Security error when clicking on the screen if there is an image in the playlist loaded from a foreign domain +- loadPlugin() was not working +- now fullscreen works with Flash versions older than 9.0.115, in versions that do not support hardware scaling +- replaying a RTMP stream with an image in front of the stream in the playlist was not working (video stayed hidden). Happened + because the server does not send metadata if replaying the same stream. +- the scrubber is disabled if the clip is not seekable in the first frame: http://flowplayer.org/forum/8/16526 + By default if the clip has one of following extensions (the typical flash video extensions) it is seekable + in the first frame: 'f4b', 'f4p', 'f4v', 'flv'. Added new clip property seekableOnBegin that can be used to override the default. + +3.0.6 +----- +- added possibility to associate a linkUrl and linkWindow to the canvas +Fixes: +- fix for entering fullscreen for Flash versions that don't support the hardware scaled fullscreen-mode +- when showing images the duration tracking starts only after the image has been completely loaded: http://flowplayer.org/forum/2/15301 +- fix for verifying license keys for domains that have more than 4 labels in them +- if plugin loading failis because of a IO error, the plugin will be discarded and the player initialization continues: + +3.0.4 +----- +- The "play" pseudo-plugin now supports fadeIn(), fadeOut(), showPlugin(), hidePlugin() and + additionally you can configure it like this: + // make only the play button invisible (buffering animation is still used) + play: { display: 'none' } + // disable the play button and the buffering animation + play: null + // disable the buffering animation + buffering: null +- Added possibility to seek when in the buffering state: http://flowplayer.org/forum/3/13896 +- Added copyright notices and other GPL required entries to the user interface + +Fixes: +- clip urls were not resolved correctly if the HTML page URL had a query string starting with a question mark (http://flowplayer.org/forum/8/14016#post-14016) +- Fixed context menu for with IE (commercial version) +- a cuepoint at time zero was fired several times +- screen is now arranged correctly even when only bottom or top is defined for it in the configuration +- Fixed context menu for with IE (commercial version) +- a cuepoint at time zero was fired several times +- screen is now arranged correctly even when only bottom or top is defined for it in the configuration +- Now possible to call play() in an onError handler: http://flowplayer.org/forum/8/12939 +- Does not throw an error if the player cannot persist the volume on the client computer: http://flowplayer.org/forum/8/13286#post-13495 +- Triggering fullscreen does not pause the player in IE +- The play button overlay no longer has a gap between it's pieces when a label is used: http://flowplayer.org/forum/8/14250 +- clip.update() JS call now resets the duration +- a label configured for the play button overlay did not work in the commercial version + +3.0.3 +----- +- fixed cuepoint firing: Does not skip cuepoints any more +- Plugins can now be loaded from a different domain to the flowplayer.swf +- Specifying a clip to play by just using the 'clip' node in the configuration was not working, a playlist definition was required. This is now fixed. +- Fixed: A playlist with different providers caused the onMetadata event to fire events with metadata from the previous clip in the playlist. Occurred when moving in the playlist with next() and prev() +- the opacity setting now works with the logo +- fadeOut() call to the "screen" plugin was sending the listenerId and pluginName arguments in wrong order +- stop(), pause(), resume(), close() no longer return the flowplayer object to JS +- changing the size of the screen in a onFullscreen listener now always works, there was a bug that caused this to fail occasionally +- fixed using arbitrary SWFs as plugins +- the API method setPlaylist() no longer starts playing if autoPlay: true, neither it starts buffering if autoBuffering: true +- the API method play() now accepts an array of clip objects as an argument, the playlist is replaced with the specified clips and playback starts from the 1st clip + +3.0.2 +----- +- setting play: null now works again +- pressing the play again button overlay does not open a linkUrl associated with a clip +- now displays a live feed even when the RTMP server does not send any metadata and the onStart method is not therefore dispatched +- added onMetaData clip event +- fixed 'orig' scaling: the player went to 'fit' scaling after coming back from fullscreen. This is now fixed and the original dimensions are preserved in non-fullscreen mode. +- cuepoint times are now given in milliseconds, the firing precision is 100 ms. All cuepoint times are rounded to the nearest 100 ms value (for example 1120 rounds to 1100) +- backgroundGradient was drawn over the background image in the canvas and in the content and controlbar plugins. Now it's drawn below the image. +- added cuepointMultiplier property to clips. This can be used to multiply the time values read from cuepoint metadata embedded into video files. +- the player's framerate was increased to 24 FPS, makes all animations smoother + +3.0.1 +----- +- Fixed negative cuepoints from common clip. Now these are properly propagated to the clips in playlist. +- buffering animation is now the same size as the play button overlay +- commercial version now supports license keys that allows the use of subdomains +- error messages are now automatically hidden after a 4 second delay. They are also hidden when a new clips + starts playing (when onBeforeBegin is fired) +- added possibility to disable the buffering animation like so: buffering: false +- pressing the play button overlay does not open a linkUrl associated with a clip +- license key verification failed if a port number was used in the URL (like in this url: http://mydomain.com:8080/video.html) +- added audio support, clip has a new "image" property +- workaround for missing "NetStream.Play.Start" notfication that was happending with Red5. Because of this issue the video was not shown. +- commercial version has the possibility to change the zIndex of the logo + +3.0.0 +----- +- Removed security errors that happened when loading images from foreign domains (domains other than the domain of the core SWF). + Using a backgroundImage on canvas, in the content plugin, and for the controls is also possible to be loaded + from a foreign domain - BUT backgroundRepeat cannot be used for foreign images. +- Now allows the embedding HTML to script the player even if the player is loaded from another domain. +- Added a 'live' property to Clips, used for live streams. +- A player embedded to a foreign domain now loads images, css files and other resources from the domain where the palyer SWF was loaded from. This is to generate shorter embed-codes. +- Added linkUrl and linkWindow properties to the logo, in commercial version you can set these to point to a linked page. The linked page gets opened + when the logo is clicked. Possible values for linkWindow: + * "_self" specifies the current frame in the current window. + * "_blank" specifies a new window. + * "_parent" specifies the parent of the current frame. + * "_top" specifies the top-level frame in the current window. +- Added linkUrl and linkWindow properties to clips. The linked page is opened when the video are is clicked and the corresponding clip has a linkUrl specified. +- Made the play button overlay and the "Play again" button slightly bigger. + +RC4 +--- +- Now shows a "Play again" button at the end of the video/playlist +- Commercial version shows a Flowplayer logo if invalidKey was supplied, but the otherwise the player works +- setting play: null in configuration will disable the play button overlay +- setting opacity for "play" also sets it for the buffering animation +- Fixed firing of cuepoints too early. Cuepoint firing is now based on stream time and does not rely on timers +- added onXMPData event listener +- Should not stop playback too early before the clip is really completed +- The START event is now delayed so that the metadata is available when the event is fired, METADATA event was removed, + new event BEGIN that is dispatched when the playback has been successfully started. Metadata is not normally + available when BEGIN is fired. + +RC3 +--- +- stopBuffering() now dispatches the onStop event first if the player is playing/paused/buffering at the time of calling it +- fixed detection of images based on file extensions +- fixed some issues with having images in the playlist +- made it possible to autoBuffer next video while showing an image (image without a duration) + +RC2 +--- +- fixed: setting the screen height in configuration did not have any effect + +RC1 +----- +- better error message if plugin loading fails, shows the URL used +- validates our redesigned multidomain license key correctly +- fix to prevent the play button going visible when the onBufferEmpty event occurs +- the commercial swf now correctly loads the controls using version information +- fixed: the play button overlay became invisible with long fadeOutSpeeds + +beta6 +----- +- removed the onFirstFramePause event +- playing a clip for the second time caused a doubled sound +- pausing on first frame did not work on some FLV files + +beta5 +----- +- logo only uses percentage scaling if it's a SWF file (there is ".swf" in it's url) +- context menu now correctly builds up from string entries in configuration +-always closes the previous connection before starting a new clip + +beta4 +----- +- now it's possible to load a plugin into the panel without specifying any position/dimensions + information, the plugin is placed to left: "50%", top: "50%" and using the plugin DisplayObject's width & height +- The Flowplayer API was not fully initialized when onLoad was invoked on Flash plugins + +beta3 +----- +- tweaking logo placement +- "play" did not show up after repeated pause/resume +- player now loads the latest controls SWF version, right now the latest SWF is called 'flowplayer.controls-3.0.0-beta2.swf' + +beta2 +----- +- fixed support for RTMP stream groups +- changed to loop through available fonts in order to find a suitable font also in IE +- Preloader was broken on IE: When the player SWf was in browser's cache it did not initialize properly +- Context menu now correctly handles menu items that are configured by their string labels only (not using json objects) +- fixed custom logo positioning (was moved to the left edge of screen in fullscreen) +- "play" now always follows the position and size of the screen +- video was stretched below the controls in fullscreen when autoHide: 'never' +- logo now takes 6.5% of the screen height, width is scaled so that the aspect ratio is preserved + +beta1 +----- +- First public beta release diff --git a/bbb-screenshare/jws/player/flowplayer/example/flowplayer-3.2.6.min.js b/bbb-screenshare/jws/player/flowplayer/example/flowplayer-3.2.6.min.js new file mode 100755 index 0000000000000000000000000000000000000000..500492e297789717ab506954af1f715cbb450785 --- /dev/null +++ b/bbb-screenshare/jws/player/flowplayer/example/flowplayer-3.2.6.min.js @@ -0,0 +1,24 @@ +/* + * flowplayer.js 3.2.6. The Flowplayer API + * + * Copyright 2009-2011 Flowplayer Oy + * + * This file is part of Flowplayer. + * + * Flowplayer is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * Flowplayer is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with Flowplayer. If not, see <http://www.gnu.org/licenses/>. + * + * Date: 2011-02-04 05:45:28 -0500 (Fri, 04 Feb 2011) + * Revision: 614 + */ +(function(){function g(o){console.log("$f.fireEvent",[].slice.call(o))}function k(q){if(!q||typeof q!="object"){return q}var o=new q.constructor();for(var p in q){if(q.hasOwnProperty(p)){o[p]=k(q[p])}}return o}function m(t,q){if(!t){return}var o,p=0,r=t.length;if(r===undefined){for(o in t){if(q.call(t[o],o,t[o])===false){break}}}else{for(var s=t[0];p<r&&q.call(s,p,s)!==false;s=t[++p]){}}return t}function c(o){return document.getElementById(o)}function i(q,p,o){if(typeof p!="object"){return q}if(q&&p){m(p,function(r,s){if(!o||typeof s!="function"){q[r]=s}})}return q}function n(s){var q=s.indexOf(".");if(q!=-1){var p=s.slice(0,q)||"*";var o=s.slice(q+1,s.length);var r=[];m(document.getElementsByTagName(p),function(){if(this.className&&this.className.indexOf(o)!=-1){r.push(this)}});return r}}function f(o){o=o||window.event;if(o.preventDefault){o.stopPropagation();o.preventDefault()}else{o.returnValue=false;o.cancelBubble=true}return false}function j(q,o,p){q[o]=q[o]||[];q[o].push(p)}function e(){return"_"+(""+Math.random()).slice(2,10)}var h=function(t,r,s){var q=this,p={},u={};q.index=r;if(typeof t=="string"){t={url:t}}i(this,t,true);m(("Begin*,Start,Pause*,Resume*,Seek*,Stop*,Finish*,LastSecond,Update,BufferFull,BufferEmpty,BufferStop").split(","),function(){var v="on"+this;if(v.indexOf("*")!=-1){v=v.slice(0,v.length-1);var w="onBefore"+v.slice(2);q[w]=function(x){j(u,w,x);return q}}q[v]=function(x){j(u,v,x);return q};if(r==-1){if(q[w]){s[w]=q[w]}if(q[v]){s[v]=q[v]}}});i(this,{onCuepoint:function(x,w){if(arguments.length==1){p.embedded=[null,x];return q}if(typeof x=="number"){x=[x]}var v=e();p[v]=[x,w];if(s.isLoaded()){s._api().fp_addCuepoints(x,r,v)}return q},update:function(w){i(q,w);if(s.isLoaded()){s._api().fp_updateClip(w,r)}var v=s.getConfig();var x=(r==-1)?v.clip:v.playlist[r];i(x,w,true)},_fireEvent:function(v,y,w,A){if(v=="onLoad"){m(p,function(B,C){if(C[0]){s._api().fp_addCuepoints(C[0],r,B)}});return false}A=A||q;if(v=="onCuepoint"){var z=p[y];if(z){return z[1].call(s,A,w)}}if(y&&"onBeforeBegin,onMetaData,onStart,onUpdate,onResume".indexOf(v)!=-1){i(A,y);if(y.metaData){if(!A.duration){A.duration=y.metaData.duration}else{A.fullDuration=y.metaData.duration}}}var x=true;m(u[v],function(){x=this.call(s,A,y,w)});return x}});if(t.onCuepoint){var o=t.onCuepoint;q.onCuepoint.apply(q,typeof o=="function"?[o]:o);delete t.onCuepoint}m(t,function(v,w){if(typeof w=="function"){j(u,v,w);delete t[v]}});if(r==-1){s.onCuepoint=this.onCuepoint}};var l=function(p,r,q,t){var o=this,s={},u=false;if(t){i(s,t)}m(r,function(v,w){if(typeof w=="function"){s[v]=w;delete r[v]}});i(this,{animate:function(y,z,x){if(!y){return o}if(typeof z=="function"){x=z;z=500}if(typeof y=="string"){var w=y;y={};y[w]=z;z=500}if(x){var v=e();s[v]=x}if(z===undefined){z=500}r=q._api().fp_animate(p,y,z,v);return o},css:function(w,x){if(x!==undefined){var v={};v[w]=x;w=v}r=q._api().fp_css(p,w);i(o,r);return o},show:function(){this.display="block";q._api().fp_showPlugin(p);return o},hide:function(){this.display="none";q._api().fp_hidePlugin(p);return o},toggle:function(){this.display=q._api().fp_togglePlugin(p);return o},fadeTo:function(y,x,w){if(typeof x=="function"){w=x;x=500}if(w){var v=e();s[v]=w}this.display=q._api().fp_fadeTo(p,y,x,v);this.opacity=y;return o},fadeIn:function(w,v){return o.fadeTo(1,w,v)},fadeOut:function(w,v){return o.fadeTo(0,w,v)},getName:function(){return p},getPlayer:function(){return q},_fireEvent:function(w,v,x){if(w=="onUpdate"){var z=q._api().fp_getPlugin(p);if(!z){return}i(o,z);delete o.methods;if(!u){m(z.methods,function(){var B=""+this;o[B]=function(){var C=[].slice.call(arguments);var D=q._api().fp_invoke(p,B,C);return D==="undefined"||D===undefined?o:D}});u=true}}var A=s[w];if(A){var y=A.apply(o,v);if(w.slice(0,1)=="_"){delete s[w]}return y}return o}})};function b(q,G,t){var w=this,v=null,D=false,u,s,F=[],y={},x={},E,r,p,C,o,A;i(w,{id:function(){return E},isLoaded:function(){return(v!==null&&v.fp_play!==undefined&&!D)},getParent:function(){return q},hide:function(H){if(H){q.style.height="0px"}if(w.isLoaded()){v.style.height="0px"}return w},show:function(){q.style.height=A+"px";if(w.isLoaded()){v.style.height=o+"px"}return w},isHidden:function(){return w.isLoaded()&&parseInt(v.style.height,10)===0},load:function(J){if(!w.isLoaded()&&w._fireEvent("onBeforeLoad")!==false){var H=function(){u=q.innerHTML;if(u&&!flashembed.isSupported(G.version)){q.innerHTML=""}if(J){J.cached=true;j(x,"onLoad",J)}flashembed(q,G,{config:t})};var I=0;m(a,function(){this.unload(function(K){if(++I==a.length){H()}})})}return w},unload:function(J){if(this.isFullscreen()&&/WebKit/i.test(navigator.userAgent)){if(J){J(false)}return w}if(u.replace(/\s/g,"")!==""){if(w._fireEvent("onBeforeUnload")===false){if(J){J(false)}return w}D=true;try{if(v){v.fp_close();w._fireEvent("onUnload")}}catch(H){}var I=function(){v=null;q.innerHTML=u;D=false;if(J){J(true)}};setTimeout(I,50)}else{if(J){J(false)}}return w},getClip:function(H){if(H===undefined){H=C}return F[H]},getCommonClip:function(){return s},getPlaylist:function(){return F},getPlugin:function(H){var J=y[H];if(!J&&w.isLoaded()){var I=w._api().fp_getPlugin(H);if(I){J=new l(H,I,w);y[H]=J}}return J},getScreen:function(){return w.getPlugin("screen")},getControls:function(){return w.getPlugin("controls")._fireEvent("onUpdate")},getLogo:function(){try{return w.getPlugin("logo")._fireEvent("onUpdate")}catch(H){}},getPlay:function(){return w.getPlugin("play")._fireEvent("onUpdate")},getConfig:function(H){return H?k(t):t},getFlashParams:function(){return G},loadPlugin:function(K,J,M,L){if(typeof M=="function"){L=M;M={}}var I=L?e():"_";w._api().fp_loadPlugin(K,J,M,I);var H={};H[I]=L;var N=new l(K,null,w,H);y[K]=N;return N},getState:function(){return w.isLoaded()?v.fp_getState():-1},play:function(I,H){var J=function(){if(I!==undefined){w._api().fp_play(I,H)}else{w._api().fp_play()}};if(w.isLoaded()){J()}else{if(D){setTimeout(function(){w.play(I,H)},50)}else{w.load(function(){J()})}}return w},getVersion:function(){var I="flowplayer.js 3.2.6";if(w.isLoaded()){var H=v.fp_getVersion();H.push(I);return H}return I},_api:function(){if(!w.isLoaded()){throw"Flowplayer "+w.id()+" not loaded when calling an API method"}return v},setClip:function(H){w.setPlaylist([H]);return w},getIndex:function(){return p},_swfHeight:function(){return v.clientHeight}});m(("Click*,Load*,Unload*,Keypress*,Volume*,Mute*,Unmute*,PlaylistReplace,ClipAdd,Fullscreen*,FullscreenExit,Error,MouseOver,MouseOut").split(","),function(){var H="on"+this;if(H.indexOf("*")!=-1){H=H.slice(0,H.length-1);var I="onBefore"+H.slice(2);w[I]=function(J){j(x,I,J);return w}}w[H]=function(J){j(x,H,J);return w}});m(("pause,resume,mute,unmute,stop,toggle,seek,getStatus,getVolume,setVolume,getTime,isPaused,isPlaying,startBuffering,stopBuffering,isFullscreen,toggleFullscreen,reset,close,setPlaylist,addClip,playFeed,setKeyboardShortcutsEnabled,isKeyboardShortcutsEnabled").split(","),function(){var H=this;w[H]=function(J,I){if(!w.isLoaded()){return w}var K=null;if(J!==undefined&&I!==undefined){K=v["fp_"+H](J,I)}else{K=(J===undefined)?v["fp_"+H]():v["fp_"+H](J)}return K==="undefined"||K===undefined?w:K}});w._fireEvent=function(Q){if(typeof Q=="string"){Q=[Q]}var R=Q[0],O=Q[1],M=Q[2],L=Q[3],K=0;if(t.debug){g(Q)}if(!w.isLoaded()&&R=="onLoad"&&O=="player"){v=v||c(r);o=w._swfHeight();m(F,function(){this._fireEvent("onLoad")});m(y,function(S,T){T._fireEvent("onUpdate")});s._fireEvent("onLoad")}if(R=="onLoad"&&O!="player"){return}if(R=="onError"){if(typeof O=="string"||(typeof O=="number"&&typeof M=="number")){O=M;M=L}}if(R=="onContextMenu"){m(t.contextMenu[O],function(S,T){T.call(w)});return}if(R=="onPluginEvent"||R=="onBeforePluginEvent"){var H=O.name||O;var I=y[H];if(I){I._fireEvent("onUpdate",O);return I._fireEvent(M,Q.slice(3))}return}if(R=="onPlaylistReplace"){F=[];var N=0;m(O,function(){F.push(new h(this,N++,w))})}if(R=="onClipAdd"){if(O.isInStream){return}O=new h(O,M,w);F.splice(M,0,O);for(K=M+1;K<F.length;K++){F[K].index++}}var P=true;if(typeof O=="number"&&O<F.length){C=O;var J=F[O];if(J){P=J._fireEvent(R,M,L)}if(!J||P!==false){P=s._fireEvent(R,M,L,J)}}m(x[R],function(){P=this.call(w,O,M);if(this.cached){x[R].splice(K,1)}if(P===false){return false}K++});return P};function B(){if($f(q)){$f(q).getParent().innerHTML="";p=$f(q).getIndex();a[p]=w}else{a.push(w);p=a.length-1}A=parseInt(q.style.height,10)||q.clientHeight;E=q.id||"fp"+e();r=G.id||E+"_api";G.id=r;t.playerId=E;if(typeof t=="string"){t={clip:{url:t}}}if(typeof t.clip=="string"){t.clip={url:t.clip}}t.clip=t.clip||{};if(q.getAttribute("href",2)&&!t.clip.url){t.clip.url=q.getAttribute("href",2)}s=new h(t.clip,-1,w);t.playlist=t.playlist||[t.clip];var I=0;m(t.playlist,function(){var K=this;if(typeof K=="object"&&K.length){K={url:""+K}}m(t.clip,function(L,M){if(M!==undefined&&K[L]===undefined&&typeof M!="function"){K[L]=M}});t.playlist[I]=K;K=new h(K,I,w);F.push(K);I++});m(t,function(K,L){if(typeof L=="function"){if(s[K]){s[K](L)}else{j(x,K,L)}delete t[K]}});m(t.plugins,function(K,L){if(L){y[K]=new l(K,L,w)}});if(!t.plugins||t.plugins.controls===undefined){y.controls=new l("controls",null,w)}y.canvas=new l("canvas",null,w);u=q.innerHTML;function J(L){var K=w.hasiPadSupport&&w.hasiPadSupport();if(/iPad|iPhone|iPod/i.test(navigator.userAgent)&&!/.flv$/i.test(F[0].url)&&!K){return true}if(!w.isLoaded()&&w._fireEvent("onBeforeClick")!==false){w.load()}return f(L)}function H(){if(u.replace(/\s/g,"")!==""){if(q.addEventListener){q.addEventListener("click",J,false)}else{if(q.attachEvent){q.attachEvent("onclick",J)}}}else{if(q.addEventListener){q.addEventListener("click",f,false)}w.load()}}setTimeout(H,0)}if(typeof q=="string"){var z=c(q);if(!z){throw"Flowplayer cannot access element: "+q}q=z;B()}else{B()}}var a=[];function d(o){this.length=o.length;this.each=function(p){m(o,p)};this.size=function(){return o.length}}window.flowplayer=window.$f=function(){var p=null;var o=arguments[0];if(!arguments.length){m(a,function(){if(this.isLoaded()){p=this;return false}});return p||a[0]}if(arguments.length==1){if(typeof o=="number"){return a[o]}else{if(o=="*"){return new d(a)}m(a,function(){if(this.id()==o.id||this.id()==o||this.getParent()==o){p=this;return false}});return p}}if(arguments.length>1){var t=arguments[1],q=(arguments.length==3)?arguments[2]:{};if(typeof t=="string"){t={src:t}}t=i({bgcolor:"#000000",version:[9,0],expressInstall:"http://static.flowplayer.org/swf/expressinstall.swf",cachebusting:false},t);if(typeof o=="string"){if(o.indexOf(".")!=-1){var s=[];m(n(o),function(){s.push(new b(this,k(t),k(q)))});return new d(s)}else{var r=c(o);return new b(r!==null?r:o,t,q)}}else{if(o){return new b(o,t,q)}}}return null};i(window.$f,{fireEvent:function(){var o=[].slice.call(arguments);var q=$f(o[0]);return q?q._fireEvent(o.slice(1)):null},addPlugin:function(o,p){b.prototype[o]=p;return $f},each:m,extend:i});if(typeof jQuery=="function"){jQuery.fn.flowplayer=function(q,p){if(!arguments.length||typeof arguments[0]=="number"){var o=[];this.each(function(){var r=$f(this);if(r){o.push(r)}});return arguments.length?o[arguments[0]]:new d(o)}return this.each(function(){$f(this,k(q),p?k(p):{})})}}})();(function(){var e=typeof jQuery=="function";var i={width:"100%",height:"100%",allowfullscreen:true,allowscriptaccess:"always",quality:"high",version:null,onFail:null,expressInstall:null,w3c:false,cachebusting:false};if(e){jQuery.tools=jQuery.tools||{};jQuery.tools.flashembed={version:"1.0.4",conf:i}}function j(){if(c.done){return false}var l=document;if(l&&l.getElementsByTagName&&l.getElementById&&l.body){clearInterval(c.timer);c.timer=null;for(var k=0;k<c.ready.length;k++){c.ready[k].call()}c.ready=null;c.done=true}}var c=e?jQuery:function(k){if(c.done){return k()}if(c.timer){c.ready.push(k)}else{c.ready=[k];c.timer=setInterval(j,13)}};function f(l,k){if(k){for(key in k){if(k.hasOwnProperty(key)){l[key]=k[key]}}}return l}function g(k){switch(h(k)){case"string":k=k.replace(new RegExp('(["\\\\])',"g"),"\\$1");k=k.replace(/^\s?(\d+)%/,"$1pct");return'"'+k+'"';case"array":return"["+b(k,function(n){return g(n)}).join(",")+"]";case"function":return'"function()"';case"object":var l=[];for(var m in k){if(k.hasOwnProperty(m)){l.push('"'+m+'":'+g(k[m]))}}return"{"+l.join(",")+"}"}return String(k).replace(/\s/g," ").replace(/\'/g,'"')}function h(l){if(l===null||l===undefined){return false}var k=typeof l;return(k=="object"&&l.push)?"array":k}if(window.attachEvent){window.attachEvent("onbeforeunload",function(){__flash_unloadHandler=function(){};__flash_savedUnloadHandler=function(){}})}function b(k,n){var m=[];for(var l in k){if(k.hasOwnProperty(l)){m[l]=n(k[l])}}return m}function a(r,t){var q=f({},r);var s=document.all;var n='<object width="'+q.width+'" height="'+q.height+'"';if(s&&!q.id){q.id="_"+(""+Math.random()).substring(9)}if(q.id){n+=' id="'+q.id+'"'}if(q.cachebusting){q.src+=((q.src.indexOf("?")!=-1?"&":"?")+Math.random())}if(q.w3c||!s){n+=' data="'+q.src+'" type="application/x-shockwave-flash"'}else{n+=' classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'}n+=">";if(q.w3c||s){n+='<param name="movie" value="'+q.src+'" />'}q.width=q.height=q.id=q.w3c=q.src=null;for(var l in q){if(q[l]!==null){n+='<param name="'+l+'" value="'+q[l]+'" />'}}var o="";if(t){for(var m in t){if(t[m]!==null){o+=m+"="+(typeof t[m]=="object"?g(t[m]):t[m])+"&"}}o=o.substring(0,o.length-1);n+='<param name="flashvars" value=\''+o+"' />"}n+="</object>";return n}function d(m,p,l){var k=flashembed.getVersion();f(this,{getContainer:function(){return m},getConf:function(){return p},getVersion:function(){return k},getFlashvars:function(){return l},getApi:function(){return m.firstChild},getHTML:function(){return a(p,l)}});var q=p.version;var r=p.expressInstall;var o=!q||flashembed.isSupported(q);if(o){p.onFail=p.version=p.expressInstall=null;m.innerHTML=a(p,l)}else{if(q&&r&&flashembed.isSupported([6,65])){f(p,{src:r});l={MMredirectURL:location.href,MMplayerType:"PlugIn",MMdoctitle:document.title};m.innerHTML=a(p,l)}else{if(m.innerHTML.replace(/\s/g,"")!==""){}else{m.innerHTML="<h2>Flash version "+q+" or greater is required</h2><h3>"+(k[0]>0?"Your version is "+k:"You have no flash plugin installed")+"</h3>"+(m.tagName=="A"?"<p>Click here to download latest version</p>":"<p>Download latest version from <a href='http://www.adobe.com/go/getflashplayer'>here</a></p>");if(m.tagName=="A"){m.onclick=function(){location.href="http://www.adobe.com/go/getflashplayer"}}}}}if(!o&&p.onFail){var n=p.onFail.call(this);if(typeof n=="string"){m.innerHTML=n}}if(document.all){window[p.id]=document.getElementById(p.id)}}window.flashembed=function(l,m,k){if(typeof l=="string"){var n=document.getElementById(l);if(n){l=n}else{c(function(){flashembed(l,m,k)});return}}if(!l){return}if(typeof m=="string"){m={src:m}}var o=f({},i);f(o,m);return new d(l,o,k)};f(window.flashembed,{getVersion:function(){var m=[0,0];if(navigator.plugins&&typeof navigator.plugins["Shockwave Flash"]=="object"){var l=navigator.plugins["Shockwave Flash"].description;if(typeof l!="undefined"){l=l.replace(/^.*\s+(\S+\s+\S+$)/,"$1");var n=parseInt(l.replace(/^(.*)\..*$/,"$1"),10);var r=/r/.test(l)?parseInt(l.replace(/^.*r(.*)$/,"$1"),10):0;m=[n,r]}}else{if(window.ActiveXObject){try{var p=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.7")}catch(q){try{p=new ActiveXObject("ShockwaveFlash.ShockwaveFlash.6");m=[6,0];p.AllowScriptAccess="always"}catch(k){if(m[0]==6){return m}}try{p=new ActiveXObject("ShockwaveFlash.ShockwaveFlash")}catch(o){}}if(typeof p=="object"){l=p.GetVariable("$version");if(typeof l!="undefined"){l=l.replace(/^\S+\s+(.*)$/,"$1").split(",");m=[parseInt(l[0],10),parseInt(l[2],10)]}}}}return m},isSupported:function(k){var m=flashembed.getVersion();var l=(m[0]>k[0])||(m[0]==k[0]&&m[1]>=k[1]);return l},domReady:c,asString:g,getHTML:a});if(e){jQuery.fn.flashembed=function(l,k){var m=null;this.each(function(){m=flashembed(this,l,k)});return l.api===false?this:m}}})(); \ No newline at end of file diff --git a/bbb-screenshare/jws/player/flowplayer/flowplayer-3.2.7.swf b/bbb-screenshare/jws/player/flowplayer/flowplayer-3.2.7.swf new file mode 100755 index 0000000000000000000000000000000000000000..20a41193f5ba6f63745632946e8ff6c1e52313ca Binary files /dev/null and b/bbb-screenshare/jws/player/flowplayer/flowplayer-3.2.7.swf differ diff --git a/bbb-screenshare/jws/player/flowplayer/flowplayer.controls-3.2.5.swf b/bbb-screenshare/jws/player/flowplayer/flowplayer.controls-3.2.5.swf new file mode 100755 index 0000000000000000000000000000000000000000..5507a531ddcc0baccaf1339c852ee1623dfd5b05 Binary files /dev/null and b/bbb-screenshare/jws/player/flowplayer/flowplayer.controls-3.2.5.swf differ diff --git a/bbb-screenshare/jws/player/flowplayer/flowplayer.rtmp-3.2.3.swf b/bbb-screenshare/jws/player/flowplayer/flowplayer.rtmp-3.2.3.swf new file mode 100755 index 0000000000000000000000000000000000000000..43f4c1961cf3be054481b068bed8b239dd9fa6cc Binary files /dev/null and b/bbb-screenshare/jws/player/flowplayer/flowplayer.rtmp-3.2.3.swf differ diff --git a/bbb-screenshare/jws/player/flowplayer/index.html b/bbb-screenshare/jws/player/flowplayer/index.html new file mode 100755 index 0000000000000000000000000000000000000000..6e9e51897f02ef5d8562b5ef9f8eb098f9bbe249 --- /dev/null +++ b/bbb-screenshare/jws/player/flowplayer/index.html @@ -0,0 +1,29 @@ +<html> + <head> + <script src="example/flowplayer-3.2.6.min.js"></script> + </head> + <body> + + <a class="rtmp" href="320x2401-1328884730718" style="display:block;width:320px;height:240px;" id="player">stream</a> + + <script language="JavaScript"> + $f("a.rtmp", "flowplayer-3.2.7.swf", { + clip: { + url: '320x2401-1328884730718', + provider: 'rtmp', + live: true, + bufferLength: 1, + autoPlay: true, + }, + plugins: { + rtmp: { + url: 'flowplayer.rtmp-3.2.3.swf', + netConnectionUrl: 'rtmp://143.54.12.217/video/183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719358' + } + } + }); + </script> + + </body> +</html> + diff --git a/bbb-screenshare/jws/player/index.html b/bbb-screenshare/jws/player/index.html new file mode 100755 index 0000000000000000000000000000000000000000..23d46b1482620691b271fb5c1395bc1b8e23cc98 --- /dev/null +++ b/bbb-screenshare/jws/player/index.html @@ -0,0 +1,46 @@ +<html> + <head> + <title>bbb-video-stream-html-client</title> + + <link href="style.css" rel="stylesheet" type="text/css"> + <script type="text/javascript" src="https://ajax.googleapis.com/ajax/libs/jquery/1.7.1/jquery.min.js"></script> + <script type="text/javascript" src="actions.js"></script> + + <script type="text/javascript" src="jw-player/jwplayer.js"></script> + <script type="text/javascript" src="flowplayer/example/flowplayer-3.2.6.min.js"></script> + </head> + <body> + + <a href="https://github.com/daronco/bbb-video-stream-html-client"><img style="position: absolute; top: 0; right: 0; border: 0;" src="https://s3.amazonaws.com/github/ribbons/forkme_right_red_aa0000.png" alt="Fork me on GitHub"></a> + + <div id="menu"> + + <form action="#"> + <div class="field"> + <label class="label" for="server_ip">Server IP</label> + <input type="text" id="server_ip" name="server_ip" value="192.168.23.23" /> + </div> + <div class="field"> + <label class="label" for="meeting_id">Meeting ID</label> + <input type="text" id="meeting_id" name="meeting_id" value="foo" /> + </div> + <div class="field"> + <label class="label" for="stream_id">Stream ID</label> + <input type="text" id="stream_id" name="stream_id" value="room2" /> + </div> + <div class="field"> + <label class="label">Player</label> + <input type="radio" name="player" value="jw-player"> jw-player + <input type="radio" name="player" value="flowplayer" checked="checked"> flowplayer + <button type="button" id="submit">Submit</button> + </div> + </form> + + </div> + + <div id="content"> + <div id="wrapper">Your video will be played here...</div> + </div> + + </body> +</html> diff --git a/bbb-screenshare/jws/player/jw-player/index.html b/bbb-screenshare/jws/player/jw-player/index.html new file mode 100755 index 0000000000000000000000000000000000000000..cde33954e79b7cf906ffff744741cb9b736735f2 --- /dev/null +++ b/bbb-screenshare/jws/player/jw-player/index.html @@ -0,0 +1,48 @@ +<html> + <head> + <script type='text/javascript' src='jwplayer.js'></script> + <!--<script type="text/javascript" + src="http://ajax.googleapis.com/ajax/libs/swfobject/2.2/swfobject.js"> + </script>--> + </head> + <body> + + <div id='mediaplayer'></div> + + <script type="text/javascript"> + jwplayer('mediaplayer').setup({ + flashplayer: 'player.swf', + id: 'playerID', + width: '320', + height: '240', + file: '320x2401-1328884730718', + streamer: 'rtmp://143.54.12.217/video/183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719358', + autostart: 'true', + provider: 'rtmp', + duration: '0', + bufferlength: '1', // it's not working + //start: '0', + //live: 'true', + //repeat: 'none', + }); + </script> + + <!-- using swfobject + <div id='container'>The player will be placed here</div> + <script type="text/javascript"> + var flashvars = { + file:'320x2401-1328884730718', + bufferlength: '1', + streamer: 'rtmp://143.54.12.217/video/183f0bf3a0982a127bdb8161e0c44eb696b3e75c-1328884719358/', + autostart: 'true', + }; + swfobject.embedSWF('player.swf','container','320','240','9.0.115','false', flashvars, + {allowfullscreen:'true',allowscriptaccess:'always', wmode:'transparent'}, + {id:'jwplayer',name:'jwplayer'} + ); + </script> + --> + + </body> +</html> + diff --git a/bbb-screenshare/jws/player/jw-player/jwplayer.js b/bbb-screenshare/jws/player/jw-player/jwplayer.js new file mode 100755 index 0000000000000000000000000000000000000000..5e8a20d1c3d152e1e054109d037d34f8462c303a --- /dev/null +++ b/bbb-screenshare/jws/player/jw-player/jwplayer.js @@ -0,0 +1 @@ +if(typeof jwplayer=="undefined"){var jwplayer=function(a){if(jwplayer.api){return jwplayer.api.selectPlayer(a)}};var $jw=jwplayer;jwplayer.version="5.9.2118";jwplayer.vid=document.createElement("video");jwplayer.audio=document.createElement("audio");jwplayer.source=document.createElement("source");(function(b){b.utils=function(){};b.utils.typeOf=function(d){var c=typeof d;if(c==="object"){if(d){if(d instanceof Array){c="array"}}else{c="null"}}return c};b.utils.extend=function(){var c=b.utils.extend["arguments"];if(c.length>1){for(var e=1;e<c.length;e++){for(var d in c[e]){c[0][d]=c[e][d]}}return c[0]}return null};b.utils.clone=function(f){var c;var d=b.utils.clone["arguments"];if(d.length==1){switch(b.utils.typeOf(d[0])){case"object":c={};for(var e in d[0]){c[e]=b.utils.clone(d[0][e])}break;case"array":c=[];for(var e in d[0]){c[e]=b.utils.clone(d[0][e])}break;default:return d[0];break}}return c};b.utils.extension=function(c){if(!c){return""}c=c.substring(c.lastIndexOf("/")+1,c.length);c=c.split("?")[0];if(c.lastIndexOf(".")>-1){return c.substr(c.lastIndexOf(".")+1,c.length).toLowerCase()}return};b.utils.html=function(c,d){c.innerHTML=d};b.utils.wrap=function(c,d){if(c.parentNode){c.parentNode.replaceChild(d,c)}d.appendChild(c)};b.utils.ajax=function(g,f,c){var e;if(window.XMLHttpRequest){e=new XMLHttpRequest()}else{e=new ActiveXObject("Microsoft.XMLHTTP")}e.onreadystatechange=function(){if(e.readyState===4){if(e.status===200){if(f){if(!b.utils.exists(e.responseXML)){try{if(window.DOMParser){var h=(new DOMParser()).parseFromString(e.responseText,"text/xml");if(h){e=b.utils.extend({},e,{responseXML:h})}}else{h=new ActiveXObject("Microsoft.XMLDOM");h.async="false";h.loadXML(e.responseText);e=b.utils.extend({},e,{responseXML:h})}}catch(j){if(c){c(g)}}}f(e)}}else{if(c){c(g)}}}};try{e.open("GET",g,true);e.send(null)}catch(d){if(c){c(g)}}return e};b.utils.load=function(d,e,c){d.onreadystatechange=function(){if(d.readyState===4){if(d.status===200){if(e){e()}}else{if(c){c()}}}}};b.utils.find=function(d,c){return d.getElementsByTagName(c)};b.utils.append=function(c,d){c.appendChild(d)};b.utils.isIE=function(){return((!+"\v1")||(typeof window.ActiveXObject!="undefined"))};b.utils.userAgentMatch=function(d){var c=navigator.userAgent.toLowerCase();return(c.match(d)!==null)};b.utils.isIOS=function(){return b.utils.userAgentMatch(/iP(hone|ad|od)/i)};b.utils.isIPad=function(){return b.utils.userAgentMatch(/iPad/i)};b.utils.isIPod=function(){return b.utils.userAgentMatch(/iP(hone|od)/i)};b.utils.isAndroid=function(){return b.utils.userAgentMatch(/android/i)};b.utils.isLegacyAndroid=function(){return b.utils.userAgentMatch(/android 2.[012]/i)};b.utils.isBlackberry=function(){return b.utils.userAgentMatch(/blackberry/i)};b.utils.isMobile=function(){return b.utils.userAgentMatch(/(iP(hone|ad|od))|android/i)};b.utils.getFirstPlaylistItemFromConfig=function(c){var d={};var e;if(c.playlist&&c.playlist.length){e=c.playlist[0]}else{e=c}d.file=e.file;d.levels=e.levels;d.streamer=e.streamer;d.playlistfile=e.playlistfile;d.provider=e.provider;if(!d.provider){if(d.file&&(d.file.toLowerCase().indexOf("youtube.com")>-1||d.file.toLowerCase().indexOf("youtu.be")>-1)){d.provider="youtube"}if(d.streamer&&d.streamer.toLowerCase().indexOf("rtmp://")==0){d.provider="rtmp"}if(e.type){d.provider=e.type.toLowerCase()}}if(d.provider=="audio"){d.provider="sound"}return d};b.utils.getOuterHTML=function(c){if(c.outerHTML){return c.outerHTML}else{try{return new XMLSerializer().serializeToString(c)}catch(d){return""}}};b.utils.setOuterHTML=function(f,e){if(f.outerHTML){f.outerHTML=e}else{var g=document.createElement("div");g.innerHTML=e;var c=document.createRange();c.selectNodeContents(g);var d=c.extractContents();f.parentNode.insertBefore(d,f);f.parentNode.removeChild(f)}};b.utils.hasFlash=function(){if(typeof navigator.plugins!="undefined"&&typeof navigator.plugins["Shockwave Flash"]!="undefined"){return true}if(typeof window.ActiveXObject!="undefined"){try{new ActiveXObject("ShockwaveFlash.ShockwaveFlash");return true}catch(c){}}return false};b.utils.getPluginName=function(c){if(c.lastIndexOf("/")>=0){c=c.substring(c.lastIndexOf("/")+1,c.length)}if(c.lastIndexOf("-")>=0){c=c.substring(0,c.lastIndexOf("-"))}if(c.lastIndexOf(".swf")>=0){c=c.substring(0,c.lastIndexOf(".swf"))}if(c.lastIndexOf(".js")>=0){c=c.substring(0,c.lastIndexOf(".js"))}return c};b.utils.getPluginVersion=function(c){if(c.lastIndexOf("-")>=0){if(c.lastIndexOf(".js")>=0){return c.substring(c.lastIndexOf("-")+1,c.lastIndexOf(".js"))}else{if(c.lastIndexOf(".swf")>=0){return c.substring(c.lastIndexOf("-")+1,c.lastIndexOf(".swf"))}else{return c.substring(c.lastIndexOf("-")+1)}}}return""};b.utils.getAbsolutePath=function(j,h){if(!b.utils.exists(h)){h=document.location.href}if(!b.utils.exists(j)){return undefined}if(a(j)){return j}var k=h.substring(0,h.indexOf("://")+3);var g=h.substring(k.length,h.indexOf("/",k.length+1));var d;if(j.indexOf("/")===0){d=j.split("/")}else{var e=h.split("?")[0];e=e.substring(k.length+g.length+1,e.lastIndexOf("/"));d=e.split("/").concat(j.split("/"))}var c=[];for(var f=0;f<d.length;f++){if(!d[f]||!b.utils.exists(d[f])||d[f]=="."){continue}else{if(d[f]==".."){c.pop()}else{c.push(d[f])}}}return k+g+"/"+c.join("/")};function a(d){if(!b.utils.exists(d)){return}var e=d.indexOf("://");var c=d.indexOf("?");return(e>0&&(c<0||(c>e)))}b.utils.pluginPathType={ABSOLUTE:"ABSOLUTE",RELATIVE:"RELATIVE",CDN:"CDN"};b.utils.getPluginPathType=function(d){if(typeof d!="string"){return}d=d.split("?")[0];var e=d.indexOf("://");if(e>0){return b.utils.pluginPathType.ABSOLUTE}var c=d.indexOf("/");var f=b.utils.extension(d);if(e<0&&c<0&&(!f||!isNaN(f))){return b.utils.pluginPathType.CDN}return b.utils.pluginPathType.RELATIVE};b.utils.mapEmpty=function(c){for(var d in c){return false}return true};b.utils.mapLength=function(d){var c=0;for(var e in d){c++}return c};b.utils.log=function(d,c){if(typeof console!="undefined"&&typeof console.log!="undefined"){if(c){console.log(d,c)}else{console.log(d)}}};b.utils.css=function(d,g,c){if(b.utils.exists(d)){for(var e in g){try{if(typeof g[e]==="undefined"){continue}else{if(typeof g[e]=="number"&&!(e=="zIndex"||e=="opacity")){if(isNaN(g[e])){continue}if(e.match(/color/i)){g[e]="#"+b.utils.strings.pad(g[e].toString(16),6)}else{g[e]=Math.ceil(g[e])+"px"}}}d.style[e]=g[e]}catch(f){}}}};b.utils.isYouTube=function(c){return(c.indexOf("youtube.com")>-1||c.indexOf("youtu.be")>-1)};b.utils.transform=function(e,d,c,g,h){if(!b.utils.exists(d)){d=1}if(!b.utils.exists(c)){c=1}if(!b.utils.exists(g)){g=0}if(!b.utils.exists(h)){h=0}if(d==1&&c==1&&g==0&&h==0){e.style.webkitTransform="";e.style.MozTransform="";e.style.OTransform=""}else{var f="scale("+d+","+c+") translate("+g+"px,"+h+"px)";e.style.webkitTransform=f;e.style.MozTransform=f;e.style.OTransform=f}};b.utils.stretch=function(k,q,p,g,n,h){if(typeof p=="undefined"||typeof g=="undefined"||typeof n=="undefined"||typeof h=="undefined"){return}var d=p/n;var f=g/h;var m=0;var l=0;var e=false;var c={};if(q.parentElement){q.parentElement.style.overflow="hidden"}b.utils.transform(q);switch(k.toUpperCase()){case b.utils.stretching.NONE:c.width=n;c.height=h;c.top=(g-c.height)/2;c.left=(p-c.width)/2;break;case b.utils.stretching.UNIFORM:if(d>f){c.width=n*f;c.height=h*f;if(c.width/p>0.95){e=true;d=Math.ceil(100*p/c.width)/100;f=1;c.width=p}}else{c.width=n*d;c.height=h*d;if(c.height/g>0.95){e=true;d=1;f=Math.ceil(100*g/c.height)/100;c.height=g}}c.top=(g-c.height)/2;c.left=(p-c.width)/2;break;case b.utils.stretching.FILL:if(d>f){c.width=n*d;c.height=h*d}else{c.width=n*f;c.height=h*f}c.top=(g-c.height)/2;c.left=(p-c.width)/2;break;case b.utils.stretching.EXACTFIT:c.width=n;c.height=h;var o=Math.round((n/2)*(1-1/d));var j=Math.round((h/2)*(1-1/f));e=true;c.top=c.left=0;break;default:break}if(e){b.utils.transform(q,d,f,o,j)}b.utils.css(q,c)};b.utils.stretching={NONE:"NONE",FILL:"FILL",UNIFORM:"UNIFORM",EXACTFIT:"EXACTFIT"};b.utils.deepReplaceKeyName=function(k,e,c){switch(b.utils.typeOf(k)){case"array":for(var g=0;g<k.length;g++){k[g]=b.utils.deepReplaceKeyName(k[g],e,c)}break;case"object":for(var f in k){var j,h;if(e instanceof Array&&c instanceof Array){if(e.length!=c.length){continue}else{j=e;h=c}}else{j=[e];h=[c]}var d=f;for(var g=0;g<j.length;g++){d=d.replace(new RegExp(e[g],"g"),c[g])}k[d]=b.utils.deepReplaceKeyName(k[f],e,c);if(f!=d){delete k[f]}}break}return k};b.utils.isInArray=function(e,d){if(!(e)||!(e instanceof Array)){return false}for(var c=0;c<e.length;c++){if(d===e[c]){return true}}return false};b.utils.exists=function(c){switch(typeof(c)){case"string":return(c.length>0);break;case"object":return(c!==null);case"undefined":return false}return true};b.utils.empty=function(c){if(typeof c.hasChildNodes=="function"){while(c.hasChildNodes()){c.removeChild(c.firstChild)}}};b.utils.parseDimension=function(c){if(typeof c=="string"){if(c===""){return 0}else{if(c.lastIndexOf("%")>-1){return c}else{return parseInt(c.replace("px",""),10)}}}return c};b.utils.getDimensions=function(c){if(c&&c.style){return{x:b.utils.parseDimension(c.style.left),y:b.utils.parseDimension(c.style.top),width:b.utils.parseDimension(c.style.width),height:b.utils.parseDimension(c.style.height)}}else{return{}}};b.utils.getElementWidth=function(c){if(!c){return null}else{if(c==document.body){return b.utils.parentNode(c).clientWidth}else{if(c.clientWidth>0){return c.clientWidth}else{if(c.style){return b.utils.parseDimension(c.style.width)}else{return null}}}}};b.utils.getElementHeight=function(c){if(!c){return null}else{if(c==document.body){return b.utils.parentNode(c).clientHeight}else{if(c.clientHeight>0){return c.clientHeight}else{if(c.style){return b.utils.parseDimension(c.style.height)}else{return null}}}}};b.utils.timeFormat=function(c){str="00:00";if(c>0){str=Math.floor(c/60)<10?"0"+Math.floor(c/60)+":":Math.floor(c/60)+":";str+=Math.floor(c%60)<10?"0"+Math.floor(c%60):Math.floor(c%60)}return str};b.utils.useNativeFullscreen=function(){return(navigator&&navigator.vendor&&navigator.vendor.indexOf("Apple")==0)};b.utils.parentNode=function(c){if(!c){return docuemnt.body}else{if(c.parentNode){return c.parentNode}else{if(c.parentElement){return c.parentElement}else{return c}}}};b.utils.getBoundingClientRect=function(c){if(typeof c.getBoundingClientRect=="function"){return c.getBoundingClientRect()}else{return{left:c.offsetLeft+document.body.scrollLeft,top:c.offsetTop+document.body.scrollTop,width:c.offsetWidth,height:c.offsetHeight}}};b.utils.translateEventResponse=function(e,c){var g=b.utils.extend({},c);if(e==b.api.events.JWPLAYER_FULLSCREEN&&!g.fullscreen){g.fullscreen=g.message=="true"?true:false;delete g.message}else{if(typeof g.data=="object"){g=b.utils.extend(g,g.data);delete g.data}else{if(typeof g.metadata=="object"){b.utils.deepReplaceKeyName(g.metadata,["__dot__","__spc__","__dsh__"],["."," ","-"])}}}var d=["position","duration","offset"];for(var f in d){if(g[d[f]]){g[d[f]]=Math.round(g[d[f]]*1000)/1000}}return g};b.utils.saveCookie=function(c,d){document.cookie="jwplayer."+c+"="+d+"; path=/"};b.utils.getCookies=function(){var f={};var e=document.cookie.split("; ");for(var d=0;d<e.length;d++){var c=e[d].split("=");if(c[0].indexOf("jwplayer.")==0){f[c[0].substring(9,c[0].length)]=c[1]}}return f};b.utils.readCookie=function(c){return b.utils.getCookies()[c]}})(jwplayer);(function(a){a.events=function(){};a.events.COMPLETE="COMPLETE";a.events.ERROR="ERROR"})(jwplayer);(function(jwplayer){jwplayer.events.eventdispatcher=function(debug){var _debug=debug;var _listeners;var _globallisteners;this.resetEventListeners=function(){_listeners={};_globallisteners=[]};this.resetEventListeners();this.addEventListener=function(type,listener,count){try{if(!jwplayer.utils.exists(_listeners[type])){_listeners[type]=[]}if(typeof(listener)=="string"){eval("listener = "+listener)}_listeners[type].push({listener:listener,count:count})}catch(err){jwplayer.utils.log("error",err)}return false};this.removeEventListener=function(type,listener){if(!_listeners[type]){return}try{for(var listenerIndex=0;listenerIndex<_listeners[type].length;listenerIndex++){if(_listeners[type][listenerIndex].listener.toString()==listener.toString()){_listeners[type].splice(listenerIndex,1);break}}}catch(err){jwplayer.utils.log("error",err)}return false};this.addGlobalListener=function(listener,count){try{if(typeof(listener)=="string"){eval("listener = "+listener)}_globallisteners.push({listener:listener,count:count})}catch(err){jwplayer.utils.log("error",err)}return false};this.removeGlobalListener=function(listener){if(!listener){return}try{for(var globalListenerIndex=0;globalListenerIndex<_globallisteners.length;globalListenerIndex++){if(_globallisteners[globalListenerIndex].listener.toString()==listener.toString()){_globallisteners.splice(globalListenerIndex,1);break}}}catch(err){jwplayer.utils.log("error",err)}return false};this.sendEvent=function(type,data){if(!jwplayer.utils.exists(data)){data={}}if(_debug){jwplayer.utils.log(type,data)}if(typeof _listeners[type]!="undefined"){for(var listenerIndex=0;listenerIndex<_listeners[type].length;listenerIndex++){try{_listeners[type][listenerIndex].listener(data)}catch(err){jwplayer.utils.log("There was an error while handling a listener: "+err.toString(),_listeners[type][listenerIndex].listener)}if(_listeners[type][listenerIndex]){if(_listeners[type][listenerIndex].count===1){delete _listeners[type][listenerIndex]}else{if(_listeners[type][listenerIndex].count>0){_listeners[type][listenerIndex].count=_listeners[type][listenerIndex].count-1}}}}}for(var globalListenerIndex=0;globalListenerIndex<_globallisteners.length;globalListenerIndex++){try{_globallisteners[globalListenerIndex].listener(data)}catch(err){jwplayer.utils.log("There was an error while handling a listener: "+err.toString(),_globallisteners[globalListenerIndex].listener)}if(_globallisteners[globalListenerIndex]){if(_globallisteners[globalListenerIndex].count===1){delete _globallisteners[globalListenerIndex]}else{if(_globallisteners[globalListenerIndex].count>0){_globallisteners[globalListenerIndex].count=_globallisteners[globalListenerIndex].count-1}}}}}}})(jwplayer);(function(a){var b={};a.utils.animations=function(){};a.utils.animations.transform=function(c,d){c.style.webkitTransform=d;c.style.MozTransform=d;c.style.OTransform=d;c.style.msTransform=d};a.utils.animations.transformOrigin=function(c,d){c.style.webkitTransformOrigin=d;c.style.MozTransformOrigin=d;c.style.OTransformOrigin=d;c.style.msTransformOrigin=d};a.utils.animations.rotate=function(c,d){a.utils.animations.transform(c,["rotate(",d,"deg)"].join(""))};a.utils.cancelAnimation=function(c){delete b[c.id]};a.utils.fadeTo=function(m,f,e,j,h,d){if(b[m.id]!=d&&a.utils.exists(d)){return}if(m.style.opacity==f){return}var c=new Date().getTime();if(d>c){setTimeout(function(){a.utils.fadeTo(m,f,e,j,0,d)},d-c)}if(m.style.display=="none"){m.style.display="block"}if(!a.utils.exists(j)){j=m.style.opacity===""?1:m.style.opacity}if(m.style.opacity==f&&m.style.opacity!==""&&a.utils.exists(d)){if(f===0){m.style.display="none"}return}if(!a.utils.exists(d)){d=c;b[m.id]=d}if(!a.utils.exists(h)){h=0}var k=(e>0)?((c-d)/(e*1000)):0;k=k>1?1:k;var l=f-j;var g=j+(k*l);if(g>1){g=1}else{if(g<0){g=0}}m.style.opacity=g;if(h>0){b[m.id]=d+h*1000;a.utils.fadeTo(m,f,e,j,0,b[m.id]);return}setTimeout(function(){a.utils.fadeTo(m,f,e,j,0,d)},10)}})(jwplayer);(function(a){a.utils.arrays=function(){};a.utils.arrays.indexOf=function(c,d){for(var b=0;b<c.length;b++){if(c[b]==d){return b}}return -1};a.utils.arrays.remove=function(c,d){var b=a.utils.arrays.indexOf(c,d);if(b>-1){c.splice(b,1)}}})(jwplayer);(function(a){a.utils.extensionmap={"3gp":{html5:"video/3gpp",flash:"video"},"3gpp":{html5:"video/3gpp"},"3g2":{html5:"video/3gpp2",flash:"video"},"3gpp2":{html5:"video/3gpp2"},flv:{flash:"video"},f4a:{html5:"audio/mp4"},f4b:{html5:"audio/mp4",flash:"video"},f4v:{html5:"video/mp4",flash:"video"},mov:{html5:"video/quicktime",flash:"video"},m4a:{html5:"audio/mp4",flash:"video"},m4b:{html5:"audio/mp4"},m4p:{html5:"audio/mp4"},m4v:{html5:"video/mp4",flash:"video"},mp4:{html5:"video/mp4",flash:"video"},rbs:{flash:"sound"},aac:{html5:"audio/aac",flash:"video"},mp3:{html5:"audio/mp3",flash:"sound"},ogg:{html5:"audio/ogg"},oga:{html5:"audio/ogg"},ogv:{html5:"video/ogg"},webm:{html5:"video/webm"},m3u8:{html5:"audio/x-mpegurl"},gif:{flash:"image"},jpeg:{flash:"image"},jpg:{flash:"image"},swf:{flash:"image"},png:{flash:"image"},wav:{html5:"audio/x-wav"}}})(jwplayer);(function(e){e.utils.mediaparser=function(){};var g={element:{width:"width",height:"height",id:"id","class":"className",name:"name"},media:{src:"file",preload:"preload",autoplay:"autostart",loop:"repeat",controls:"controls"},source:{src:"file",type:"type",media:"media","data-jw-width":"width","data-jw-bitrate":"bitrate"},video:{poster:"image"}};var f={};e.utils.mediaparser.parseMedia=function(j){return d(j)};function c(k,j){if(!e.utils.exists(j)){j=g[k]}else{e.utils.extend(j,g[k])}return j}function d(n,j){if(f[n.tagName.toLowerCase()]&&!e.utils.exists(j)){return f[n.tagName.toLowerCase()](n)}else{j=c("element",j);var o={};for(var k in j){if(k!="length"){var m=n.getAttribute(k);if(e.utils.exists(m)){o[j[k]]=m}}}var l=n.style["#background-color"];if(l&&!(l=="transparent"||l=="rgba(0, 0, 0, 0)")){o.screencolor=l}return o}}function h(n,k){k=c("media",k);var l=[];var j=e.utils.selectors("source",n);for(var m in j){if(!isNaN(m)){l.push(a(j[m]))}}var o=d(n,k);if(e.utils.exists(o.file)){l[0]={file:o.file}}o.levels=l;return o}function a(l,k){k=c("source",k);var j=d(l,k);j.width=j.width?j.width:0;j.bitrate=j.bitrate?j.bitrate:0;return j}function b(l,k){k=c("video",k);var j=h(l,k);return j}f.media=h;f.audio=h;f.source=a;f.video=b})(jwplayer);(function(a){a.utils.loaderstatus={NEW:"NEW",LOADING:"LOADING",ERROR:"ERROR",COMPLETE:"COMPLETE"};a.utils.scriptloader=function(c){var d=a.utils.loaderstatus.NEW;var b=new a.events.eventdispatcher();a.utils.extend(this,b);this.load=function(){if(d==a.utils.loaderstatus.NEW){d=a.utils.loaderstatus.LOADING;var e=document.createElement("script");e.onload=function(f){d=a.utils.loaderstatus.COMPLETE;b.sendEvent(a.events.COMPLETE)};e.onerror=function(f){d=a.utils.loaderstatus.ERROR;b.sendEvent(a.events.ERROR)};e.onreadystatechange=function(){if(e.readyState=="loaded"||e.readyState=="complete"){d=a.utils.loaderstatus.COMPLETE;b.sendEvent(a.events.COMPLETE)}};document.getElementsByTagName("head")[0].appendChild(e);e.src=c}};this.getStatus=function(){return d}}})(jwplayer);(function(a){a.utils.selectors=function(b,e){if(!a.utils.exists(e)){e=document}b=a.utils.strings.trim(b);var c=b.charAt(0);if(c=="#"){return e.getElementById(b.substr(1))}else{if(c=="."){if(e.getElementsByClassName){return e.getElementsByClassName(b.substr(1))}else{return a.utils.selectors.getElementsByTagAndClass("*",b.substr(1))}}else{if(b.indexOf(".")>0){var d=b.split(".");return a.utils.selectors.getElementsByTagAndClass(d[0],d[1])}else{return e.getElementsByTagName(b)}}}return null};a.utils.selectors.getElementsByTagAndClass=function(e,h,g){var j=[];if(!a.utils.exists(g)){g=document}var f=g.getElementsByTagName(e);for(var d=0;d<f.length;d++){if(a.utils.exists(f[d].className)){var c=f[d].className.split(" ");for(var b=0;b<c.length;b++){if(c[b]==h){j.push(f[d])}}}}return j}})(jwplayer);(function(a){a.utils.strings=function(){};a.utils.strings.trim=function(b){return b.replace(/^\s*/,"").replace(/\s*$/,"")};a.utils.strings.pad=function(c,d,b){if(!b){b="0"}while(c.length<d){c=b+c}return c};a.utils.strings.serialize=function(b){if(b==null){return null}else{if(b=="true"){return true}else{if(b=="false"){return false}else{if(isNaN(Number(b))||b.length>5||b.length==0){return b}else{return Number(b)}}}}};a.utils.strings.seconds=function(d){d=d.replace(",",".");var b=d.split(":");var c=0;if(d.substr(-1)=="s"){c=Number(d.substr(0,d.length-1))}else{if(d.substr(-1)=="m"){c=Number(d.substr(0,d.length-1))*60}else{if(d.substr(-1)=="h"){c=Number(d.substr(0,d.length-1))*3600}else{if(b.length>1){c=Number(b[b.length-1]);c+=Number(b[b.length-2])*60;if(b.length==3){c+=Number(b[b.length-3])*3600}}else{c=Number(d)}}}}return c};a.utils.strings.xmlAttribute=function(b,c){for(var d=0;d<b.attributes.length;d++){if(b.attributes[d].name&&b.attributes[d].name.toLowerCase()==c.toLowerCase()){return b.attributes[d].value.toString()}}return""};a.utils.strings.jsonToString=function(f){var h=h||{};if(h&&h.stringify){return h.stringify(f)}var c=typeof(f);if(c!="object"||f===null){if(c=="string"){f='"'+f.replace(/"/g,'\\"')+'"'}else{return String(f)}}else{var g=[],b=(f&&f.constructor==Array);for(var d in f){var e=f[d];switch(typeof(e)){case"string":e='"'+e.replace(/"/g,'\\"')+'"';break;case"object":if(a.utils.exists(e)){e=a.utils.strings.jsonToString(e)}break}if(b){if(typeof(e)!="function"){g.push(String(e))}}else{if(typeof(e)!="function"){g.push('"'+d+'":'+String(e))}}}if(b){return"["+String(g)+"]"}else{return"{"+String(g)+"}"}}}})(jwplayer);(function(c){var d=new RegExp(/^(#|0x)[0-9a-fA-F]{3,6}/);c.utils.typechecker=function(g,f){f=!c.utils.exists(f)?b(g):f;return e(g,f)};function b(f){var g=["true","false","t","f"];if(g.toString().indexOf(f.toLowerCase().replace(" ",""))>=0){return"boolean"}else{if(d.test(f)){return"color"}else{if(!isNaN(parseInt(f,10))&&parseInt(f,10).toString().length==f.length){return"integer"}else{if(!isNaN(parseFloat(f))&&parseFloat(f).toString().length==f.length){return"float"}}}}return"string"}function e(g,f){if(!c.utils.exists(f)){return g}switch(f){case"color":if(g.length>0){return a(g)}return null;case"integer":return parseInt(g,10);case"float":return parseFloat(g);case"boolean":if(g.toLowerCase()=="true"){return true}else{if(g=="1"){return true}}return false}return g}function a(f){switch(f.toLowerCase()){case"blue":return parseInt("0000FF",16);case"green":return parseInt("00FF00",16);case"red":return parseInt("FF0000",16);case"cyan":return parseInt("00FFFF",16);case"magenta":return parseInt("FF00FF",16);case"yellow":return parseInt("FFFF00",16);case"black":return parseInt("000000",16);case"white":return parseInt("FFFFFF",16);default:f=f.replace(/(#|0x)?([0-9A-F]{3,6})$/gi,"$2");if(f.length==3){f=f.charAt(0)+f.charAt(0)+f.charAt(1)+f.charAt(1)+f.charAt(2)+f.charAt(2)}return parseInt(f,16)}return parseInt("000000",16)}})(jwplayer);(function(a){a.utils.parsers=function(){};a.utils.parsers.localName=function(b){if(!b){return""}else{if(b.localName){return b.localName}else{if(b.baseName){return b.baseName}else{return""}}}};a.utils.parsers.textContent=function(b){if(!b){return""}else{if(b.textContent){return b.textContent}else{if(b.text){return b.text}else{return""}}}}})(jwplayer);(function(a){a.utils.parsers.jwparser=function(){};a.utils.parsers.jwparser.PREFIX="jwplayer";a.utils.parsers.jwparser.parseEntry=function(c,d){for(var b=0;b<c.childNodes.length;b++){if(c.childNodes[b].prefix==a.utils.parsers.jwparser.PREFIX){d[a.utils.parsers.localName(c.childNodes[b])]=a.utils.strings.serialize(a.utils.parsers.textContent(c.childNodes[b]));if(a.utils.parsers.localName(c.childNodes[b])=="file"&&d.levels){delete d.levels}}if(!d.file&&String(d.link).toLowerCase().indexOf("youtube")>-1){d.file=d.link}}return d};a.utils.parsers.jwparser.getProvider=function(c){if(c.type){return c.type}else{if(c.file.indexOf("youtube.com/w")>-1||c.file.indexOf("youtube.com/v")>-1||c.file.indexOf("youtu.be/")>-1){return"youtube"}else{if(c.streamer&&c.streamer.indexOf("rtmp")==0){return"rtmp"}else{if(c.streamer&&c.streamer.indexOf("http")==0){return"http"}else{var b=a.utils.strings.extension(c.file);if(extensions.hasOwnProperty(b)){return extensions[b]}}}}}return""}})(jwplayer);(function(a){a.utils.parsers.mediaparser=function(){};a.utils.parsers.mediaparser.PREFIX="media";a.utils.parsers.mediaparser.parseGroup=function(d,f){var e=false;for(var c=0;c<d.childNodes.length;c++){if(d.childNodes[c].prefix==a.utils.parsers.mediaparser.PREFIX){if(!a.utils.parsers.localName(d.childNodes[c])){continue}switch(a.utils.parsers.localName(d.childNodes[c]).toLowerCase()){case"content":if(!e){f.file=a.utils.strings.xmlAttribute(d.childNodes[c],"url")}if(a.utils.strings.xmlAttribute(d.childNodes[c],"duration")){f.duration=a.utils.strings.seconds(a.utils.strings.xmlAttribute(d.childNodes[c],"duration"))}if(a.utils.strings.xmlAttribute(d.childNodes[c],"start")){f.start=a.utils.strings.seconds(a.utils.strings.xmlAttribute(d.childNodes[c],"start"))}if(d.childNodes[c].childNodes&&d.childNodes[c].childNodes.length>0){f=a.utils.parsers.mediaparser.parseGroup(d.childNodes[c],f)}if(a.utils.strings.xmlAttribute(d.childNodes[c],"width")||a.utils.strings.xmlAttribute(d.childNodes[c],"bitrate")||a.utils.strings.xmlAttribute(d.childNodes[c],"url")){if(!f.levels){f.levels=[]}f.levels.push({width:a.utils.strings.xmlAttribute(d.childNodes[c],"width"),bitrate:a.utils.strings.xmlAttribute(d.childNodes[c],"bitrate"),file:a.utils.strings.xmlAttribute(d.childNodes[c],"url")})}break;case"title":f.title=a.utils.parsers.textContent(d.childNodes[c]);break;case"description":f.description=a.utils.parsers.textContent(d.childNodes[c]);break;case"keywords":f.tags=a.utils.parsers.textContent(d.childNodes[c]);break;case"thumbnail":f.image=a.utils.strings.xmlAttribute(d.childNodes[c],"url");break;case"credit":f.author=a.utils.parsers.textContent(d.childNodes[c]);break;case"player":var b=d.childNodes[c].url;if(b.indexOf("youtube.com")>=0||b.indexOf("youtu.be")>=0){e=true;f.file=a.utils.strings.xmlAttribute(d.childNodes[c],"url")}break;case"group":a.utils.parsers.mediaparser.parseGroup(d.childNodes[c],f);break}}}return f}})(jwplayer);(function(b){b.utils.parsers.rssparser=function(){};b.utils.parsers.rssparser.parse=function(f){var c=[];for(var e=0;e<f.childNodes.length;e++){if(b.utils.parsers.localName(f.childNodes[e]).toLowerCase()=="channel"){for(var d=0;d<f.childNodes[e].childNodes.length;d++){if(b.utils.parsers.localName(f.childNodes[e].childNodes[d]).toLowerCase()=="item"){c.push(a(f.childNodes[e].childNodes[d]))}}}}return c};function a(d){var e={};for(var c=0;c<d.childNodes.length;c++){if(!b.utils.parsers.localName(d.childNodes[c])){continue}switch(b.utils.parsers.localName(d.childNodes[c]).toLowerCase()){case"enclosure":e.file=b.utils.strings.xmlAttribute(d.childNodes[c],"url");break;case"title":e.title=b.utils.parsers.textContent(d.childNodes[c]);break;case"pubdate":e.date=b.utils.parsers.textContent(d.childNodes[c]);break;case"description":e.description=b.utils.parsers.textContent(d.childNodes[c]);break;case"link":e.link=b.utils.parsers.textContent(d.childNodes[c]);break;case"category":if(e.tags){e.tags+=b.utils.parsers.textContent(d.childNodes[c])}else{e.tags=b.utils.parsers.textContent(d.childNodes[c])}break}}e=b.utils.parsers.mediaparser.parseGroup(d,e);e=b.utils.parsers.jwparser.parseEntry(d,e);return new b.html5.playlistitem(e)}})(jwplayer);(function(a){var c={};var b={};a.plugins=function(){};a.plugins.loadPlugins=function(e,d){b[e]=new a.plugins.pluginloader(new a.plugins.model(c),d);return b[e]};a.plugins.registerPlugin=function(h,f,e){var d=a.utils.getPluginName(h);if(c[d]){c[d].registerPlugin(h,f,e)}else{a.utils.log("A plugin ("+h+") was registered with the player that was not loaded. Please check your configuration.");for(var g in b){b[g].pluginFailed()}}}})(jwplayer);(function(a){a.plugins.model=function(b){this.addPlugin=function(c){var d=a.utils.getPluginName(c);if(!b[d]){b[d]=new a.plugins.plugin(c)}return b[d]}}})(jwplayer);(function(a){a.plugins.pluginmodes={FLASH:"FLASH",JAVASCRIPT:"JAVASCRIPT",HYBRID:"HYBRID"};a.plugins.plugin=function(b){var d="http://plugins.longtailvideo.com";var j=a.utils.loaderstatus.NEW;var k;var h;var l;var c=new a.events.eventdispatcher();a.utils.extend(this,c);function e(){switch(a.utils.getPluginPathType(b)){case a.utils.pluginPathType.ABSOLUTE:return b;case a.utils.pluginPathType.RELATIVE:return a.utils.getAbsolutePath(b,window.location.href);case a.utils.pluginPathType.CDN:var o=a.utils.getPluginName(b);var n=a.utils.getPluginVersion(b);var m=(window.location.href.indexOf("https://")==0)?d.replace("http://","https://secure"):d;return m+"/"+a.version.split(".")[0]+"/"+o+"/"+o+(n!==""?("-"+n):"")+".js"}}function g(m){l=setTimeout(function(){j=a.utils.loaderstatus.COMPLETE;c.sendEvent(a.events.COMPLETE)},1000)}function f(m){j=a.utils.loaderstatus.ERROR;c.sendEvent(a.events.ERROR)}this.load=function(){if(j==a.utils.loaderstatus.NEW){if(b.lastIndexOf(".swf")>0){k=b;j=a.utils.loaderstatus.COMPLETE;c.sendEvent(a.events.COMPLETE);return}j=a.utils.loaderstatus.LOADING;var m=new a.utils.scriptloader(e());m.addEventListener(a.events.COMPLETE,g);m.addEventListener(a.events.ERROR,f);m.load()}};this.registerPlugin=function(o,n,m){if(l){clearTimeout(l);l=undefined}if(n&&m){k=m;h=n}else{if(typeof n=="string"){k=n}else{if(typeof n=="function"){h=n}else{if(!n&&!m){k=o}}}}j=a.utils.loaderstatus.COMPLETE;c.sendEvent(a.events.COMPLETE)};this.getStatus=function(){return j};this.getPluginName=function(){return a.utils.getPluginName(b)};this.getFlashPath=function(){if(k){switch(a.utils.getPluginPathType(k)){case a.utils.pluginPathType.ABSOLUTE:return k;case a.utils.pluginPathType.RELATIVE:if(b.lastIndexOf(".swf")>0){return a.utils.getAbsolutePath(k,window.location.href)}return a.utils.getAbsolutePath(k,e());case a.utils.pluginPathType.CDN:if(k.indexOf("-")>-1){return k+"h"}return k+"-h"}}return null};this.getJS=function(){return h};this.getPluginmode=function(){if(typeof k!="undefined"&&typeof h!="undefined"){return a.plugins.pluginmodes.HYBRID}else{if(typeof k!="undefined"){return a.plugins.pluginmodes.FLASH}else{if(typeof h!="undefined"){return a.plugins.pluginmodes.JAVASCRIPT}}}};this.getNewInstance=function(n,m,o){return new h(n,m,o)};this.getURL=function(){return b}}})(jwplayer);(function(a){a.plugins.pluginloader=function(h,e){var g={};var k=a.utils.loaderstatus.NEW;var d=false;var b=false;var c=new a.events.eventdispatcher();a.utils.extend(this,c);function f(){if(!b){b=true;k=a.utils.loaderstatus.COMPLETE;c.sendEvent(a.events.COMPLETE)}}function j(){if(!b){var m=0;for(plugin in g){var l=g[plugin].getStatus();if(l==a.utils.loaderstatus.LOADING||l==a.utils.loaderstatus.NEW){m++}}if(m==0){f()}}}this.setupPlugins=function(n,l,s){var m={length:0,plugins:{}};var p={length:0,plugins:{}};for(var o in g){var q=g[o].getPluginName();if(g[o].getFlashPath()){m.plugins[g[o].getFlashPath()]=l.plugins[o];m.plugins[g[o].getFlashPath()].pluginmode=g[o].getPluginmode();m.length++}if(g[o].getJS()){var r=document.createElement("div");r.id=n.id+"_"+q;r.style.position="absolute";r.style.zIndex=p.length+10;p.plugins[q]=g[o].getNewInstance(n,l.plugins[o],r);p.length++;if(typeof p.plugins[q].resize!="undefined"){n.onReady(s(p.plugins[q],r,true));n.onResize(s(p.plugins[q],r))}}}n.plugins=p.plugins;return m};this.load=function(){k=a.utils.loaderstatus.LOADING;d=true;for(var l in e){if(a.utils.exists(l)){g[l]=h.addPlugin(l);g[l].addEventListener(a.events.COMPLETE,j);g[l].addEventListener(a.events.ERROR,j)}}for(l in g){g[l].load()}d=false;j()};this.pluginFailed=function(){f()};this.getStatus=function(){return k}}})(jwplayer);(function(b){var a=[];b.api=function(d){this.container=d;this.id=d.id;var m={};var s={};var p={};var c=[];var g=undefined;var k=false;var h=[];var q=undefined;var o=b.utils.getOuterHTML(d);var r={};var j={};this.getBuffer=function(){return this.callInternal("jwGetBuffer")};this.getContainer=function(){return this.container};function e(u,t){return function(z,v,w,x){if(u.renderingMode=="flash"||u.renderingMode=="html5"){var y;if(v){j[z]=v;y="jwplayer('"+u.id+"').callback('"+z+"')"}else{if(!v&&j[z]){delete j[z]}}g.jwDockSetButton(z,y,w,x)}return t}}this.getPlugin=function(t){var v=this;var u={};if(t=="dock"){return b.utils.extend(u,{setButton:e(v,u),show:function(){v.callInternal("jwDockShow");return u},hide:function(){v.callInternal("jwDockHide");return u},onShow:function(w){v.componentListener("dock",b.api.events.JWPLAYER_COMPONENT_SHOW,w);return u},onHide:function(w){v.componentListener("dock",b.api.events.JWPLAYER_COMPONENT_HIDE,w);return u}})}else{if(t=="controlbar"){return b.utils.extend(u,{show:function(){v.callInternal("jwControlbarShow");return u},hide:function(){v.callInternal("jwControlbarHide");return u},onShow:function(w){v.componentListener("controlbar",b.api.events.JWPLAYER_COMPONENT_SHOW,w);return u},onHide:function(w){v.componentListener("controlbar",b.api.events.JWPLAYER_COMPONENT_HIDE,w);return u}})}else{if(t=="display"){return b.utils.extend(u,{show:function(){v.callInternal("jwDisplayShow");return u},hide:function(){v.callInternal("jwDisplayHide");return u},onShow:function(w){v.componentListener("display",b.api.events.JWPLAYER_COMPONENT_SHOW,w);return u},onHide:function(w){v.componentListener("display",b.api.events.JWPLAYER_COMPONENT_HIDE,w);return u}})}else{return this.plugins[t]}}}};this.callback=function(t){if(j[t]){return j[t]()}};this.getDuration=function(){return this.callInternal("jwGetDuration")};this.getFullscreen=function(){return this.callInternal("jwGetFullscreen")};this.getHeight=function(){return this.callInternal("jwGetHeight")};this.getLockState=function(){return this.callInternal("jwGetLockState")};this.getMeta=function(){return this.getItemMeta()};this.getMute=function(){return this.callInternal("jwGetMute")};this.getPlaylist=function(){var u=this.callInternal("jwGetPlaylist");if(this.renderingMode=="flash"){b.utils.deepReplaceKeyName(u,["__dot__","__spc__","__dsh__"],["."," ","-"])}for(var t=0;t<u.length;t++){if(!b.utils.exists(u[t].index)){u[t].index=t}}return u};this.getPlaylistItem=function(t){if(!b.utils.exists(t)){t=this.getCurrentItem()}return this.getPlaylist()[t]};this.getPosition=function(){return this.callInternal("jwGetPosition")};this.getRenderingMode=function(){return this.renderingMode};this.getState=function(){return this.callInternal("jwGetState")};this.getVolume=function(){return this.callInternal("jwGetVolume")};this.getWidth=function(){return this.callInternal("jwGetWidth")};this.setFullscreen=function(t){if(!b.utils.exists(t)){this.callInternal("jwSetFullscreen",!this.callInternal("jwGetFullscreen"))}else{this.callInternal("jwSetFullscreen",t)}return this};this.setMute=function(t){if(!b.utils.exists(t)){this.callInternal("jwSetMute",!this.callInternal("jwGetMute"))}else{this.callInternal("jwSetMute",t)}return this};this.lock=function(){return this};this.unlock=function(){return this};this.load=function(t){this.callInternal("jwLoad",t);return this};this.playlistItem=function(t){this.callInternal("jwPlaylistItem",t);return this};this.playlistPrev=function(){this.callInternal("jwPlaylistPrev");return this};this.playlistNext=function(){this.callInternal("jwPlaylistNext");return this};this.resize=function(u,t){if(this.renderingMode=="html5"){g.jwResize(u,t)}else{this.container.width=u;this.container.height=t;var v=document.getElementById(this.id+"_wrapper");if(v){v.style.width=u+"px";v.style.height=t+"px"}}return this};this.play=function(t){if(typeof t=="undefined"){t=this.getState();if(t==b.api.events.state.PLAYING||t==b.api.events.state.BUFFERING){this.callInternal("jwPause")}else{this.callInternal("jwPlay")}}else{this.callInternal("jwPlay",t)}return this};this.pause=function(t){if(typeof t=="undefined"){t=this.getState();if(t==b.api.events.state.PLAYING||t==b.api.events.state.BUFFERING){this.callInternal("jwPause")}else{this.callInternal("jwPlay")}}else{this.callInternal("jwPause",t)}return this};this.stop=function(){this.callInternal("jwStop");return this};this.seek=function(t){this.callInternal("jwSeek",t);return this};this.setVolume=function(t){this.callInternal("jwSetVolume",t);return this};this.loadInstream=function(u,t){q=new b.api.instream(this,g,u,t);return q};this.onBufferChange=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_BUFFER,t)};this.onBufferFull=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_BUFFER_FULL,t)};this.onError=function(t){return this.eventListener(b.api.events.JWPLAYER_ERROR,t)};this.onFullscreen=function(t){return this.eventListener(b.api.events.JWPLAYER_FULLSCREEN,t)};this.onMeta=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_META,t)};this.onMute=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_MUTE,t)};this.onPlaylist=function(t){return this.eventListener(b.api.events.JWPLAYER_PLAYLIST_LOADED,t)};this.onPlaylistItem=function(t){return this.eventListener(b.api.events.JWPLAYER_PLAYLIST_ITEM,t)};this.onReady=function(t){return this.eventListener(b.api.events.API_READY,t)};this.onResize=function(t){return this.eventListener(b.api.events.JWPLAYER_RESIZE,t)};this.onComplete=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_COMPLETE,t)};this.onSeek=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_SEEK,t)};this.onTime=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_TIME,t)};this.onVolume=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_VOLUME,t)};this.onBeforePlay=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_BEFOREPLAY,t)};this.onBeforeComplete=function(t){return this.eventListener(b.api.events.JWPLAYER_MEDIA_BEFORECOMPLETE,t)};this.onBuffer=function(t){return this.stateListener(b.api.events.state.BUFFERING,t)};this.onPause=function(t){return this.stateListener(b.api.events.state.PAUSED,t)};this.onPlay=function(t){return this.stateListener(b.api.events.state.PLAYING,t)};this.onIdle=function(t){return this.stateListener(b.api.events.state.IDLE,t)};this.remove=function(){m={};h=[];if(b.utils.getOuterHTML(this.container)!=o){b.api.destroyPlayer(this.id,o)}};this.setup=function(u){if(b.embed){var t=this.id;this.remove();var v=b(t);v.config=u;return new b.embed(v)}return this};this.registerPlugin=function(v,u,t){b.plugins.registerPlugin(v,u,t)};this.setPlayer=function(t,u){g=t;this.renderingMode=u};this.stateListener=function(t,u){if(!s[t]){s[t]=[];this.eventListener(b.api.events.JWPLAYER_PLAYER_STATE,f(t))}s[t].push(u);return this};this.detachMedia=function(){if(this.renderingMode=="html5"){return this.callInternal("jwDetachMedia")}};this.attachMedia=function(){if(this.renderingMode=="html5"){return this.callInternal("jwAttachMedia")}};function f(t){return function(v){var u=v.newstate,x=v.oldstate;if(u==t){var w=s[u];if(w){for(var y=0;y<w.length;y++){if(typeof w[y]=="function"){w[y].call(this,{oldstate:x,newstate:u})}}}}}}this.componentListener=function(t,u,v){if(!p[t]){p[t]={}}if(!p[t][u]){p[t][u]=[];this.eventListener(u,l(t,u))}p[t][u].push(v);return this};function l(t,u){return function(w){if(t==w.component){var v=p[t][u];if(v){for(var x=0;x<v.length;x++){if(typeof v[x]=="function"){v[x].call(this,w)}}}}}}this.addInternalListener=function(t,u){t.jwAddEventListener(u,'function(dat) { jwplayer("'+this.id+'").dispatchEvent("'+u+'", dat); }')};this.eventListener=function(t,u){if(!m[t]){m[t]=[];if(g&&k){this.addInternalListener(g,t)}}m[t].push(u);return this};this.dispatchEvent=function(v){if(m[v]){var u=_utils.translateEventResponse(v,arguments[1]);for(var t=0;t<m[v].length;t++){if(typeof m[v][t]=="function"){m[v][t].call(this,u)}}}};this.dispatchInstreamEvent=function(t){if(q){q.dispatchEvent(t,arguments)}};this.callInternal=function(){if(k){var v=arguments[0],t=[];for(var u=1;u<arguments.length;u++){t.push(arguments[u])}if(typeof g!="undefined"&&typeof g[v]=="function"){if(t.length==2){return(g[v])(t[0],t[1])}else{if(t.length==1){return(g[v])(t[0])}else{return(g[v])()}}}return null}else{h.push(arguments)}};this.playerReady=function(u){k=true;if(!g){this.setPlayer(document.getElementById(u.id))}this.container=document.getElementById(this.id);for(var t in m){this.addInternalListener(g,t)}this.eventListener(b.api.events.JWPLAYER_PLAYLIST_ITEM,function(v){r={}});this.eventListener(b.api.events.JWPLAYER_MEDIA_META,function(v){b.utils.extend(r,v.metadata)});this.dispatchEvent(b.api.events.API_READY);while(h.length>0){this.callInternal.apply(this,h.shift())}};this.getItemMeta=function(){return r};this.getCurrentItem=function(){return this.callInternal("jwGetPlaylistIndex")};function n(v,x,w){var t=[];if(!x){x=0}if(!w){w=v.length-1}for(var u=x;u<=w;u++){t.push(v[u])}return t}return this};b.api.selectPlayer=function(d){var c;if(!b.utils.exists(d)){d=0}if(d.nodeType){c=d}else{if(typeof d=="string"){c=document.getElementById(d)}}if(c){var e=b.api.playerById(c.id);if(e){return e}else{return b.api.addPlayer(new b.api(c))}}else{if(typeof d=="number"){return b.getPlayers()[d]}}return null};b.api.events={API_READY:"jwplayerAPIReady",JWPLAYER_READY:"jwplayerReady",JWPLAYER_FULLSCREEN:"jwplayerFullscreen",JWPLAYER_RESIZE:"jwplayerResize",JWPLAYER_ERROR:"jwplayerError",JWPLAYER_MEDIA_BEFOREPLAY:"jwplayerMediaBeforePlay",JWPLAYER_MEDIA_BEFORECOMPLETE:"jwplayerMediaBeforeComplete",JWPLAYER_COMPONENT_SHOW:"jwplayerComponentShow",JWPLAYER_COMPONENT_HIDE:"jwplayerComponentHide",JWPLAYER_MEDIA_BUFFER:"jwplayerMediaBuffer",JWPLAYER_MEDIA_BUFFER_FULL:"jwplayerMediaBufferFull",JWPLAYER_MEDIA_ERROR:"jwplayerMediaError",JWPLAYER_MEDIA_LOADED:"jwplayerMediaLoaded",JWPLAYER_MEDIA_COMPLETE:"jwplayerMediaComplete",JWPLAYER_MEDIA_SEEK:"jwplayerMediaSeek",JWPLAYER_MEDIA_TIME:"jwplayerMediaTime",JWPLAYER_MEDIA_VOLUME:"jwplayerMediaVolume",JWPLAYER_MEDIA_META:"jwplayerMediaMeta",JWPLAYER_MEDIA_MUTE:"jwplayerMediaMute",JWPLAYER_PLAYER_STATE:"jwplayerPlayerState",JWPLAYER_PLAYLIST_LOADED:"jwplayerPlaylistLoaded",JWPLAYER_PLAYLIST_ITEM:"jwplayerPlaylistItem",JWPLAYER_INSTREAM_CLICK:"jwplayerInstreamClicked",JWPLAYER_INSTREAM_DESTROYED:"jwplayerInstreamDestroyed"};b.api.events.state={BUFFERING:"BUFFERING",IDLE:"IDLE",PAUSED:"PAUSED",PLAYING:"PLAYING"};b.api.playerById=function(d){for(var c=0;c<a.length;c++){if(a[c].id==d){return a[c]}}return null};b.api.addPlayer=function(c){for(var d=0;d<a.length;d++){if(a[d]==c){return c}}a.push(c);return c};b.api.destroyPlayer=function(g,d){var f=-1;for(var j=0;j<a.length;j++){if(a[j].id==g){f=j;continue}}if(f>=0){var c=document.getElementById(a[f].id);if(document.getElementById(a[f].id+"_wrapper")){c=document.getElementById(a[f].id+"_wrapper")}if(c){if(d){b.utils.setOuterHTML(c,d)}else{var h=document.createElement("div");var e=c.id;if(c.id.indexOf("_wrapper")==c.id.length-8){newID=c.id.substring(0,c.id.length-8)}h.setAttribute("id",e);c.parentNode.replaceChild(h,c)}}a.splice(f,1)}return null};b.getPlayers=function(){return a.slice(0)}})(jwplayer);var _userPlayerReady=(typeof playerReady=="function")?playerReady:undefined;playerReady=function(b){var a=jwplayer.api.playerById(b.id);if(a){a.playerReady(b)}else{jwplayer.api.selectPlayer(b.id).playerReady(b)}if(_userPlayerReady){_userPlayerReady.call(this,b)}};(function(a){a.api.instream=function(c,j,n,q){var h=c;var b=j;var g=n;var k=q;var e={};var p={};function f(){h.callInternal("jwLoadInstream",n,q)}function m(r,s){b.jwInstreamAddEventListener(s,'function(dat) { jwplayer("'+h.id+'").dispatchInstreamEvent("'+s+'", dat); }')}function d(r,s){if(!e[r]){e[r]=[];m(b,r)}e[r].push(s);return this}function o(r,s){if(!p[r]){p[r]=[];d(a.api.events.JWPLAYER_PLAYER_STATE,l(r))}p[r].push(s);return this}function l(r){return function(t){var s=t.newstate,v=t.oldstate;if(s==r){var u=p[s];if(u){for(var w=0;w<u.length;w++){if(typeof u[w]=="function"){u[w].call(this,{oldstate:v,newstate:s,type:t.type})}}}}}}this.dispatchEvent=function(u,t){if(e[u]){var s=_utils.translateEventResponse(u,t[1]);for(var r=0;r<e[u].length;r++){if(typeof e[u][r]=="function"){e[u][r].call(this,s)}}}};this.onError=function(r){return d(a.api.events.JWPLAYER_ERROR,r)};this.onFullscreen=function(r){return d(a.api.events.JWPLAYER_FULLSCREEN,r)};this.onMeta=function(r){return d(a.api.events.JWPLAYER_MEDIA_META,r)};this.onMute=function(r){return d(a.api.events.JWPLAYER_MEDIA_MUTE,r)};this.onComplete=function(r){return d(a.api.events.JWPLAYER_MEDIA_COMPLETE,r)};this.onSeek=function(r){return d(a.api.events.JWPLAYER_MEDIA_SEEK,r)};this.onTime=function(r){return d(a.api.events.JWPLAYER_MEDIA_TIME,r)};this.onVolume=function(r){return d(a.api.events.JWPLAYER_MEDIA_VOLUME,r)};this.onBuffer=function(r){return o(a.api.events.state.BUFFERING,r)};this.onPause=function(r){return o(a.api.events.state.PAUSED,r)};this.onPlay=function(r){return o(a.api.events.state.PLAYING,r)};this.onIdle=function(r){return o(a.api.events.state.IDLE,r)};this.onInstreamClick=function(r){return d(a.api.events.JWPLAYER_INSTREAM_CLICK,r)};this.onInstreamDestroyed=function(r){return d(a.api.events.JWPLAYER_INSTREAM_DESTROYED,r)};this.play=function(r){b.jwInstreamPlay(r)};this.pause=function(r){b.jwInstreamPause(r)};this.seek=function(r){b.jwInstreamSeek(r)};this.destroy=function(){b.jwInstreamDestroy()};this.getState=function(){return b.jwInstreamGetState()};this.getDuration=function(){return b.jwInstreamGetDuration()};this.getPosition=function(){return b.jwInstreamGetPosition()};f()}})(jwplayer);(function(a){var c=a.utils;a.embed=function(h){var k={width:400,height:300,components:{controlbar:{position:"over"}}};var g=c.mediaparser.parseMedia(h.container);var f=new a.embed.config(c.extend(k,g,h.config),this);var j=a.plugins.loadPlugins(h.id,f.plugins);function d(n,m){for(var l in m){if(typeof n[l]=="function"){(n[l]).call(n,m[l])}}}function e(){if(j.getStatus()==c.loaderstatus.COMPLETE){for(var n=0;n<f.modes.length;n++){if(f.modes[n].type&&a.embed[f.modes[n].type]){var p=f.modes[n].config;var t=f;if(p){t=c.extend(c.clone(f),p);var s=["file","levels","playlist"];for(var m=0;m<s.length;m++){var q=s[m];if(c.exists(p[q])){for(var l=0;l<s.length;l++){if(l!=m){var o=s[l];if(c.exists(t[o])&&!c.exists(p[o])){delete t[o]}}}}}}var r=new a.embed[f.modes[n].type](document.getElementById(h.id),f.modes[n],t,j,h);if(r.supportsConfig()){r.embed();d(h,f.events);return h}}}c.log("No suitable players found");new a.embed.logo(c.extend({hide:true},f.components.logo),"none",h.id)}}j.addEventListener(a.events.COMPLETE,e);j.addEventListener(a.events.ERROR,e);j.load();return h};function b(){if(!document.body){return setTimeout(b,15)}var d=c.selectors.getElementsByTagAndClass("video","jwplayer");for(var e=0;e<d.length;e++){var f=d[e];if(f.id==""){f.id="jwplayer_"+Math.round(Math.random()*100000)}a(f.id).setup({})}}b()})(jwplayer);(function(e){var k=e.utils;function h(m){var l=[{type:"flash",src:m?m:"/jwplayer/player.swf"},{type:"html5"},{type:"download"}];if(k.isAndroid()){l[0]=l.splice(1,1,l[0])[0]}return l}var a={players:"modes",autoplay:"autostart"};function b(o){var n=o.toLowerCase();var m=["left","right","top","bottom"];for(var l=0;l<m.length;l++){if(n==m[l]){return true}}return false}function c(m){var l=false;l=(m instanceof Array)||(typeof m=="object"&&!m.position&&!m.size);return l}function j(l){if(typeof l=="string"){if(parseInt(l).toString()==l||l.toLowerCase().indexOf("px")>-1){return parseInt(l)}}return l}var g=["playlist","dock","controlbar","logo","display"];function f(l){var o={};switch(k.typeOf(l.plugins)){case"object":for(var n in l.plugins){o[k.getPluginName(n)]=n}break;case"string":var p=l.plugins.split(",");for(var m=0;m<p.length;m++){o[k.getPluginName(p[m])]=p[m]}break}return o}function d(p,o,n,l){if(k.typeOf(p[o])!="object"){p[o]={}}var m=p[o][n];if(k.typeOf(m)!="object"){p[o][n]=m={}}if(l){if(o=="plugins"){var q=k.getPluginName(n);m[l]=p[q+"."+l];delete p[q+"."+l]}else{m[l]=p[n+"."+l];delete p[n+"."+l]}}}e.embed.deserialize=function(m){var n=f(m);for(var l in n){d(m,"plugins",n[l])}for(var q in m){if(q.indexOf(".")>-1){var p=q.split(".");var o=p[0];var q=p[1];if(k.isInArray(g,o)){d(m,"components",o,q)}else{if(n[o]){d(m,"plugins",n[o],q)}}}}return m};e.embed.config=function(l,v){var u=k.extend({},l);var s;if(c(u.playlist)){s=u.playlist;delete u.playlist}u=e.embed.deserialize(u);u.height=j(u.height);u.width=j(u.width);if(typeof u.plugins=="string"){var m=u.plugins.split(",");if(typeof u.plugins!="object"){u.plugins={}}for(var q=0;q<m.length;q++){var r=k.getPluginName(m[q]);if(typeof u[r]=="object"){u.plugins[m[q]]=u[r];delete u[r]}else{u.plugins[m[q]]={}}}}for(var t=0;t<g.length;t++){var p=g[t];if(k.exists(u[p])){if(typeof u[p]!="object"){if(!u.components[p]){u.components[p]={}}if(p=="logo"){u.components[p].file=u[p]}else{u.components[p].position=u[p]}delete u[p]}else{if(!u.components[p]){u.components[p]={}}k.extend(u.components[p],u[p]);delete u[p]}}if(typeof u[p+"size"]!="undefined"){if(!u.components[p]){u.components[p]={}}u.components[p].size=u[p+"size"];delete u[p+"size"]}}if(typeof u.icons!="undefined"){if(!u.components.display){u.components.display={}}u.components.display.icons=u.icons;delete u.icons}for(var o in a){if(u[o]){if(!u[a[o]]){u[a[o]]=u[o]}delete u[o]}}var n;if(u.flashplayer&&!u.modes){n=h(u.flashplayer);delete u.flashplayer}else{if(u.modes){if(typeof u.modes=="string"){n=h(u.modes)}else{if(u.modes instanceof Array){n=u.modes}else{if(typeof u.modes=="object"&&u.modes.type){n=[u.modes]}}}delete u.modes}else{n=h()}}u.modes=n;if(s){u.playlist=s}return u}})(jwplayer);(function(a){a.embed.download=function(c,g,b,d,f){this.embed=function(){var k=a.utils.extend({},b);var q={};var j=b.width?b.width:480;if(typeof j!="number"){j=parseInt(j,10)}var m=b.height?b.height:320;if(typeof m!="number"){m=parseInt(m,10)}var u,o,n;var s={};if(b.playlist&&b.playlist.length){s.file=b.playlist[0].file;o=b.playlist[0].image;s.levels=b.playlist[0].levels}else{s.file=b.file;o=b.image;s.levels=b.levels}if(s.file){u=s.file}else{if(s.levels&&s.levels.length){u=s.levels[0].file}}n=u?"pointer":"auto";var l={display:{style:{cursor:n,width:j,height:m,backgroundColor:"#000",position:"relative",textDecoration:"none",border:"none",display:"block"}},display_icon:{style:{cursor:n,position:"absolute",display:u?"block":"none",top:0,left:0,border:0,margin:0,padding:0,zIndex:3,width:50,height:50,backgroundImage:"url()"}},display_iconBackground:{style:{cursor:n,position:"absolute",display:u?"block":"none",top:((m-50)/2),left:((j-50)/2),border:0,width:50,height:50,margin:0,padding:0,zIndex:2,backgroundImage:"url()"}},display_image:{style:{width:j,height:m,display:o?"block":"none",position:"absolute",cursor:n,left:0,top:0,margin:0,padding:0,textDecoration:"none",zIndex:1,border:"none"}}};var h=function(v,x,y){var w=document.createElement(v);if(y){w.id=y}else{w.id=c.id+"_jwplayer_"+x}a.utils.css(w,l[x].style);return w};q.display=h("a","display",c.id);if(u){q.display.setAttribute("href",a.utils.getAbsolutePath(u))}q.display_image=h("img","display_image");q.display_image.setAttribute("alt","Click to download...");if(o){q.display_image.setAttribute("src",a.utils.getAbsolutePath(o))}if(true){q.display_icon=h("div","display_icon");q.display_iconBackground=h("div","display_iconBackground");q.display.appendChild(q.display_image);q.display_iconBackground.appendChild(q.display_icon);q.display.appendChild(q.display_iconBackground)}_css=a.utils.css;_hide=function(v){_css(v,{display:"none"})};function r(v){_imageWidth=q.display_image.naturalWidth;_imageHeight=q.display_image.naturalHeight;t()}function t(){a.utils.stretch(a.utils.stretching.UNIFORM,q.display_image,j,m,_imageWidth,_imageHeight)}q.display_image.onerror=function(v){_hide(q.display_image)};q.display_image.onload=r;c.parentNode.replaceChild(q.display,c);var p=(b.plugins&&b.plugins.logo)?b.plugins.logo:{};q.display.appendChild(new a.embed.logo(b.components.logo,"download",c.id));f.container=document.getElementById(f.id);f.setPlayer(q.display,"download")};this.supportsConfig=function(){if(b){var j=a.utils.getFirstPlaylistItemFromConfig(b);if(typeof j.file=="undefined"&&typeof j.levels=="undefined"){return true}else{if(j.file){return e(j.file,j.provider,j.playlistfile)}else{if(j.levels&&j.levels.length){for(var h=0;h<j.levels.length;h++){if(j.levels[h].file&&e(j.levels[h].file,j.provider,j.playlistfile)){return true}}}}}}else{return true}};function e(j,l,h){if(h){return false}var k=["image","sound","youtube","http"];if(l&&(k.toString().indexOf(l)>-1)){return true}if(!l||(l&&l=="video")){var m=a.utils.extension(j);if(m&&a.utils.extensionmap[m]){return true}}return false}}})(jwplayer);(function(a){a.embed.flash=function(f,g,l,e,j){function m(o,n,p){var q=document.createElement("param");q.setAttribute("name",n);q.setAttribute("value",p);o.appendChild(q)}function k(o,p,n){return function(q){if(n){document.getElementById(j.id+"_wrapper").appendChild(p)}var s=document.getElementById(j.id).getPluginConfig("display");o.resize(s.width,s.height);var r={left:s.x,top:s.y};a.utils.css(p,r)}}function d(p){if(!p){return{}}var r={};for(var o in p){var n=p[o];for(var q in n){r[o+"."+q]=n[q]}}return r}function h(q,p){if(q[p]){var s=q[p];for(var o in s){var n=s[o];if(typeof n=="string"){if(!q[o]){q[o]=n}}else{for(var r in n){if(!q[o+"."+r]){q[o+"."+r]=n[r]}}}}delete q[p]}}function b(q){if(!q){return{}}var t={},s=[];for(var n in q){var p=a.utils.getPluginName(n);var o=q[n];s.push(n);for(var r in o){t[p+"."+r]=o[r]}}t.plugins=s.join(",");return t}function c(p){var n=p.netstreambasepath?"":"netstreambasepath="+encodeURIComponent(window.location.href.split("#")[0])+"&";for(var o in p){if(typeof(p[o])=="object"){n+=o+"="+encodeURIComponent("[[JSON]]"+a.utils.strings.jsonToString(p[o]))+"&"}else{n+=o+"="+encodeURIComponent(p[o])+"&"}}return n.substring(0,n.length-1)}this.embed=function(){l.id=j.id;var A;var r=a.utils.extend({},l);var o=r.width;var y=r.height;if(f.id+"_wrapper"==f.parentNode.id){A=document.getElementById(f.id+"_wrapper")}else{A=document.createElement("div");A.id=f.id+"_wrapper";a.utils.wrap(f,A);a.utils.css(A,{position:"relative",width:o,height:y})}var p=e.setupPlugins(j,r,k);if(p.length>0){a.utils.extend(r,b(p.plugins))}else{delete r.plugins}var s=["height","width","modes","events"];for(var v=0;v<s.length;v++){delete r[s[v]]}var q="opaque";if(r.wmode){q=r.wmode}h(r,"components");h(r,"providers");if(typeof r["dock.position"]!="undefined"){if(r["dock.position"].toString().toLowerCase()=="false"){r.dock=r["dock.position"];delete r["dock.position"]}}var x=a.utils.getCookies();for(var n in x){if(typeof(r[n])=="undefined"){r[n]=x[n]}}var z="#000000";var u;if(a.utils.isIE()){var w='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" bgcolor="'+z+'" width="100%" height="100%" id="'+f.id+'" name="'+f.id+'" tabindex=0"">';w+='<param name="movie" value="'+g.src+'">';w+='<param name="allowfullscreen" value="true">';w+='<param name="allowscriptaccess" value="always">';w+='<param name="seamlesstabbing" value="true">';w+='<param name="wmode" value="'+q+'">';w+='<param name="flashvars" value="'+c(r)+'">';w+="</object>";a.utils.setOuterHTML(f,w);u=document.getElementById(f.id)}else{var t=document.createElement("object");t.setAttribute("type","application/x-shockwave-flash");t.setAttribute("data",g.src);t.setAttribute("width","100%");t.setAttribute("height","100%");t.setAttribute("bgcolor","#000000");t.setAttribute("id",f.id);t.setAttribute("name",f.id);t.setAttribute("tabindex",0);m(t,"allowfullscreen","true");m(t,"allowscriptaccess","always");m(t,"seamlesstabbing","true");m(t,"wmode",q);m(t,"flashvars",c(r));f.parentNode.replaceChild(t,f);u=t}j.container=u;j.setPlayer(u,"flash")};this.supportsConfig=function(){if(a.utils.hasFlash()){if(l){var o=a.utils.getFirstPlaylistItemFromConfig(l);if(typeof o.file=="undefined"&&typeof o.levels=="undefined"){return true}else{if(o.file){return flashCanPlay(o.file,o.provider)}else{if(o.levels&&o.levels.length){for(var n=0;n<o.levels.length;n++){if(o.levels[n].file&&flashCanPlay(o.levels[n].file,o.provider)){return true}}}}}}else{return true}}return false};flashCanPlay=function(n,p){var o=["video","http","sound","image"];if(p&&(o.toString().indexOf(p)<0)){return true}var q=a.utils.extension(n);if(!q){return true}if(a.utils.exists(a.utils.extensionmap[q])&&!a.utils.exists(a.utils.extensionmap[q].flash)){return false}return true}}})(jwplayer);(function(a){a.embed.html5=function(c,g,b,d,f){function e(j,k,h){return function(l){var m=document.getElementById(c.id+"_displayarea");if(h){m.appendChild(k)}j.resize(m.clientWidth,m.clientHeight);k.left=m.style.left;k.top=m.style.top}}this.embed=function(){if(a.html5){d.setupPlugins(f,b,e);c.innerHTML="";var j=a.utils.extend({screencolor:"0x000000"},b);var h=["plugins","modes","events"];for(var k=0;k<h.length;k++){delete j[h[k]]}if(j.levels&&!j.sources){j.sources=b.levels}if(j.skin&&j.skin.toLowerCase().indexOf(".zip")>0){j.skin=j.skin.replace(/\.zip/i,".xml")}var l=new (a.html5(c)).setup(j);f.container=document.getElementById(f.id);f.setPlayer(l,"html5")}else{return null}};this.supportsConfig=function(){if(!!a.vid.canPlayType){if(b){var j=a.utils.getFirstPlaylistItemFromConfig(b);if(typeof j.file=="undefined"&&typeof j.levels=="undefined"){return true}else{if(j.file){return html5CanPlay(a.vid,j.file,j.provider,j.playlistfile)}else{if(j.levels&&j.levels.length){for(var h=0;h<j.levels.length;h++){if(j.levels[h].file&&html5CanPlay(a.vid,j.levels[h].file,j.provider,j.playlistfile)){return true}}}}}}else{return true}}return false};html5CanPlay=function(k,j,l,h){if(h){return false}if(l&&l=="youtube"){return true}if(l&&l!="video"&&l!="http"&&l!="sound"){return false}if(navigator.userAgent.match(/BlackBerry/i)!==null){return false}var m=a.utils.extension(j);if(!a.utils.exists(m)||!a.utils.exists(a.utils.extensionmap[m])){return true}if(!a.utils.exists(a.utils.extensionmap[m].html5)){return false}if(a.utils.isLegacyAndroid()&&m.match(/m4v|mp4/)){return true}return browserCanPlay(k,a.utils.extensionmap[m].html5)};browserCanPlay=function(j,h){if(!h){return true}if(j.canPlayType(h)){return true}else{if(h=="audio/mp3"&&navigator.userAgent.match(/safari/i)){return j.canPlayType("audio/mpeg")}else{return false}}}}})(jwplayer);(function(a){a.embed.logo=function(m,l,d){var j={prefix:"http://l.longtailvideo.com/"+l+"/",file:"logo.png",link:"http://www.longtailvideo.com/players/jw-flv-player/",linktarget:"_top",margin:8,out:0.5,over:1,timeout:5,hide:false,position:"bottom-left"};_css=a.utils.css;var b;var h;k();function k(){o();c();f()}function o(){if(j.prefix){var q=a.version.split(/\W/).splice(0,2).join("/");if(j.prefix.indexOf(q)<0){j.prefix+=q+"/"}}h=a.utils.extend({},j)}function p(){var s={border:"none",textDecoration:"none",position:"absolute",cursor:"pointer",zIndex:10};s.display=h.hide?"none":"block";var r=h.position.toLowerCase().split("-");for(var q in r){s[r[q]]=h.margin}return s}function c(){b=document.createElement("img");b.id=d+"_jwplayer_logo";b.style.display="none";b.onload=function(q){_css(b,p());e()};if(!h.file){return}if(h.file.indexOf("http://")===0){b.src=h.file}else{b.src=h.prefix+h.file}}if(!h.file){return}function f(){if(h.link){b.onmouseover=g;b.onmouseout=e;b.onclick=n}else{this.mouseEnabled=false}}function n(q){if(typeof q!="undefined"){q.preventDefault();q.stopPropagation()}if(h.link){window.open(h.link,h.linktarget)}return}function e(q){if(h.link){b.style.opacity=h.out}return}function g(q){if(h.hide){b.style.opacity=h.over}return}return b}})(jwplayer);(function(a){a.html5=function(b){var c=b;this.setup=function(d){a.utils.extend(this,new a.html5.api(c,d));return this};return this}})(jwplayer);(function(a){var c=a.utils;var b=c.css;a.html5.view=function(v,u,g){var A=v;var o=u;var C=g;var B;var h;var L;var w;var M;var s;var I;var t=false;var F,r;var x,f,e;function E(){B=document.createElement("div");B.id=o.id;B.className=o.className;_videowrapper=document.createElement("div");_videowrapper.id=B.id+"_video_wrapper";o.id=B.id+"_video";b(B,{position:"relative",height:C.height,width:C.width,padding:0,backgroundColor:N(),zIndex:0});function N(){if(A.skin.getComponentSettings("display")&&A.skin.getComponentSettings("display").backgroundcolor){return A.skin.getComponentSettings("display").backgroundcolor}return parseInt("000000",16)}b(o,{width:"100%",height:"100%",top:0,left:0,zIndex:1,margin:"auto",display:"block"});b(_videowrapper,{overflow:"hidden",position:"absolute",top:0,left:0,bottom:0,right:0});c.wrap(o,B);c.wrap(o,_videowrapper);w=document.createElement("div");w.id=B.id+"_displayarea";B.appendChild(w);_instreamArea=document.createElement("div");_instreamArea.id=B.id+"_instreamarea";b(_instreamArea,{overflow:"hidden",position:"absolute",top:0,left:0,bottom:0,right:0,zIndex:100,background:"000000",display:"none"});B.appendChild(_instreamArea)}function l(){for(var N=0;N<C.plugins.order.length;N++){var O=C.plugins.order[N];if(c.exists(C.plugins.object[O].getDisplayElement)){C.plugins.object[O].height=c.parseDimension(C.plugins.object[O].getDisplayElement().style.height);C.plugins.object[O].width=c.parseDimension(C.plugins.object[O].getDisplayElement().style.width);C.plugins.config[O].currentPosition=C.plugins.config[O].position}}z()}function n(N){if(f){return}if(C.getMedia()&&C.getMedia().hasChrome()){w.style.display="none"}else{switch(N.newstate){case N.newstate==a.api.events.state.PLAYING:w.style.display="none";break;default:w.style.display="block";break}}}function z(O){var Q=C.getMedia()?C.getMedia().getDisplayElement():null;if(c.exists(Q)){if(I!=Q){if(I&&I.parentNode){I.parentNode.replaceChild(Q,I)}I=Q}for(var N=0;N<C.plugins.order.length;N++){var P=C.plugins.order[N];if(c.exists(C.plugins.object[P].getDisplayElement)){C.plugins.config[P].currentPosition=C.plugins.config[P].position}}}k(C.width,C.height)}this.setup=function(){if(C&&C.getMedia()){o=C.getMedia().getDisplayElement()}E();l();A.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,n);A.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_LOADED,z);A.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_META,function(O){D()});var N;if(c.exists(window.onresize)){N=window.onresize}window.onresize=function(O){if(c.exists(N)){try{N(O)}catch(Q){}}if(A.jwGetFullscreen()){if(!G()){var P=c.getBoundingClientRect(document.body);C.width=Math.abs(P.left)+Math.abs(P.right);C.height=window.innerHeight;k(C.width,C.height)}}else{k(C.width,C.height)}}};function j(N){switch(N.keyCode){case 27:if(A.jwGetFullscreen()){A.jwSetFullscreen(false)}break;case 32:if(A.jwGetState()!=a.api.events.state.IDLE&&A.jwGetState()!=a.api.events.state.PAUSED){A.jwPause()}else{A.jwPlay()}break}}function k(N,W){if(B.style.display=="none"){return}var Q=[].concat(C.plugins.order);Q.reverse();M=Q.length+2;if(G()){try{if(C.fullscreen&&!C.getMedia().getDisplayElement().webkitDisplayingFullscreen){C.fullscreen=false}}catch(T){}}if(!C.fullscreen){h=N;L=W;if(typeof N=="string"&&N.indexOf("%")>0){h=c.getElementWidth(c.parentNode(B))*parseInt(N.replace("%"),"")/100}else{h=N}if(typeof W=="string"&&W.indexOf("%")>0){L=c.getElementHeight(c.parentNode(B))*parseInt(W.replace("%"),"")/100}else{L=W}var R={top:0,bottom:0,left:0,right:0,width:h,height:L,position:"absolute"};b(w,R);var X={};var U;try{U=C.plugins.object.display.getDisplayElement()}catch(T){}if(U){X.width=c.parseDimension(U.style.width);X.height=c.parseDimension(U.style.height)}var V=c.extend({},R,X,{zIndex:_instreamArea.style.zIndex,display:_instreamArea.style.display});b(_instreamArea,V);b(B,{height:L,width:h});var S=q(y,Q);if(S.length>0){M+=S.length;var P=S.indexOf("playlist"),O=S.indexOf("controlbar");if(P>=0&&O>=0){S[P]=S.splice(O,1,S[P])[0]}q(m,S,true)}F=c.getElementWidth(w);r=c.getElementHeight(w)}else{if(!G()){q(K,Q,true)}}D()}function q(U,Q,R){var S=[];for(var P=0;P<Q.length;P++){var T=Q[P];if(c.exists(C.plugins.object[T].getDisplayElement)){if(C.plugins.config[T].currentPosition!=a.html5.view.positions.NONE){var N=U(T,M--);if(!N){S.push(T)}else{var O=N.width;var V=N.height;if(R){delete N.width;delete N.height}b(C.plugins.object[T].getDisplayElement(),N);C.plugins.object[T].resize(O,V)}}else{b(C.plugins.object[T].getDisplayElement(),{display:"none"})}}}return S}function y(O,P){if(c.exists(C.plugins.object[O].getDisplayElement)){if(C.plugins.config[O].position&&H(C.plugins.config[O].position)){if(!c.exists(C.plugins.object[O].getDisplayElement().parentNode)){B.appendChild(C.plugins.object[O].getDisplayElement())}var N=d(O);N.zIndex=P;return N}}return false}function m(N,O){if(!c.exists(C.plugins.object[N].getDisplayElement().parentNode)){w.appendChild(C.plugins.object[N].getDisplayElement())}return{position:"absolute",width:(c.getElementWidth(w)-c.parseDimension(w.style.left)-c.parseDimension(w.style.right)),height:(c.getElementHeight(w)-c.parseDimension(w.style.top)-c.parseDimension(w.style.bottom)),zIndex:O}}function K(N,O){return{position:"fixed",width:C.width,height:C.height,zIndex:O}}var D=this.resizeMedia=function(){w.style.position="absolute";var P=C.getMedia()?C.getMedia().getDisplayElement():e;if(!P){return}if(P&&P.tagName.toLowerCase()=="video"){if(!P.videoWidth||!P.videoHeight){return}P.style.position="absolute";c.fadeTo(P,1,0.25);if(P.parentNode){P.parentNode.style.left=w.style.left;P.parentNode.style.top=w.style.top}if(C.fullscreen&&A.jwGetStretching()==a.utils.stretching.EXACTFIT&&!c.isMobile()){var N=document.createElement("div");c.stretch(a.utils.stretching.UNIFORM,N,c.getElementWidth(w),c.getElementHeight(w),F,r);c.stretch(a.utils.stretching.EXACTFIT,P,c.parseDimension(N.style.width),c.parseDimension(N.style.height),P.videoWidth?P.videoWidth:400,P.videoHeight?P.videoHeight:300);b(P,{left:N.style.left,top:N.style.top})}else{c.stretch(A.jwGetStretching(),P,c.getElementWidth(w),c.getElementHeight(w),P.videoWidth?P.videoWidth:400,P.videoHeight?P.videoHeight:300)}}else{var O=C.plugins.object.display.getDisplayElement();if(O){C.getMedia().resize(c.parseDimension(O.style.width),c.parseDimension(O.style.height))}else{C.getMedia().resize(c.parseDimension(w.style.width),c.parseDimension(w.style.height))}}};var d=this.getComponentPosition=function(O){var P={position:"absolute",margin:0,padding:0,top:null};var N=C.plugins.config[O].currentPosition.toLowerCase();switch(N.toUpperCase()){case a.html5.view.positions.TOP:P.top=c.parseDimension(w.style.top);P.left=c.parseDimension(w.style.left);P.width=c.getElementWidth(w)-c.parseDimension(w.style.left)-c.parseDimension(w.style.right);P.height=C.plugins.object[O].height;w.style[N]=c.parseDimension(w.style[N])+C.plugins.object[O].height+"px";w.style.height=c.getElementHeight(w)-P.height+"px";break;case a.html5.view.positions.RIGHT:P.top=c.parseDimension(w.style.top);P.right=c.parseDimension(w.style.right);P.width=C.plugins.object[O].width;P.height=c.getElementHeight(w)-c.parseDimension(w.style.top)-c.parseDimension(w.style.bottom);w.style.width=c.getElementWidth(w)-P.width+"px";break;case a.html5.view.positions.BOTTOM:P.bottom=c.parseDimension(w.style.bottom);P.left=c.parseDimension(w.style.left);P.width=c.getElementWidth(w)-c.parseDimension(w.style.left)-c.parseDimension(w.style.right);P.height=C.plugins.object[O].height;w.style.height=c.getElementHeight(w)-P.height+"px";break;case a.html5.view.positions.LEFT:P.top=c.parseDimension(w.style.top);P.left=c.parseDimension(w.style.left);P.width=C.plugins.object[O].width;P.height=c.getElementHeight(w)-c.parseDimension(w.style.top)-c.parseDimension(w.style.bottom);w.style[N]=c.parseDimension(w.style[N])+C.plugins.object[O].width+"px";w.style.width=c.getElementWidth(w)-P.width+"px";break;default:break}return P};this.resize=k;var p;this.fullscreen=function(Q){var S;try{S=C.getMedia().getDisplayElement()}catch(R){}if(G()&&S&&S.webkitSupportsFullscreen){if(Q&&!S.webkitDisplayingFullscreen){try{c.transform(S);p=w.style.display;w.style.display="none";S.webkitEnterFullscreen()}catch(P){}}else{if(!Q){D();if(S.webkitDisplayingFullscreen){try{S.webkitExitFullscreen()}catch(P){}}w.style.display=p}}t=false}else{if(Q){document.onkeydown=j;clearInterval(s);var O=c.getBoundingClientRect(document.body);C.width=Math.abs(O.left)+Math.abs(O.right);C.height=window.innerHeight;var N={position:"fixed",width:"100%",height:"100%",top:0,left:0,zIndex:2147483000};b(B,N);N.zIndex=1;if(C.getMedia()&&C.getMedia().getDisplayElement()){b(C.getMedia().getDisplayElement(),N)}N.zIndex=2;b(w,N);t=true}else{document.onkeydown="";C.width=h;C.height=L;b(B,{position:"relative",height:C.height,width:C.width,zIndex:0});t=false}k(C.width,C.height)}};function H(N){return([a.html5.view.positions.TOP,a.html5.view.positions.RIGHT,a.html5.view.positions.BOTTOM,a.html5.view.positions.LEFT].toString().indexOf(N.toUpperCase())>-1)}function G(){if(A.jwGetState()!=a.api.events.state.IDLE&&!t&&(C.getMedia()&&C.getMedia().getDisplayElement()&&C.getMedia().getDisplayElement().webkitSupportsFullscreen)&&c.useNativeFullscreen()){return true}return false}this.setupInstream=function(N,O){c.css(_instreamArea,{display:"block",position:"absolute"});w.style.display="none";_instreamArea.appendChild(N);e=O;f=true};var J=this.destroyInstream=function(){_instreamArea.style.display="none";_instreamArea.innerHTML="";w.style.display="block";e=null;f=false;k(C.width,C.height)}};a.html5.view.positions={TOP:"TOP",RIGHT:"RIGHT",BOTTOM:"BOTTOM",LEFT:"LEFT",OVER:"OVER",NONE:"NONE"}})(jwplayer);(function(a){var b={backgroundcolor:"",margin:10,font:"Arial,sans-serif",fontsize:10,fontcolor:parseInt("000000",16),fontstyle:"normal",fontweight:"bold",buttoncolor:parseInt("ffffff",16),position:a.html5.view.positions.BOTTOM,idlehide:false,hideplaylistcontrols:false,forcenextprev:false,layout:{left:{position:"left",elements:[{name:"play",type:"button"},{name:"divider",type:"divider"},{name:"prev",type:"button"},{name:"divider",type:"divider"},{name:"next",type:"button"},{name:"divider",type:"divider"},{name:"elapsed",type:"text"}]},center:{position:"center",elements:[{name:"time",type:"slider"}]},right:{position:"right",elements:[{name:"duration",type:"text"},{name:"blank",type:"button"},{name:"divider",type:"divider"},{name:"mute",type:"button"},{name:"volume",type:"slider"},{name:"divider",type:"divider"},{name:"fullscreen",type:"button"}]}}};_utils=a.utils;_css=_utils.css;_hide=function(c){_css(c,{display:"none"})};_show=function(c){_css(c,{display:"block"})};a.html5.controlbar=function(m,X){window.controlbar=this;var l=m;var D=_utils.extend({},b,l.skin.getComponentSettings("controlbar"),X);if(D.position==a.html5.view.positions.NONE||typeof a.html5.view.positions[D.position]=="undefined"){return}if(_utils.mapLength(l.skin.getComponentLayout("controlbar"))>0){D.layout=l.skin.getComponentLayout("controlbar")}var af;var Q;var ae;var E;var w="none";var h;var k;var ag;var g;var f;var z;var R={};var q=false;var c={};var ab;var j=false;var p;var d;var U=false;var G=false;var H;var Z=new a.html5.eventdispatcher();_utils.extend(this,Z);function K(){if(!ab){ab=l.skin.getSkinElement("controlbar","background");if(!ab){ab={width:0,height:0,src:null}}}return ab}function O(){ae=0;E=0;Q=0;if(!q){var ao={height:K().height,backgroundColor:D.backgroundcolor};af=document.createElement("div");af.id=l.id+"_jwplayer_controlbar";_css(af,ao)}var an=(l.skin.getSkinElement("controlbar","capLeft"));var am=(l.skin.getSkinElement("controlbar","capRight"));if(an){y("capLeft","left",false,af)}ac("background",af,{position:"absolute",height:K().height,left:(an?an.width:0),zIndex:0},"img");if(K().src){R.background.src=K().src}ac("elements",af,{position:"relative",height:K().height,zIndex:1});if(am){y("capRight","right",false,af)}}this.getDisplayElement=function(){return af};this.resize=function(ao,am){S();_utils.cancelAnimation(af);f=ao;z=am;if(G!=l.jwGetFullscreen()){G=l.jwGetFullscreen();if(!G){Y()}d=undefined}var an=x();J({id:l.id,duration:ag,position:k});v({id:l.id,bufferPercent:g});return an};this.show=function(){if(j){j=false;_show(af);V()}};this.hide=function(){if(!j){j=true;_hide(af);ad()}};function r(){var an=["timeSlider","volumeSlider","timeSliderRail","volumeSliderRail"];for(var ao in an){var am=an[ao];if(typeof R[am]!="undefined"){c[am]=_utils.getBoundingClientRect(R[am])}}}var e;function Y(am){if(j){return}clearTimeout(p);if(D.position==a.html5.view.positions.OVER||l.jwGetFullscreen()){switch(l.jwGetState()){case a.api.events.state.PAUSED:case a.api.events.state.IDLE:if(af&&af.style.opacity<1&&(!D.idlehide||_utils.exists(am))){e=false;setTimeout(function(){if(!e){W()}},100)}if(D.idlehide){p=setTimeout(function(){A()},2000)}break;default:e=true;if(am){W()}p=setTimeout(function(){A()},2000);break}}else{W()}}function A(){if(!j){ad();if(af.style.opacity==1){_utils.cancelAnimation(af);_utils.fadeTo(af,0,0.1,1,0)}}}function W(){if(!j){V();if(af.style.opacity==0){_utils.cancelAnimation(af);_utils.fadeTo(af,1,0.1,0,0)}}}function I(am){return function(){if(U&&d!=am){d=am;Z.sendEvent(am,{component:"controlbar",boundingRect:P()})}}}var V=I(a.api.events.JWPLAYER_COMPONENT_SHOW);var ad=I(a.api.events.JWPLAYER_COMPONENT_HIDE);function P(){if(D.position==a.html5.view.positions.OVER||l.jwGetFullscreen()){return _utils.getDimensions(af)}else{return{x:0,y:0,width:0,height:0}}}function ac(aq,ap,ao,am){var an;if(!q){if(!am){am="div"}an=document.createElement(am);R[aq]=an;an.id=af.id+"_"+aq;ap.appendChild(an)}else{an=document.getElementById(af.id+"_"+aq)}if(_utils.exists(ao)){_css(an,ao)}return an}function N(){if(l.jwGetHeight()<=40){D.layout=_utils.clone(D.layout);for(var am=0;am<D.layout.left.elements.length;am++){if(D.layout.left.elements[am].name=="fullscreen"){D.layout.left.elements.splice(am,1)}}for(am=0;am<D.layout.right.elements.length;am++){if(D.layout.right.elements[am].name=="fullscreen"){D.layout.right.elements.splice(am,1)}}o()}al(D.layout.left);al(D.layout.center);al(D.layout.right)}function al(ap,am){var aq=ap.position=="right"?"right":"left";var ao=_utils.extend([],ap.elements);if(_utils.exists(am)){ao.reverse()}var ap=ac(ap.position+"Group",R.elements,{"float":"left",styleFloat:"left",cssFloat:"left",height:"100%"});for(var an=0;an<ao.length;an++){C(ao[an],aq,ap)}}function L(){return Q++}function C(aq,at,av){var ap,an,ao,am,aw;if(!av){av=R.elements}if(aq.type=="divider"){y("divider"+L(),at,true,av,undefined,aq.width,aq.element);return}switch(aq.name){case"play":y("playButton",at,false,av);y("pauseButton",at,true,av);T("playButton","jwPlay");T("pauseButton","jwPause");break;case"prev":y("prevButton",at,true,av);T("prevButton","jwPlaylistPrev");break;case"stop":y("stopButton",at,true,av);T("stopButton","jwStop");break;case"next":y("nextButton",at,true,av);T("nextButton","jwPlaylistNext");break;case"elapsed":y("elapsedText",at,true,av,null,null,l.skin.getSkinElement("controlbar","elapsedBackground"));break;case"time":an=!_utils.exists(l.skin.getSkinElement("controlbar","timeSliderCapLeft"))?0:l.skin.getSkinElement("controlbar","timeSliderCapLeft").width;ao=!_utils.exists(l.skin.getSkinElement("controlbar","timeSliderCapRight"))?0:l.skin.getSkinElement("controlbar","timeSliderCapRight").width;ap=at=="left"?an:ao;aw={height:K().height,position:"relative","float":"left",styleFloat:"left",cssFloat:"left"};var ar=ac("timeSlider",av,aw);y("timeSliderCapLeft",at,true,ar,"relative");y("timeSliderRail",at,false,ar,"relative");y("timeSliderBuffer",at,false,ar,"absolute");y("timeSliderProgress",at,false,ar,"absolute");y("timeSliderThumb",at,false,ar,"absolute");y("timeSliderCapRight",at,true,ar,"relative");aa("time");break;case"fullscreen":y("fullscreenButton",at,false,av);y("normalscreenButton",at,true,av);T("fullscreenButton","jwSetFullscreen",true);T("normalscreenButton","jwSetFullscreen",false);break;case"volume":an=!_utils.exists(l.skin.getSkinElement("controlbar","volumeSliderCapLeft"))?0:l.skin.getSkinElement("controlbar","volumeSliderCapLeft").width;ao=!_utils.exists(l.skin.getSkinElement("controlbar","volumeSliderCapRight"))?0:l.skin.getSkinElement("controlbar","volumeSliderCapRight").width;ap=at=="left"?an:ao;am=l.skin.getSkinElement("controlbar","volumeSliderRail").width+an+ao;aw={height:K().height,position:"relative",width:am,"float":"left",styleFloat:"left",cssFloat:"left"};var au=ac("volumeSlider",av,aw);y("volumeSliderCapLeft",at,false,au,"relative");y("volumeSliderRail",at,false,au,"relative");y("volumeSliderProgress",at,false,au,"absolute");y("volumeSliderThumb",at,false,au,"absolute");y("volumeSliderCapRight",at,false,au,"relative");aa("volume");break;case"mute":y("muteButton",at,false,av);y("unmuteButton",at,true,av);T("muteButton","jwSetMute",true);T("unmuteButton","jwSetMute",false);break;case"duration":y("durationText",at,true,av,null,null,l.skin.getSkinElement("controlbar","durationBackground"));break}}function y(ap,at,an,aw,aq,am,ao){if(_utils.exists(l.skin.getSkinElement("controlbar",ap))||ap.indexOf("Text")>0||ap.indexOf("divider")===0){var ar={height:"100%",position:aq?aq:"relative",display:"block","float":"left",styleFloat:"left",cssFloat:"left"};if((ap.indexOf("next")===0||ap.indexOf("prev")===0)&&(l.jwGetPlaylist().length<2||D.hideplaylistcontrols.toString()=="true")){if(D.forcenextprev.toString()!="true"){an=false;ar.display="none"}}var ax;if(ap.indexOf("Text")>0){ap.innerhtml="00:00";ar.font=D.fontsize+"px/"+(K().height+1)+"px "+D.font;ar.color=D.fontcolor;ar.textAlign="center";ar.fontWeight=D.fontweight;ar.fontStyle=D.fontstyle;ar.cursor="default";if(ao){ar.background="url("+ao.src+") no-repeat center";ar.backgroundSize="100% "+K().height+"px"}ar.padding="0 5px"}else{if(ap.indexOf("divider")===0){if(am){if(!isNaN(parseInt(am))){ax=parseInt(am)}}else{if(ao){var au=l.skin.getSkinElement("controlbar",ao);if(au){ar.background="url("+au.src+") repeat-x center left";ax=au.width}}else{ar.background="url("+l.skin.getSkinElement("controlbar","divider").src+") repeat-x center left";ax=l.skin.getSkinElement("controlbar","divider").width}}}else{ar.background="url("+l.skin.getSkinElement("controlbar",ap).src+") repeat-x center left";ax=l.skin.getSkinElement("controlbar",ap).width}}if(at=="left"){if(an){ae+=ax}}else{if(at=="right"){if(an){E+=ax}}}if(_utils.typeOf(aw)=="undefined"){aw=R.elements}ar.width=ax;if(q){_css(R[ap],ar)}else{var av=ac(ap,aw,ar);if(_utils.exists(l.skin.getSkinElement("controlbar",ap+"Over"))){av.onmouseover=function(ay){av.style.backgroundImage=["url(",l.skin.getSkinElement("controlbar",ap+"Over").src,")"].join("")};av.onmouseout=function(ay){av.style.backgroundImage=["url(",l.skin.getSkinElement("controlbar",ap).src,")"].join("")}}if(ap.indexOf("divider")==0){av.setAttribute("class","divider")}av.innerHTML=" "}}}function F(){l.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,B);l.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_ITEM,t);l.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_BUFFER,v);l.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,s);l.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_TIME,J);l.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_MUTE,ak);l.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_VOLUME,n);l.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_COMPLETE,M)}function B(){if(!D.hideplaylistcontrols){if(l.jwGetPlaylist().length>1||D.forcenextprev.toString()=="true"){_show(R.nextButton);_show(R.prevButton)}else{_hide(R.nextButton);_hide(R.prevButton)}x();ah()}}function t(am){ag=l.jwGetPlaylist()[am.index].duration;J({id:l.id,duration:ag,position:0});v({id:l.id,bufferProgress:0})}function ah(){J({id:l.id,duration:l.jwGetDuration(),position:0});v({id:l.id,bufferProgress:0});ak({id:l.id,mute:l.jwGetMute()});s({id:l.id,newstate:a.api.events.state.IDLE});n({id:l.id,volume:l.jwGetVolume()})}function T(ao,ap,an){if(q){return}if(_utils.exists(l.skin.getSkinElement("controlbar",ao))){var am=R[ao];if(_utils.exists(am)){_css(am,{cursor:"pointer"});if(ap=="fullscreen"){am.onmouseup=function(aq){aq.stopPropagation();l.jwSetFullscreen(!l.jwGetFullscreen())}}else{am.onmouseup=function(aq){aq.stopPropagation();if(_utils.exists(an)){l[ap](an)}else{l[ap]()}}}}}}function aa(am){if(q){return}var an=R[am+"Slider"];_css(R.elements,{cursor:"pointer"});_css(an,{cursor:"pointer"});an.onmousedown=function(ao){w=am};an.onmouseup=function(ao){ao.stopPropagation();aj(ao.pageX)};an.onmousemove=function(ao){if(w=="time"){h=true;var ap=ao.pageX-c[am+"Slider"].left-window.pageXOffset;_css(R[w+"SliderThumb"],{left:ap})}}}function aj(an){h=false;var am;if(w=="time"){am=an-c.timeSliderRail.left+window.pageXOffset;var ap=am/c.timeSliderRail.width*ag;if(ap<0){ap=0}else{if(ap>ag){ap=ag-3}}if(l.jwGetState()==a.api.events.state.PAUSED||l.jwGetState()==a.api.events.state.IDLE){l.jwPlay()}l.jwSeek(ap)}else{if(w=="volume"){am=an-c.volumeSliderRail.left-window.pageXOffset;var ao=Math.round(am/c.volumeSliderRail.width*100);if(ao<10){ao=0}else{if(ao>100){ao=100}}if(l.jwGetMute()){l.jwSetMute(false)}l.jwSetVolume(ao)}}w="none"}function v(an){if(_utils.exists(an.bufferPercent)){g=an.bufferPercent}if(c.timeSliderRail){var ap=l.skin.getSkinElement("controlbar","timeSliderCapLeft");var ao=c.timeSliderRail.width;var am=isNaN(Math.round(ao*g/100))?0:Math.round(ao*g/100);_css(R.timeSliderBuffer,{width:am,left:ap?ap.width:0})}}function ak(am){if(am.mute){_hide(R.muteButton);_show(R.unmuteButton);_hide(R.volumeSliderProgress)}else{_show(R.muteButton);_hide(R.unmuteButton);_show(R.volumeSliderProgress)}}function s(am){if(am.newstate==a.api.events.state.BUFFERING||am.newstate==a.api.events.state.PLAYING){_show(R.pauseButton);_hide(R.playButton)}else{_hide(R.pauseButton);_show(R.playButton)}Y();if(am.newstate==a.api.events.state.IDLE){_hide(R.timeSliderBuffer);_hide(R.timeSliderProgress);_hide(R.timeSliderThumb);J({id:l.id,duration:l.jwGetDuration(),position:0})}else{_show(R.timeSliderBuffer);if(am.newstate!=a.api.events.state.BUFFERING){_show(R.timeSliderProgress);_show(R.timeSliderThumb)}}}function M(am){v({bufferPercent:0});J(_utils.extend(am,{position:0,duration:ag}))}function J(ap){if(_utils.exists(ap.position)){k=ap.position}if(_utils.exists(ap.duration)){ag=ap.duration}var an=(k===ag===0)?0:k/ag;var ar=c.timeSliderRail;if(ar){var am=isNaN(Math.round(ar.width*an))?0:Math.round(ar.width*an);var aq=l.skin.getSkinElement("controlbar","timeSliderCapLeft");var ao=am+(aq?aq.width:0);if(R.timeSliderProgress){_css(R.timeSliderProgress,{width:am,left:aq?aq.width:0});if(!h){if(R.timeSliderThumb){R.timeSliderThumb.style.left=ao+"px"}}}}if(R.durationText){R.durationText.innerHTML=_utils.timeFormat(ag)}if(R.elapsedText){R.elapsedText.innerHTML=_utils.timeFormat(k)}}function o(){var am=R.elements.childNodes;var ar,ap;for(var ao=0;ao<am.length;ao++){var aq=am[ao].childNodes;for(var an in aq){if(isNaN(parseInt(an,10))){continue}if(aq[an].id.indexOf(af.id+"_divider")===0&&ap&&ap.id.indexOf(af.id+"_divider")===0&&aq[an].style.backgroundImage==ap.style.backgroundImage){aq[an].style.display="none"}else{if(aq[an].id.indexOf(af.id+"_divider")===0&&ar&&ar.style.display!="none"){aq[an].style.display="block"}}if(aq[an].style.display!="none"){ap=aq[an]}ar=aq[an]}}}function ai(){if(l.jwGetFullscreen()){_show(R.normalscreenButton);_hide(R.fullscreenButton)}else{_hide(R.normalscreenButton);_show(R.fullscreenButton)}if(l.jwGetState()==a.api.events.state.BUFFERING||l.jwGetState()==a.api.events.state.PLAYING){_show(R.pauseButton);_hide(R.playButton)}else{_hide(R.pauseButton);_show(R.playButton)}if(l.jwGetMute()==true){_hide(R.muteButton);_show(R.unmuteButton);_hide(R.volumeSliderProgress)}else{_show(R.muteButton);_hide(R.unmuteButton);_show(R.volumeSliderProgress)}}function x(){o();ai();var ao={width:f};var aw={"float":"left",styleFloat:"left",cssFloat:"left"};if(D.position==a.html5.view.positions.OVER||l.jwGetFullscreen()){ao.left=D.margin;ao.width-=2*D.margin;ao.top=z-K().height-D.margin;ao.height=K().height}var aq=l.skin.getSkinElement("controlbar","capLeft");var au=l.skin.getSkinElement("controlbar","capRight");aw.width=ao.width-(aq?aq.width:0)-(au?au.width:0);var ap=_utils.getBoundingClientRect(R.leftGroup).width;var at=_utils.getBoundingClientRect(R.rightGroup).width;var ar=aw.width-ap-at-1;var an=ar;var am=l.skin.getSkinElement("controlbar","timeSliderCapLeft");var av=l.skin.getSkinElement("controlbar","timeSliderCapRight");if(_utils.exists(am)){an-=am.width}if(_utils.exists(av)){an-=av.width}R.timeSlider.style.width=ar+"px";R.timeSliderRail.style.width=an+"px";_css(af,ao);_css(R.elements,aw);_css(R.background,aw);r();return ao}function n(ar){if(_utils.exists(R.volumeSliderRail)){var ao=isNaN(ar.volume/100)?1:ar.volume/100;var ap=_utils.parseDimension(R.volumeSliderRail.style.width);var am=isNaN(Math.round(ap*ao))?0:Math.round(ap*ao);var at=_utils.parseDimension(R.volumeSliderRail.style.right);var an=(!_utils.exists(l.skin.getSkinElement("controlbar","volumeSliderCapLeft")))?0:l.skin.getSkinElement("controlbar","volumeSliderCapLeft").width;_css(R.volumeSliderProgress,{width:am,left:an});if(R.volumeSliderThumb){var aq=(am-Math.round(_utils.parseDimension(R.volumeSliderThumb.style.width)/2));aq=Math.min(Math.max(aq,0),ap-_utils.parseDimension(R.volumeSliderThumb.style.width));_css(R.volumeSliderThumb,{left:aq})}if(_utils.exists(R.volumeSliderCapLeft)){_css(R.volumeSliderCapLeft,{left:0})}}}function S(){try{var an=(l.id.indexOf("_instream")>0?l.id.replace("_instream",""):l.id);H=document.getElementById(an);H.addEventListener("mousemove",Y)}catch(am){_utils.log("Could not add mouse listeners to controlbar: "+am)}}function u(){O();N();r();q=true;F();D.idlehide=(D.idlehide.toString().toLowerCase()=="true");if(D.position==a.html5.view.positions.OVER&&D.idlehide){af.style.opacity=0;U=true}else{af.style.opacity=1;setTimeout((function(){U=true;V()}),1)}S();ah()}u();return this}})(jwplayer);(function(b){var a=["width","height","state","playlist","item","position","buffer","duration","volume","mute","fullscreen"];var c=b.utils;b.html5.controller=function(o,K,f,h){var n=o,m=f,j=h,y=K,M=true,G=-1,A=false,d=false,P,C=[],q=false;var D=(c.exists(m.config.debug)&&(m.config.debug.toString().toLowerCase()=="console")),N=new b.html5.eventdispatcher(y.id,D);c.extend(this,N);function L(T){if(q){N.sendEvent(T.type,T)}else{C.push(T)}}function s(T){if(!q){q=true;N.sendEvent(b.api.events.JWPLAYER_READY,T);if(b.utils.exists(window.playerReady)){playerReady(T)}if(b.utils.exists(window[f.config.playerReady])){window[f.config.playerReady](T)}while(C.length>0){var V=C.shift();N.sendEvent(V.type,V)}if(f.config.autostart&&!b.utils.isIOS()){O()}while(x.length>0){var U=x.shift();B(U.method,U.arguments)}}}m.addGlobalListener(L);m.addEventListener(b.api.events.JWPLAYER_MEDIA_BUFFER_FULL,function(){m.getMedia().play()});m.addEventListener(b.api.events.JWPLAYER_MEDIA_TIME,function(T){if(T.position>=m.playlist[m.item].start&&G>=0){m.playlist[m.item].start=G;G=-1}});m.addEventListener(b.api.events.JWPLAYER_MEDIA_COMPLETE,function(T){setTimeout(E,25)});m.addEventListener(b.api.events.JWPLAYER_PLAYLIST_LOADED,O);m.addEventListener(b.api.events.JWPLAYER_FULLSCREEN,p);function F(){try{P=F;if(!A){A=true;N.sendEvent(b.api.events.JWPLAYER_MEDIA_BEFOREPLAY);A=false;if(d){d=false;P=null;return}}v(m.item);if(m.playlist[m.item].levels[0].file.length>0){if(M||m.state==b.api.events.state.IDLE){m.getMedia().load(m.playlist[m.item]);M=false}else{if(m.state==b.api.events.state.PAUSED){m.getMedia().play()}}}return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T);P=null}return false}function e(){try{if(m.playlist[m.item].levels[0].file.length>0){switch(m.state){case b.api.events.state.PLAYING:case b.api.events.state.BUFFERING:if(m.getMedia()){m.getMedia().pause()}break;default:if(A){d=true}}}return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function z(T){try{if(m.playlist[m.item].levels[0].file.length>0){if(typeof T!="number"){T=parseFloat(T)}switch(m.state){case b.api.events.state.IDLE:if(G<0){G=m.playlist[m.item].start;m.playlist[m.item].start=T}if(!A){F()}break;case b.api.events.state.PLAYING:case b.api.events.state.PAUSED:case b.api.events.state.BUFFERING:m.seek(T);break}}return true}catch(U){N.sendEvent(b.api.events.JWPLAYER_ERROR,U)}return false}function w(T){P=null;if(!c.exists(T)){T=true}try{if((m.state!=b.api.events.state.IDLE||T)&&m.getMedia()){m.getMedia().stop(T)}if(A){d=true}return true}catch(U){N.sendEvent(b.api.events.JWPLAYER_ERROR,U)}return false}function k(){try{if(m.playlist[m.item].levels[0].file.length>0){if(m.config.shuffle){v(S())}else{if(m.item+1==m.playlist.length){v(0)}else{v(m.item+1)}}}if(m.state!=b.api.events.state.IDLE){var U=m.state;m.state=b.api.events.state.IDLE;N.sendEvent(b.api.events.JWPLAYER_PLAYER_STATE,{oldstate:U,newstate:b.api.events.state.IDLE})}F();return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function I(){try{if(m.playlist[m.item].levels[0].file.length>0){if(m.config.shuffle){v(S())}else{if(m.item===0){v(m.playlist.length-1)}else{v(m.item-1)}}}if(m.state!=b.api.events.state.IDLE){var U=m.state;m.state=b.api.events.state.IDLE;N.sendEvent(b.api.events.JWPLAYER_PLAYER_STATE,{oldstate:U,newstate:b.api.events.state.IDLE})}F();return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function S(){var T=null;if(m.playlist.length>1){while(!c.exists(T)){T=Math.floor(Math.random()*m.playlist.length);if(T==m.item){T=null}}}else{T=0}return T}function H(U){if(!m.playlist||!m.playlist[U]){return false}try{if(m.playlist[U].levels[0].file.length>0){var V=m.state;if(V!==b.api.events.state.IDLE){if(m.playlist[m.item]&&m.playlist[m.item].provider==m.playlist[U].provider){w(false)}else{w()}}v(U);F()}return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function v(T){if(!m.playlist[T]){return}m.setActiveMediaProvider(m.playlist[T]);if(m.item!=T){m.item=T;M=true;N.sendEvent(b.api.events.JWPLAYER_PLAYLIST_ITEM,{index:T})}}function g(U){try{v(m.item);var V=m.getMedia();switch(typeof(U)){case"number":V.volume(U);break;case"string":V.volume(parseInt(U,10));break}m.setVolume(U);return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function r(U){try{v(m.item);var V=m.getMedia();if(typeof U=="undefined"){V.mute(!m.mute);m.setMute(!m.mute)}else{if(U.toString().toLowerCase()=="true"){V.mute(true);m.setMute(true)}else{V.mute(false);m.setMute(false)}}return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function J(U,T){try{m.width=U;m.height=T;j.resize(U,T);N.sendEvent(b.api.events.JWPLAYER_RESIZE,{width:m.width,height:m.height});return true}catch(V){N.sendEvent(b.api.events.JWPLAYER_ERROR,V)}return false}function u(U,V){try{if(typeof U=="undefined"){U=!m.fullscreen}if(typeof V=="undefined"){V=true}if(U!=m.fullscreen){m.fullscreen=(U.toString().toLowerCase()=="true");j.fullscreen(m.fullscreen);if(V){N.sendEvent(b.api.events.JWPLAYER_FULLSCREEN,{fullscreen:m.fullscreen})}N.sendEvent(b.api.events.JWPLAYER_RESIZE,{width:m.width,height:m.height})}return true}catch(T){N.sendEvent(b.api.events.JWPLAYER_ERROR,T)}return false}function R(T){try{w();if(A){d=false}m.loadPlaylist(T);if(m.playlist[m.item].provider){v(m.item);if(m.config.autostart.toString().toLowerCase()=="true"&&!c.isIOS()&&!A){F()}return true}else{return false}}catch(U){N.sendEvent(b.api.events.JWPLAYER_ERROR,U)}return false}function O(T){if(!c.isIOS()){v(m.item);if(m.config.autostart.toString().toLowerCase()=="true"&&!c.isIOS()){F()}}}function p(T){u(T.fullscreen,false)}function t(){try{return m.getMedia().detachMedia()}catch(T){return null}}function l(){try{var T=m.getMedia().attachMedia();if(typeof P=="function"){P()}}catch(U){return null}}b.html5.controller.repeatoptions={LIST:"LIST",ALWAYS:"ALWAYS",SINGLE:"SINGLE",NONE:"NONE"};function E(){if(m.state!=b.api.events.state.IDLE){return}P=E;switch(m.config.repeat.toUpperCase()){case b.html5.controller.repeatoptions.SINGLE:F();break;case b.html5.controller.repeatoptions.ALWAYS:if(m.item==m.playlist.length-1&&!m.config.shuffle){H(0)}else{k()}break;case b.html5.controller.repeatoptions.LIST:if(m.item==m.playlist.length-1&&!m.config.shuffle){w();v(0)}else{k()}break;default:w();break}}var x=[];function Q(T){return function(){if(q){B(T,arguments)}else{x.push({method:T,arguments:arguments})}}}function B(V,U){var T=[];for(i=0;i<U.length;i++){T.push(U[i])}V.apply(this,T)}this.play=Q(F);this.pause=Q(e);this.seek=Q(z);this.stop=Q(w);this.next=Q(k);this.prev=Q(I);this.item=Q(H);this.setVolume=Q(g);this.setMute=Q(r);this.resize=Q(J);this.setFullscreen=Q(u);this.load=Q(R);this.playerReady=s;this.detachMedia=t;this.attachMedia=l;this.beforePlay=function(){return A}}})(jwplayer);(function(a){a.html5.defaultSkin=function(){this.text='<?xml version="1.0" ?><skin author="LongTail Video" name="Five" version="1.1"><components><component name="controlbar"><settings><setting name="margin" value="20"/><setting name="fontsize" value="11"/><setting name="fontcolor" value="0x000000"/></settings><layout><group position="left"><button name="play"/><divider name="divider"/><button name="prev"/><divider name="divider"/><button name="next"/><divider name="divider"/><text name="elapsed"/></group><group position="center"><slider name="time"/></group><group position="right"><text name="duration"/><divider name="divider"/><button name="blank"/><divider name="divider"/><button name="mute"/><slider name="volume"/><divider name="divider"/><button name="fullscreen"/></group></layout><elements><element name="background" src=""/><element name="blankButton" src=""/><element name="capLeft" src=""/><element name="capRight" src=""/><element name="divider" src=""/><element name="playButton" src=""/><element name="pauseButton" src=""/><element name="prevButton" src=""/><element name="nextButton" src=""/><element name="timeSliderRail" src=""/><element name="timeSliderBuffer" src=""/><element name="timeSliderProgress" src=""/><element name="timeSliderThumb" src=""/><element name="muteButton" src=""/><element name="unmuteButton" src=""/><element name="volumeSliderRail" src=""/><element name="volumeSliderProgress" src=""/><element name="volumeSliderCapRight" src=""/><element name="fullscreenButton" src=""/><element name="normalscreenButton" src=""/></elements></component><component name="display"><elements><element name="background" src=""/><element name="playIcon" src=""/><element name="muteIcon" src=""/><element name="errorIcon" src=""/><element name="bufferIcon" src=""/></elements></component><component name="dock"><settings><setting name="fontcolor" value="0xffffff"/></settings><elements><element name="button" src=""/></elements></component><component name="playlist"><settings><setting name="backgroundcolor" value="0xe8e8e8"/></settings><elements><element name="item" src=""/><element name="sliderCapTop" src=""/><element name="sliderRail" src=""/><element name="sliderThumb" src=""/><element name="sliderCapBottom" src=""/></elements></component></components></skin>';this.xml=null;if(window.DOMParser){parser=new DOMParser();this.xml=parser.parseFromString(this.text,"text/xml")}else{this.xml=new ActiveXObject("Microsoft.XMLDOM");this.xml.async="false";this.xml.loadXML(this.text)}return this}})(jwplayer);(function(a){_utils=a.utils;_css=_utils.css;_hide=function(b){_css(b,{display:"none"})};_show=function(b){_css(b,{display:"block"})};a.html5.display=function(k,K){var j={icons:true,showmute:false};var X=_utils.extend({},j,K);var h=k;var W={};var e;var w;var z;var T;var u;var M;var E;var N=!_utils.exists(h.skin.getComponentSettings("display").bufferrotation)?15:parseInt(h.skin.getComponentSettings("display").bufferrotation,10);var s=!_utils.exists(h.skin.getComponentSettings("display").bufferinterval)?100:parseInt(h.skin.getComponentSettings("display").bufferinterval,10);var D=-1;var v=a.api.events.state.IDLE;var O=true;var d;var C=false,V=true;var p="";var g=false;var o=false;var m;var y,R;var L=new a.html5.eventdispatcher();_utils.extend(this,L);var H={display:{style:{cursor:"pointer",top:0,left:0,overflow:"hidden"},click:n},display_icon:{style:{cursor:"pointer",position:"absolute",top:((h.skin.getSkinElement("display","background").height-h.skin.getSkinElement("display","playIcon").height)/2),left:((h.skin.getSkinElement("display","background").width-h.skin.getSkinElement("display","playIcon").width)/2),border:0,margin:0,padding:0,zIndex:3,display:"none"}},display_iconBackground:{style:{cursor:"pointer",position:"absolute",top:((w-h.skin.getSkinElement("display","background").height)/2),left:((e-h.skin.getSkinElement("display","background").width)/2),border:0,backgroundImage:(["url(",h.skin.getSkinElement("display","background").src,")"]).join(""),width:h.skin.getSkinElement("display","background").width,height:h.skin.getSkinElement("display","background").height,margin:0,padding:0,zIndex:2,display:"none"}},display_image:{style:{display:"none",width:e,height:w,position:"absolute",cursor:"pointer",left:0,top:0,margin:0,padding:0,textDecoration:"none",zIndex:1}},display_text:{style:{zIndex:4,position:"relative",opacity:0.8,backgroundColor:parseInt("000000",16),color:parseInt("ffffff",16),textAlign:"center",fontFamily:"Arial,sans-serif",padding:"0 5px",fontSize:14}}};h.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,q);h.jwAddEventListener(a.api.events.JWPLAYER_MEDIA_MUTE,q);h.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,P);h.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_ITEM,q);h.jwAddEventListener(a.api.events.JWPLAYER_ERROR,r);Q();function Q(){W.display=G("div","display");W.display_text=G("div","display_text");W.display.appendChild(W.display_text);W.display_image=G("img","display_image");W.display_image.onerror=function(Y){_hide(W.display_image)};W.display_image.onload=B;W.display_icon=G("div","display_icon");W.display_iconBackground=G("div","display_iconBackground");W.display.appendChild(W.display_image);W.display_iconBackground.appendChild(W.display_icon);W.display.appendChild(W.display_iconBackground);f();setTimeout((function(){o=true;if(X.icons.toString()=="true"){J()}}),1)}this.getDisplayElement=function(){return W.display};this.resize=function(Z,Y){if(h.jwGetFullscreen()&&_utils.useNativeFullscreen()){return}_css(W.display,{width:Z,height:Y});_css(W.display_text,{width:(Z-10),top:((Y-_utils.getBoundingClientRect(W.display_text).height)/2)});_css(W.display_iconBackground,{top:((Y-h.skin.getSkinElement("display","background").height)/2),left:((Z-h.skin.getSkinElement("display","background").width)/2)});if(e!=Z||w!=Y){e=Z;w=Y;d=undefined;J()}if(!h.jwGetFullscreen()){y=Z;R=Y}c();q({})};this.show=function(){if(g){g=false;t(h.jwGetState())}};this.hide=function(){if(!g){F();g=true}};function B(Y){z=W.display_image.naturalWidth;T=W.display_image.naturalHeight;c();if(h.jwGetState()==a.api.events.state.IDLE){_css(W.display_image,{display:"block",opacity:0});_utils.fadeTo(W.display_image,1,0.1)}C=false}function c(){if(h.jwGetFullscreen()&&h.jwGetStretching()==a.utils.stretching.EXACTFIT){var Y=document.createElement("div");_utils.stretch(a.utils.stretching.UNIFORM,Y,e,w,y,R);_utils.stretch(a.utils.stretching.EXACTFIT,W.display_image,_utils.parseDimension(Y.style.width),_utils.parseDimension(Y.style.height),z,T);_css(W.display_image,{left:Y.style.left,top:Y.style.top})}else{_utils.stretch(h.jwGetStretching(),W.display_image,e,w,z,T)}}function G(Y,aa){var Z=document.createElement(Y);Z.id=h.id+"_jwplayer_"+aa;_css(Z,H[aa].style);return Z}function f(){for(var Y in W){if(_utils.exists(H[Y].click)){W[Y].onclick=H[Y].click}}}function n(Y){if(typeof Y.preventDefault!="undefined"){Y.preventDefault()}else{Y.returnValue=false}if(typeof m=="function"){m(Y);return}else{if(h.jwGetState()!=a.api.events.state.PLAYING){h.jwPlay()}else{h.jwPause()}}}function U(Y){if(E){F();return}W.display_icon.style.backgroundImage=(["url(",h.skin.getSkinElement("display",Y).src,")"]).join("");_css(W.display_icon,{width:h.skin.getSkinElement("display",Y).width,height:h.skin.getSkinElement("display",Y).height,top:(h.skin.getSkinElement("display","background").height-h.skin.getSkinElement("display",Y).height)/2,left:(h.skin.getSkinElement("display","background").width-h.skin.getSkinElement("display",Y).width)/2});b();if(_utils.exists(h.skin.getSkinElement("display",Y+"Over"))){W.display_icon.onmouseover=function(Z){W.display_icon.style.backgroundImage=["url(",h.skin.getSkinElement("display",Y+"Over").src,")"].join("")};W.display_icon.onmouseout=function(Z){W.display_icon.style.backgroundImage=["url(",h.skin.getSkinElement("display",Y).src,")"].join("")}}else{W.display_icon.onmouseover=null;W.display_icon.onmouseout=null}}function F(){if(X.icons.toString()=="true"){_hide(W.display_icon);_hide(W.display_iconBackground);S()}}function b(){if(!g&&X.icons.toString()=="true"){_show(W.display_icon);_show(W.display_iconBackground);J()}}function r(Y){E=true;F();W.display_text.innerHTML=Y.message;_show(W.display_text);W.display_text.style.top=((w-_utils.getBoundingClientRect(W.display_text).height)/2)+"px"}function I(){V=false;W.display_image.style.display="none"}function P(){v=""}function q(Y){if((Y.type==a.api.events.JWPLAYER_PLAYER_STATE||Y.type==a.api.events.JWPLAYER_PLAYLIST_ITEM)&&E){E=false;_hide(W.display_text)}var Z=h.jwGetState();if(Z==v){return}v=Z;if(D>=0){clearTimeout(D)}if(O||h.jwGetState()==a.api.events.state.PLAYING||h.jwGetState()==a.api.events.state.PAUSED){t(h.jwGetState())}else{D=setTimeout(l(h.jwGetState()),500)}}function l(Y){return(function(){t(Y)})}function t(Y){if(_utils.exists(M)){clearInterval(M);M=null;_utils.animations.rotate(W.display_icon,0)}switch(Y){case a.api.events.state.BUFFERING:if(_utils.isIPod()){I();F()}else{if(h.jwGetPlaylist()[h.jwGetPlaylistIndex()].provider=="sound"){x()}u=0;M=setInterval(function(){u+=N;_utils.animations.rotate(W.display_icon,u%360)},s);U("bufferIcon");O=true}break;case a.api.events.state.PAUSED:if(!_utils.isIPod()){if(h.jwGetPlaylist()[h.jwGetPlaylistIndex()].provider!="sound"){_css(W.display_image,{background:"transparent no-repeat center center"})}U("playIcon");O=true}break;case a.api.events.state.IDLE:if(h.jwGetPlaylist()[h.jwGetPlaylistIndex()]&&h.jwGetPlaylist()[h.jwGetPlaylistIndex()].image){x()}else{I()}U("playIcon");O=true;break;default:if(h.jwGetPlaylist()[h.jwGetPlaylistIndex()]&&h.jwGetPlaylist()[h.jwGetPlaylistIndex()].provider=="sound"){if(_utils.isIPod()){I();O=false}else{x()}}else{I();O=false}if(h.jwGetMute()&&X.showmute){U("muteIcon")}else{F()}break}D=-1}function x(){if(h.jwGetPlaylist()[h.jwGetPlaylistIndex()]){var Y=h.jwGetPlaylist()[h.jwGetPlaylistIndex()].image;if(Y){if(Y!=p){p=Y;W.display_image.style.display="none";C=true;W.display_image.src=_utils.getAbsolutePath(Y)}else{if(!(C||V)){V=true;W.display_image.style.opacity=0;W.display_image.style.display="block";_utils.fadeTo(W.display_image,1,0.1)}}}}}function A(Y){return function(){if(!o){return}if(!g&&d!=Y){d=Y;L.sendEvent(Y,{component:"display",boundingRect:_utils.getDimensions(W.display_iconBackground)})}}}var J=A(a.api.events.JWPLAYER_COMPONENT_SHOW);var S=A(a.api.events.JWPLAYER_COMPONENT_HIDE);this.setAlternateClickHandler=function(Y){m=Y};this.revertAlternateClickHandler=function(){m=undefined};return this}})(jwplayer);(function(a){var c=a.utils;var b=c.css;a.html5.dock=function(w,D){function x(){return{align:a.html5.view.positions.RIGHT}}var n=c.extend({},x(),D);if(n.align=="FALSE"){return}var j={};var A=[];var k;var F;var f=false;var C=false;var g={x:0,y:0,width:0,height:0};var z;var o;var y;var m=new a.html5.eventdispatcher();c.extend(this,m);var r=document.createElement("div");r.id=w.id+"_jwplayer_dock";r.style.opacity=1;p();w.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,q);this.getDisplayElement=function(){return r};this.setButton=function(K,H,I,J){if(!H&&j[K]){c.arrays.remove(A,K);r.removeChild(j[K].div);delete j[K]}else{if(H){if(!j[K]){j[K]={}}j[K].handler=H;j[K].outGraphic=I;j[K].overGraphic=J;if(!j[K].div){A.push(K);j[K].div=document.createElement("div");j[K].div.style.position="absolute";r.appendChild(j[K].div);j[K].div.appendChild(document.createElement("div"));j[K].div.childNodes[0].style.position="relative";j[K].div.childNodes[0].style.width="100%";j[K].div.childNodes[0].style.height="100%";j[K].div.childNodes[0].style.zIndex=10;j[K].div.childNodes[0].style.cursor="pointer";j[K].div.appendChild(document.createElement("img"));j[K].div.childNodes[1].style.position="absolute";j[K].div.childNodes[1].style.left=0;j[K].div.childNodes[1].style.top=0;if(w.skin.getSkinElement("dock","button")){j[K].div.childNodes[1].src=w.skin.getSkinElement("dock","button").src}j[K].div.childNodes[1].style.zIndex=9;j[K].div.childNodes[1].style.cursor="pointer";j[K].div.onmouseover=function(){if(j[K].overGraphic){j[K].div.childNodes[0].style.background=h(j[K].overGraphic)}if(w.skin.getSkinElement("dock","buttonOver")){j[K].div.childNodes[1].src=w.skin.getSkinElement("dock","buttonOver").src}};j[K].div.onmouseout=function(){if(j[K].outGraphic){j[K].div.childNodes[0].style.background=h(j[K].outGraphic)}if(w.skin.getSkinElement("dock","button")){j[K].div.childNodes[1].src=w.skin.getSkinElement("dock","button").src}};if(w.skin.getSkinElement("dock","button")){j[K].div.childNodes[1].src=w.skin.getSkinElement("dock","button").src}}if(j[K].outGraphic){j[K].div.childNodes[0].style.background=h(j[K].outGraphic)}else{if(j[K].overGraphic){j[K].div.childNodes[0].style.background=h(j[K].overGraphic)}}if(H){j[K].div.onclick=function(L){L.preventDefault();a(w.id).callback(K);if(j[K].overGraphic){j[K].div.childNodes[0].style.background=h(j[K].overGraphic)}if(w.skin.getSkinElement("dock","button")){j[K].div.childNodes[1].src=w.skin.getSkinElement("dock","button").src}}}}}l(k,F)};function h(H){return"url("+H+") no-repeat center center"}function t(H){}function l(H,T){p();if(A.length>0){var I=10;var S=I;var P=-1;var Q=w.skin.getSkinElement("dock","button").height;var O=w.skin.getSkinElement("dock","button").width;var M=H-O-I;var R,L;if(n.align==a.html5.view.positions.LEFT){P=1;M=I}for(var J=0;J<A.length;J++){var U=Math.floor(S/T);if((S+Q+I)>((U+1)*T)){S=((U+1)*T)+I;U=Math.floor(S/T)}var K=j[A[J]].div;K.style.top=(S%T)+"px";K.style.left=(M+(w.skin.getSkinElement("dock","button").width+I)*U*P)+"px";var N={x:c.parseDimension(K.style.left),y:c.parseDimension(K.style.top),width:O,height:Q};if(!R||(N.x<=R.x&&N.y<=R.y)){R=N}if(!L||(N.x>=L.x&&N.y>=L.y)){L=N}K.style.width=O+"px";K.style.height=Q+"px";S+=w.skin.getSkinElement("dock","button").height+I}g={x:R.x,y:R.y,width:L.x-R.x+L.width,height:R.y-L.y+L.height}}if(C!=w.jwGetFullscreen()||k!=H||F!=T){k=H;F=T;C=w.jwGetFullscreen();z=undefined;setTimeout(s,1)}}function d(H){return function(){if(!f&&z!=H&&A.length>0){z=H;m.sendEvent(H,{component:"dock",boundingRect:g})}}}function q(H){if(c.isMobile()){if(H.newstate==a.api.events.state.IDLE){v()}else{e()}}else{B()}}function B(H){if(f){return}clearTimeout(y);if(D.position==a.html5.view.positions.OVER||w.jwGetFullscreen()){switch(w.jwGetState()){case a.api.events.state.PAUSED:case a.api.events.state.IDLE:if(r&&r.style.opacity<1&&(!D.idlehide||c.exists(H))){E()}if(D.idlehide){y=setTimeout(function(){u()},2000)}break;default:if(c.exists(H)){E()}y=setTimeout(function(){u()},2000);break}}else{E()}}var s=d(a.api.events.JWPLAYER_COMPONENT_SHOW);var G=d(a.api.events.JWPLAYER_COMPONENT_HIDE);this.resize=l;var v=function(){b(r,{display:"block"});if(f){f=false;s()}};var e=function(){b(r,{display:"none"});if(!f){G();f=true}};function u(){if(!f){G();if(r.style.opacity==1){c.cancelAnimation(r);c.fadeTo(r,0,0.1,1,0)}}}function E(){if(!f){s();if(r.style.opacity==0){c.cancelAnimation(r);c.fadeTo(r,1,0.1,0,0)}}}function p(){try{o=document.getElementById(w.id);o.addEventListener("mousemove",B)}catch(H){c.log("Could not add mouse listeners to dock: "+H)}}this.hide=e;this.show=v;return this}})(jwplayer);(function(a){a.html5.eventdispatcher=function(d,b){var c=new a.events.eventdispatcher(b);a.utils.extend(this,c);this.sendEvent=function(e,f){if(!a.utils.exists(f)){f={}}a.utils.extend(f,{id:d,version:a.version,type:e});c.sendEvent(e,f)}}})(jwplayer);(function(a){var b=a.utils;a.html5.instream=function(y,m,x,z){var t={controlbarseekable:"always",controlbarpausable:true,controlbarstoppable:true,playlistclickable:true};var v,A,C=y,E=m,j=x,w=z,r,H,o,G,e,f,g,l,q,h=false,k,d,n=this;this.load=function(M,K){c();h=true;A=b.extend(t,K);v=a.html5.playlistitem(M);F();d=document.createElement("div");d.id=n.id+"_instream_container";w.detachMedia();r=g.getDisplayElement();f=E.playlist[E.item];e=C.jwGetState();if(e==a.api.events.state.BUFFERING||e==a.api.events.state.PLAYING){r.pause()}H=r.src?r.src:r.currentSrc;o=r.innerHTML;G=r.currentTime;q=new a.html5.display(n,b.extend({},E.plugins.config.display));q.setAlternateClickHandler(function(N){if(_fakemodel.state==a.api.events.state.PAUSED){n.jwInstreamPlay()}else{D(a.api.events.JWPLAYER_INSTREAM_CLICK,N)}});d.appendChild(q.getDisplayElement());if(!b.isMobile()){l=new a.html5.controlbar(n,b.extend({},E.plugins.config.controlbar,{}));if(E.plugins.config.controlbar.position==a.html5.view.positions.OVER){d.appendChild(l.getDisplayElement())}else{var L=E.plugins.object.controlbar.getDisplayElement().parentNode;L.appendChild(l.getDisplayElement())}}j.setupInstream(d,r);p();g.load(v)};this.jwInstreamDestroy=function(K){if(!h){return}h=false;if(e!=a.api.events.state.IDLE){g.load(f,false);g.stop(false)}else{g.stop(true)}g.detachMedia();j.destroyInstream();if(l){try{l.getDisplayElement().parentNode.removeChild(l.getDisplayElement())}catch(L){}}D(a.api.events.JWPLAYER_INSTREAM_DESTROYED,{reason:(K?"complete":"destroyed")},true);w.attachMedia();if(e==a.api.events.state.BUFFERING||e==a.api.events.state.PLAYING){r.play();if(E.playlist[E.item]==f){E.getMedia().seek(G)}}return};this.jwInstreamAddEventListener=function(K,L){k.addEventListener(K,L)};this.jwInstreamRemoveEventListener=function(K,L){k.removeEventListener(K,L)};this.jwInstreamPlay=function(){if(!h){return}g.play(true)};this.jwInstreamPause=function(){if(!h){return}g.pause(true)};this.jwInstreamSeek=function(K){if(!h){return}g.seek(K)};this.jwInstreamGetState=function(){if(!h){return undefined}return _fakemodel.state};this.jwInstreamGetPosition=function(){if(!h){return undefined}return _fakemodel.position};this.jwInstreamGetDuration=function(){if(!h){return undefined}return _fakemodel.duration};this.playlistClickable=function(){return(!h||A.playlistclickable.toString().toLowerCase()=="true")};function s(){_fakemodel=new a.html5.model(this,E.getMedia()?E.getMedia().getDisplayElement():E.container,E);k=new a.html5.eventdispatcher();C.jwAddEventListener(a.api.events.JWPLAYER_RESIZE,p);C.jwAddEventListener(a.api.events.JWPLAYER_FULLSCREEN,p)}function c(){_fakemodel.setMute(E.mute);_fakemodel.setVolume(E.volume)}function F(){if(!g){g=new a.html5.mediavideo(_fakemodel,E.getMedia()?E.getMedia().getDisplayElement():E.container);g.addGlobalListener(I);g.addEventListener(a.api.events.JWPLAYER_MEDIA_META,J);g.addEventListener(a.api.events.JWPLAYER_MEDIA_COMPLETE,u);g.addEventListener(a.api.events.JWPLAYER_MEDIA_BUFFER_FULL,B)}g.attachMedia()}function I(K){if(h){D(K.type,K)}}function B(K){if(h){g.play()}}function u(K){if(h){setTimeout(function(){n.jwInstreamDestroy(true)},10)}}function J(K){if(K.metadata.width&&K.metadata.height){j.resizeMedia()}}function D(K,L,M){if(h||M){k.sendEvent(K,L)}}function p(){var K=E.plugins.object.display.getDisplayElement().style;if(l){var L=E.plugins.object.controlbar.getDisplayElement().style;l.resize(b.parseDimension(K.width),b.parseDimension(L.height));_css(l.getDisplayElement(),b.extend({},L,{zIndex:1001,opacity:1}))}if(q){q.resize(b.parseDimension(K.width),b.parseDimension(K.height));_css(q.getDisplayElement(),b.extend({},K,{zIndex:1000}))}if(j){j.resizeMedia()}}this.jwPlay=function(K){if(A.controlbarpausable.toString().toLowerCase()=="true"){this.jwInstreamPlay()}};this.jwPause=function(K){if(A.controlbarpausable.toString().toLowerCase()=="true"){this.jwInstreamPause()}};this.jwStop=function(){if(A.controlbarstoppable.toString().toLowerCase()=="true"){this.jwInstreamDestroy();C.jwStop()}};this.jwSeek=function(K){switch(A.controlbarseekable.toLowerCase()){case"always":this.jwInstreamSeek(K);break;case"backwards":if(_fakemodel.position>K){this.jwInstreamSeek(K)}break}};this.jwGetPosition=function(){};this.jwGetDuration=function(){};this.jwGetWidth=C.jwGetWidth;this.jwGetHeight=C.jwGetHeight;this.jwGetFullscreen=C.jwGetFullscreen;this.jwSetFullscreen=C.jwSetFullscreen;this.jwGetVolume=function(){return E.volume};this.jwSetVolume=function(K){g.volume(K);C.jwSetVolume(K)};this.jwGetMute=function(){return E.mute};this.jwSetMute=function(K){g.mute(K);C.jwSetMute(K)};this.jwGetState=function(){return _fakemodel.state};this.jwGetPlaylist=function(){return[v]};this.jwGetPlaylistIndex=function(){return 0};this.jwGetStretching=function(){return E.config.stretching};this.jwAddEventListener=function(L,K){k.addEventListener(L,K)};this.jwRemoveEventListener=function(L,K){k.removeEventListener(L,K)};this.skin=C.skin;this.id=C.id+"_instream";s();return this}})(jwplayer);(function(a){var b={prefix:"http://l.longtailvideo.com/html5/",file:"logo.png",link:"http://www.longtailvideo.com/players/jw-flv-player/",linktarget:"_top",margin:8,out:0.5,over:1,timeout:5,hide:true,position:"bottom-left"};_css=a.utils.css;a.html5.logo=function(n,r){var q=n;var u;var d;var t;var h=false;g();function g(){o();q.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,j);c();l()}function o(){if(b.prefix){var v=n.version.split(/\W/).splice(0,2).join("/");if(b.prefix.indexOf(v)<0){b.prefix+=v+"/"}}if(r.position==a.html5.view.positions.OVER){r.position=b.position}try{if(window.location.href.indexOf("https")==0){b.prefix=b.prefix.replace("http://l.longtailvideo.com","https://securel.longtailvideo.com")}}catch(w){}d=a.utils.extend({},b)}function c(){t=document.createElement("img");t.id=q.id+"_jwplayer_logo";t.style.display="none";t.onload=function(v){_css(t,k());p()};if(!d.file){return}if(d.file.indexOf("/")>=0){t.src=d.file}else{t.src=d.prefix+d.file}}if(!d.file){return}this.resize=function(w,v){};this.getDisplayElement=function(){return t};function l(){if(d.link){t.onmouseover=f;t.onmouseout=p;t.onclick=s}else{this.mouseEnabled=false}}function s(v){if(typeof v!="undefined"){v.stopPropagation()}if(!h){return}q.jwPause();q.jwSetFullscreen(false);if(d.link){window.open(d.link,d.linktarget)}return}function p(v){if(d.link&&h){t.style.opacity=d.out}return}function f(v){if(h){t.style.opacity=d.over}return}function k(){var x={textDecoration:"none",position:"absolute",cursor:"pointer"};x.display=(d.hide.toString()=="true"&&!h)?"none":"block";var w=d.position.toLowerCase().split("-");for(var v in w){x[w[v]]=parseInt(d.margin)}return x}function m(){if(d.hide.toString()=="true"){t.style.display="block";t.style.opacity=0;a.utils.fadeTo(t,d.out,0.1,parseFloat(t.style.opacity));u=setTimeout(function(){e()},d.timeout*1000)}h=true}function e(){h=false;if(d.hide.toString()=="true"){a.utils.fadeTo(t,0,0.1,parseFloat(t.style.opacity))}}function j(v){if(v.newstate==a.api.events.state.BUFFERING){clearTimeout(u);m()}}return this}})(jwplayer);(function(b){var d={ended:b.api.events.state.IDLE,playing:b.api.events.state.PLAYING,pause:b.api.events.state.PAUSED,buffering:b.api.events.state.BUFFERING};var e=b.utils;var a=e.isMobile();var c={};b.html5.mediavideo=function(h,F){var J={abort:y,canplay:p,canplaythrough:p,durationchange:u,emptied:y,ended:p,error:o,loadeddata:u,loadedmetadata:u,loadstart:p,pause:p,play:y,playing:p,progress:D,ratechange:y,seeked:p,seeking:p,stalled:p,suspend:p,timeupdate:N,volumechange:l,waiting:p,canshowcurrentframe:y,dataunavailable:y,empty:y,load:g,loadedfirstframe:y,webkitfullscreenchange:k};var K=new b.html5.eventdispatcher();e.extend(this,K);var j=h,B=F,m,f,C,T,E,M,L=false,t=false,x=false,I,G,Q;R();this.load=function(V,W){if(typeof W=="undefined"){W=true}if(!t){return}T=V;x=(T.duration>0);j.duration=T.duration;e.empty(m);Q=0;q(V.levels);if(V.levels&&V.levels.length>0){if(V.levels.length==1||e.isIOS()){m.src=V.levels[0].file}else{if(m.src){m.removeAttribute("src")}for(var U=0;U<V.levels.length;U++){var X=m.ownerDocument.createElement("source");X.src=V.levels[U].file;m.appendChild(X);Q++}}}else{m.src=V.file}m.style.display="block";m.style.opacity=1;m.volume=j.volume/100;m.muted=j.mute;if(a){P()}I=G=C=false;j.buffer=0;if(!e.exists(V.start)){V.start=0}M=(V.start>0)?V.start:-1;s(b.api.events.JWPLAYER_MEDIA_LOADED);if((!a&&V.levels.length==1)||!L){m.load()}L=false;if(W){w(b.api.events.state.BUFFERING);s(b.api.events.JWPLAYER_MEDIA_BUFFER,{bufferPercent:0});A()}if(m.videoWidth>0&&m.videoHeight>0){u()}};this.play=function(){if(!t){return}A();if(G){w(b.api.events.state.PLAYING)}else{w(b.api.events.state.BUFFERING)}m.play()};this.pause=function(){if(!t){return}m.pause();w(b.api.events.state.PAUSED)};this.seek=function(U){if(!t){return}if(!C&&m.readyState>0){if(!(j.duration<=0||isNaN(j.duration))&&!(j.position<=0||isNaN(j.position))){m.currentTime=U;m.play()}}else{M=U}};var z=this.stop=function(U){if(!t){return}if(!e.exists(U)){U=true}r();if(U){G=false;var V=navigator.userAgent;if(m.webkitSupportsFullscreen){try{m.webkitExitFullscreen()}catch(W){}}m.style.opacity=0;v();if(e.isIE()){m.src=""}else{m.removeAttribute("src")}e.empty(m);m.load();L=true}w(b.api.events.state.IDLE)};this.fullscreen=function(U){if(U===true){this.resize("100%","100%")}else{this.resize(j.config.width,j.config.height)}};this.resize=function(V,U){};this.volume=function(U){if(!a){m.volume=U/100;s(b.api.events.JWPLAYER_MEDIA_VOLUME,{volume:(U/100)})}};this.mute=function(U){if(!a){m.muted=U;s(b.api.events.JWPLAYER_MEDIA_MUTE,{mute:U})}};this.getDisplayElement=function(){return m};this.hasChrome=function(){return a&&(f==b.api.events.state.PLAYING)};this.detachMedia=function(){t=false;return this.getDisplayElement()};this.attachMedia=function(){t=true};function H(V,U){return function(W){if(e.exists(W.target.parentNode)){U(W)}}}function R(){f=b.api.events.state.IDLE;t=true;m=n();m.setAttribute("x-webkit-airplay","allow");if(B.parentNode){B.parentNode.replaceChild(m,B)}}function n(){var U;if(!c[j.id]){if(B.tagName.toLowerCase()=="video"){U=B}else{U=document.createElement("video")}c[j.id]=U;if(!U.id){U.id=B.id}for(var V in J){U.addEventListener(V,H(V,J[V]),true)}}return c[j.id]}function w(U){if(U==b.api.events.state.PAUSED&&f==b.api.events.state.IDLE){return}if(a){switch(U){case b.api.events.state.PLAYING:P();break;case b.api.events.state.BUFFERING:case b.api.events.state.PAUSED:v();break}}if(f!=U){var V=f;j.state=f=U;s(b.api.events.JWPLAYER_PLAYER_STATE,{oldstate:V,newstate:U})}}function y(U){}function l(U){var V=Math.round(m.volume*100);s(b.api.events.JWPLAYER_MEDIA_VOLUME,{volume:V},true);s(b.api.events.JWPLAYER_MEDIA_MUTE,{mute:m.muted},true)}function D(W){if(!t){return}var V;if(e.exists(W)&&W.lengthComputable&&W.total){V=W.loaded/W.total*100}else{if(e.exists(m.buffered)&&(m.buffered.length>0)){var U=m.buffered.length-1;if(U>=0){V=m.buffered.end(U)/m.duration*100}}}if(e.useNativeFullscreen()&&e.exists(m.webkitDisplayingFullscreen)){if(j.fullscreen!=m.webkitDisplayingFullscreen){s(b.api.events.JWPLAYER_FULLSCREEN,{fullscreen:m.webkitDisplayingFullscreen},true)}}if(G===false&&f==b.api.events.state.BUFFERING){s(b.api.events.JWPLAYER_MEDIA_BUFFER_FULL);G=true}if(!I){if(V==100){I=true}if(e.exists(V)&&(V>j.buffer)){j.buffer=Math.round(V);s(b.api.events.JWPLAYER_MEDIA_BUFFER,{bufferPercent:Math.round(V)})}}}function N(V){if(!t){return}if(e.exists(V)&&e.exists(V.target)){if(x>0){if(!isNaN(V.target.duration)&&(isNaN(j.duration)||j.duration<1)){if(V.target.duration==Infinity){j.duration=0}else{j.duration=Math.round(V.target.duration*10)/10}}}if(!C&&m.readyState>0){w(b.api.events.state.PLAYING)}if(f==b.api.events.state.PLAYING){if(m.readyState>0&&(M>-1||!C)){C=true;try{if(m.currentTime!=M&&M>-1){m.currentTime=M;M=-1}}catch(U){}m.volume=j.volume/100;m.muted=j.mute}j.position=j.duration>0?(Math.round(V.target.currentTime*10)/10):0;s(b.api.events.JWPLAYER_MEDIA_TIME,{position:j.position,duration:j.duration});if(j.position>=j.duration&&(j.position>0||j.duration>0)){O();return}}}D(V)}function g(U){}function p(U){if(!t){return}if(d[U.type]){if(U.type=="ended"){O()}else{w(d[U.type])}}}function u(V){if(!t){return}var U=Math.round(m.duration*10)/10;var W={height:m.videoHeight,width:m.videoWidth,duration:U};if(!x){if((j.duration<U||isNaN(j.duration))&&m.duration!=Infinity){j.duration=U}}s(b.api.events.JWPLAYER_MEDIA_META,{metadata:W})}function o(W){if(!t){return}if(f==b.api.events.state.IDLE){return}var V="There was an error: ";if((W.target.error&&W.target.tagName.toLowerCase()=="video")||W.target.parentNode.error&&W.target.parentNode.tagName.toLowerCase()=="video"){var U=!e.exists(W.target.error)?W.target.parentNode.error:W.target.error;switch(U.code){case U.MEDIA_ERR_ABORTED:e.log("User aborted the video playback.");return;case U.MEDIA_ERR_NETWORK:V="A network error caused the video download to fail part-way: ";break;case U.MEDIA_ERR_DECODE:V="The video playback was aborted due to a corruption problem or because the video used features your browser did not support: ";break;case U.MEDIA_ERR_SRC_NOT_SUPPORTED:V="The video could not be loaded, either because the server or network failed or because the format is not supported: ";break;default:V="An unknown error occurred: ";break}}else{if(W.target.tagName.toLowerCase()=="source"){Q--;if(Q>0){return}if(e.userAgentMatch(/firefox/i)){e.log("The video could not be loaded, either because the server or network failed or because the format is not supported.");z(false);return}else{V="The video could not be loaded, either because the server or network failed or because the format is not supported: "}}else{e.log("An unknown error occurred. Continuing...");return}}z(false);V+=S();_error=true;s(b.api.events.JWPLAYER_ERROR,{message:V});return}function S(){var W="";for(var V in T.levels){var U=T.levels[V];var X=B.ownerDocument.createElement("source");W+=b.utils.getAbsolutePath(U.file);if(V<(T.levels.length-1)){W+=", "}}return W}function A(){if(!e.exists(E)){E=setInterval(function(){D()},100)}}function r(){clearInterval(E);E=null}function O(){if(f==b.api.events.state.PLAYING){z(false);s(b.api.events.JWPLAYER_MEDIA_BEFORECOMPLETE);s(b.api.events.JWPLAYER_MEDIA_COMPLETE)}}function k(U){if(e.exists(m.webkitDisplayingFullscreen)){if(j.fullscreen&&!m.webkitDisplayingFullscreen){s(b.api.events.JWPLAYER_FULLSCREEN,{fullscreen:false},true)}}}function q(W){if(W.length>0&&e.userAgentMatch(/Safari/i)){var U=-1;for(var V=0;V<W.length;V++){switch(e.extension(W[V].file)){case"mp4":if(U<0){U=V}break;case"webm":W.splice(V,1);break}}if(U>0){var X=W.splice(U,1)[0];W.unshift(X)}}}function P(){setTimeout(function(){m.setAttribute("controls","controls")},100)}function v(){setTimeout(function(){m.removeAttribute("controls")},250)}function s(U,W,V){if(t||V){if(W){K.sendEvent(U,W)}else{K.sendEvent(U)}}}}})(jwplayer);(function(a){var c={ended:a.api.events.state.IDLE,playing:a.api.events.state.PLAYING,pause:a.api.events.state.PAUSED,buffering:a.api.events.state.BUFFERING};var b=a.utils.css;a.html5.mediayoutube=function(j,e){var f=new a.html5.eventdispatcher();a.utils.extend(this,f);var l=j;var h=document.getElementById(e.id);var g=a.api.events.state.IDLE;var n,m;function k(p){if(g!=p){var q=g;l.state=p;g=p;f.sendEvent(a.api.events.JWPLAYER_PLAYER_STATE,{oldstate:q,newstate:p})}}this.getDisplayElement=this.detachMedia=function(){return h};this.attachMedia=function(){};this.play=function(){if(g==a.api.events.state.IDLE){f.sendEvent(a.api.events.JWPLAYER_MEDIA_BUFFER,{bufferPercent:100});f.sendEvent(a.api.events.JWPLAYER_MEDIA_BUFFER_FULL);k(a.api.events.state.PLAYING)}else{if(g==a.api.events.state.PAUSED){k(a.api.events.state.PLAYING)}}};this.pause=function(){k(a.api.events.state.PAUSED)};this.seek=function(p){};this.stop=function(p){if(!_utils.exists(p)){p=true}l.position=0;k(a.api.events.state.IDLE);if(p){b(h,{display:"none"})}};this.volume=function(p){l.setVolume(p);f.sendEvent(a.api.events.JWPLAYER_MEDIA_VOLUME,{volume:Math.round(p)})};this.mute=function(p){h.muted=p;f.sendEvent(a.api.events.JWPLAYER_MEDIA_MUTE,{mute:p})};this.resize=function(q,p){if(q*p>0&&n){n.width=m.width=q;n.height=m.height=p}};this.fullscreen=function(p){if(p===true){this.resize("100%","100%")}else{this.resize(l.config.width,l.config.height)}};this.load=function(p){o(p);b(n,{display:"block"});k(a.api.events.state.BUFFERING);f.sendEvent(a.api.events.JWPLAYER_MEDIA_BUFFER,{bufferPercent:0});f.sendEvent(a.api.events.JWPLAYER_MEDIA_LOADED);this.play()};this.hasChrome=function(){return(g!=a.api.events.state.IDLE)};function o(v){var s=v.levels[0].file;s=["http://www.youtube.com/v/",d(s),"&hl=en_US&fs=1&autoplay=1"].join("");n=document.createElement("object");n.id=h.id;n.style.position="absolute";var u={movie:s,allowfullscreen:"true",allowscriptaccess:"always"};for(var p in u){var t=document.createElement("param");t.name=p;t.value=u[p];n.appendChild(t)}m=document.createElement("embed");n.appendChild(m);var q={src:s,type:"application/x-shockwave-flash",allowfullscreen:"true",allowscriptaccess:"always",width:n.width,height:n.height};for(var r in q){m.setAttribute(r,q[r])}n.appendChild(m);n.style.zIndex=2147483000;if(h!=n&&h.parentNode){h.parentNode.replaceChild(n,h)}h=n}function d(q){var p=q.split(/\?|\#\!/);var s="";for(var r=0;r<p.length;r++){if(p[r].substr(0,2)=="v="){s=p[r].substr(2)}}if(s==""){if(q.indexOf("/v/")>=0){s=q.substr(q.indexOf("/v/")+3)}else{if(q.indexOf("youtu.be")>=0){s=q.substr(q.indexOf("youtu.be/")+9)}else{s=q}}}if(s.indexOf("?")>-1){s=s.substr(0,s.indexOf("?"))}if(s.indexOf("&")>-1){s=s.substr(0,s.indexOf("&"))}return s}this.embed=m;return this}})(jwplayer);(function(jwplayer){var _configurableStateVariables=["width","height","start","duration","volume","mute","fullscreen","item","plugins","stretching"];var _utils=jwplayer.utils;jwplayer.html5.model=function(api,container,options){var _api=api;var _container=container;var _cookies=_utils.getCookies();var _model={id:_container.id,playlist:[],state:jwplayer.api.events.state.IDLE,position:0,buffer:0,container:_container,config:{width:480,height:320,item:-1,skin:undefined,file:undefined,image:undefined,start:0,duration:0,bufferlength:5,volume:_cookies.volume?_cookies.volume:90,mute:_cookies.mute&&_cookies.mute.toString().toLowerCase()=="true"?true:false,fullscreen:false,repeat:"",stretching:jwplayer.utils.stretching.UNIFORM,autostart:false,debug:undefined,screencolor:undefined}};var _media;var _eventDispatcher=new jwplayer.html5.eventdispatcher();var _components=["display","logo","controlbar","playlist","dock"];jwplayer.utils.extend(_model,_eventDispatcher);for(var option in options){if(typeof options[option]=="string"){var type=/color$/.test(option)?"color":null;options[option]=jwplayer.utils.typechecker(options[option],type)}var config=_model.config;var path=option.split(".");for(var edge in path){if(edge==path.length-1){config[path[edge]]=options[option]}else{if(!jwplayer.utils.exists(config[path[edge]])){config[path[edge]]={}}config=config[path[edge]]}}}for(var index in _configurableStateVariables){var configurableStateVariable=_configurableStateVariables[index];_model[configurableStateVariable]=_model.config[configurableStateVariable]}var pluginorder=_components.concat([]);if(jwplayer.utils.exists(_model.plugins)){if(typeof _model.plugins=="string"){var userplugins=_model.plugins.split(",");for(var userplugin in userplugins){if(typeof userplugins[userplugin]=="string"){pluginorder.push(userplugins[userplugin].replace(/^\s+|\s+$/g,""))}}}}if(jwplayer.utils.isMobile()){pluginorder=["display","logo","dock","playlist"];if(!jwplayer.utils.exists(_model.config.repeat)){_model.config.repeat="list"}}else{if(_model.config.chromeless){pluginorder=["logo","dock","playlist"];if(!jwplayer.utils.exists(_model.config.repeat)){_model.config.repeat="list"}}}_model.plugins={order:pluginorder,config:{},object:{}};if(typeof _model.config.components!="undefined"){for(var component in _model.config.components){_model.plugins.config[component]=_model.config.components[component]}}var playlistVisible=false;for(var pluginIndex in _model.plugins.order){var pluginName=_model.plugins.order[pluginIndex];var pluginConfig=!jwplayer.utils.exists(_model.plugins.config[pluginName])?{}:_model.plugins.config[pluginName];_model.plugins.config[pluginName]=!jwplayer.utils.exists(_model.plugins.config[pluginName])?pluginConfig:jwplayer.utils.extend(_model.plugins.config[pluginName],pluginConfig);if(!jwplayer.utils.exists(_model.plugins.config[pluginName].position)){if(pluginName=="playlist"){_model.plugins.config[pluginName].position=jwplayer.html5.view.positions.NONE}else{_model.plugins.config[pluginName].position=jwplayer.html5.view.positions.OVER}}else{if(pluginName=="playlist"){playlistVisible=true}_model.plugins.config[pluginName].position=_model.plugins.config[pluginName].position.toString().toUpperCase()}}if(_model.plugins.config.controlbar&&playlistVisible){_model.plugins.config.controlbar.hideplaylistcontrols=true}if(typeof _model.plugins.config.dock!="undefined"){if(typeof _model.plugins.config.dock!="object"){var position=_model.plugins.config.dock.toString().toUpperCase();_model.plugins.config.dock={position:position}}if(typeof _model.plugins.config.dock.position!="undefined"){_model.plugins.config.dock.align=_model.plugins.config.dock.position;_model.plugins.config.dock.position=jwplayer.html5.view.positions.OVER}if(typeof _model.plugins.config.dock.idlehide=="undefined"){try{_model.plugins.config.dock.idlehide=_model.plugins.config.controlbar.idlehide}catch(e){}}}function _loadExternal(playlistfile){var loader=new jwplayer.html5.playlistloader();loader.addEventListener(jwplayer.api.events.JWPLAYER_PLAYLIST_LOADED,function(evt){_model.playlist=new jwplayer.html5.playlist(evt);_loadComplete(true)});loader.addEventListener(jwplayer.api.events.JWPLAYER_ERROR,function(evt){_model.playlist=new jwplayer.html5.playlist({playlist:[]});_loadComplete(false)});loader.load(playlistfile)}function _loadComplete(){if(_model.config.shuffle){_model.item=_getShuffleItem()}else{if(_model.config.item>=_model.playlist.length){_model.config.item=_model.playlist.length-1}else{if(_model.config.item<0){_model.config.item=0}}_model.item=_model.config.item}_model.position=0;_model.duration=_model.playlist.length>0?_model.playlist[_model.item].duration:0;_eventDispatcher.sendEvent(jwplayer.api.events.JWPLAYER_PLAYLIST_LOADED,{playlist:_model.playlist});_eventDispatcher.sendEvent(jwplayer.api.events.JWPLAYER_PLAYLIST_ITEM,{index:_model.item})}_model.loadPlaylist=function(arg){var input;if(typeof arg=="string"){if(arg.indexOf("[")==0||arg.indexOf("{")=="0"){try{input=eval(arg)}catch(err){input=arg}}else{input=arg}}else{input=arg}var config;switch(jwplayer.utils.typeOf(input)){case"object":config=input;break;case"array":config={playlist:input};break;default:config={file:input};break}_model.playlist=new jwplayer.html5.playlist(config);_model.item=_model.config.item>=0?_model.config.item:0;if(!_model.playlist[0].provider&&_model.playlist[0].file){_loadExternal(_model.playlist[0].file)}else{_loadComplete()}};function _getShuffleItem(){var result=null;if(_model.playlist.length>1){while(!jwplayer.utils.exists(result)){result=Math.floor(Math.random()*_model.playlist.length);if(result==_model.item){result=null}}}else{result=0}return result}function forward(evt){switch(evt.type){case jwplayer.api.events.JWPLAYER_MEDIA_LOADED:_container=_media.getDisplayElement();break;case jwplayer.api.events.JWPLAYER_MEDIA_MUTE:this.mute=evt.mute;break;case jwplayer.api.events.JWPLAYER_MEDIA_VOLUME:this.volume=evt.volume;break}_eventDispatcher.sendEvent(evt.type,evt)}var _mediaProviders={};_model.setActiveMediaProvider=function(playlistItem){if(playlistItem.provider=="audio"){playlistItem.provider="sound"}var provider=playlistItem.provider;var current=_media?_media.getDisplayElement():null;if(provider=="sound"||provider=="http"||provider==""){provider="video"}if(!jwplayer.utils.exists(_mediaProviders[provider])){switch(provider){case"video":_media=new jwplayer.html5.mediavideo(_model,current?current:_container);break;case"youtube":_media=new jwplayer.html5.mediayoutube(_model,current?current:_container);break}if(!jwplayer.utils.exists(_media)){return false}_media.addGlobalListener(forward);_mediaProviders[provider]=_media}else{if(_media!=_mediaProviders[provider]){if(_media){_media.stop()}_media=_mediaProviders[provider]}}return true};_model.getMedia=function(){return _media};_model.seek=function(pos){_eventDispatcher.sendEvent(jwplayer.api.events.JWPLAYER_MEDIA_SEEK,{position:_model.position,offset:pos});return _media.seek(pos)};_model.setVolume=function(newVol){_utils.saveCookie("volume",newVol);_model.volume=newVol};_model.setMute=function(state){_utils.saveCookie("mute",state);_model.mute=state};_model.setupPlugins=function(){if(!jwplayer.utils.exists(_model.plugins)||!jwplayer.utils.exists(_model.plugins.order)||_model.plugins.order.length==0){jwplayer.utils.log("No plugins to set up");return _model}for(var i=0;i<_model.plugins.order.length;i++){try{var pluginName=_model.plugins.order[i];if(jwplayer.utils.exists(jwplayer.html5[pluginName])){if(pluginName=="playlist"){_model.plugins.object[pluginName]=new jwplayer.html5.playlistcomponent(_api,_model.plugins.config[pluginName])}else{_model.plugins.object[pluginName]=new jwplayer.html5[pluginName](_api,_model.plugins.config[pluginName])}}else{_model.plugins.order.splice(plugin,plugin+1)}if(typeof _model.plugins.object[pluginName].addGlobalListener=="function"){_model.plugins.object[pluginName].addGlobalListener(forward)}}catch(err){jwplayer.utils.log("Could not setup "+pluginName)}}};return _model}})(jwplayer);(function(a){a.html5.playlist=function(b){var d=[];if(b.playlist&&b.playlist instanceof Array&&b.playlist.length>0){for(var c in b.playlist){if(!isNaN(parseInt(c))){d.push(new a.html5.playlistitem(b.playlist[c]))}}}else{d.push(new a.html5.playlistitem(b))}return d}})(jwplayer);(function(a){var c={size:180,position:a.html5.view.positions.NONE,itemheight:60,thumbs:true,fontcolor:"#000000",overcolor:"",activecolor:"",backgroundcolor:"#f8f8f8",font:"_sans",fontsize:"",fontstyle:"",fontweight:""};var b={_sans:"Arial, Helvetica, sans-serif",_serif:"Times, Times New Roman, serif",_typewriter:"Courier New, Courier, monospace"};_utils=a.utils;_css=_utils.css;_hide=function(d){_css(d,{display:"none"})};_show=function(d){_css(d,{display:"block"})};a.html5.playlistcomponent=function(r,C){var x=r;var e=a.utils.extend({},c,x.skin.getComponentSettings("playlist"),C);if(e.position==a.html5.view.positions.NONE||typeof a.html5.view.positions[e.position]=="undefined"){return}var y;var l;var D;var d;var g;var f;var k=-1;var h={background:undefined,item:undefined,itemOver:undefined,itemImage:undefined,itemActive:undefined};this.getDisplayElement=function(){return y};this.resize=function(G,E){l=G;D=E;if(x.jwGetFullscreen()){_hide(y)}else{var F={display:"block",width:l,height:D};_css(y,F)}};this.show=function(){_show(y)};this.hide=function(){_hide(y)};function j(){y=document.createElement("div");y.id=x.id+"_jwplayer_playlistcomponent";y.style.overflow="hidden";switch(e.position){case a.html5.view.positions.RIGHT:case a.html5.view.positions.LEFT:y.style.width=e.size+"px";break;case a.html5.view.positions.TOP:case a.html5.view.positions.BOTTOM:y.style.height=e.size+"px";break}B();if(h.item){e.itemheight=h.item.height}y.style.backgroundColor="#C6C6C6";x.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,s);x.jwAddEventListener(a.api.events.JWPLAYER_PLAYLIST_ITEM,v);x.jwAddEventListener(a.api.events.JWPLAYER_PLAYER_STATE,m)}function p(){var E=document.createElement("ul");_css(E,{width:y.style.width,minWidth:y.style.width,height:y.style.height,backgroundColor:e.backgroundcolor,backgroundImage:h.background?"url("+h.background.src+")":"",color:e.fontcolor,listStyle:"none",margin:0,padding:0,fontFamily:b[e.font]?b[e.font]:b._sans,fontSize:(e.fontsize?e.fontsize:11)+"px",fontStyle:e.fontstyle,fontWeight:e.fontweight,overflowY:"auto"});return E}function z(E){return function(){var F=f.getElementsByClassName("item")[E];var G=e.fontcolor;var H=h.item?"url("+h.item.src+")":"";if(E==x.jwGetPlaylistIndex()){if(e.activecolor!==""){G=e.activecolor}if(h.itemActive){H="url("+h.itemActive.src+")"}}_css(F,{color:e.overcolor!==""?e.overcolor:G,backgroundImage:h.itemOver?"url("+h.itemOver.src+")":H})}}function o(E){return function(){var F=f.getElementsByClassName("item")[E];var G=e.fontcolor;var H=h.item?"url("+h.item.src+")":"";if(E==x.jwGetPlaylistIndex()){if(e.activecolor!==""){G=e.activecolor}if(h.itemActive){H="url("+h.itemActive.src+")"}}_css(F,{color:G,backgroundImage:H})}}function q(J){var Q=d[J];var P=document.createElement("li");P.className="item";_css(P,{height:e.itemheight,display:"block",cursor:"pointer",backgroundImage:h.item?"url("+h.item.src+")":"",backgroundSize:"100% "+e.itemheight+"px"});P.onmouseover=z(J);P.onmouseout=o(J);var K=document.createElement("div");var G=new Image();var L=0;var M=0;var N=0;if(w()&&(Q.image||Q["playlist.image"]||h.itemImage)){G.className="image";if(h.itemImage){L=(e.itemheight-h.itemImage.height)/2;M=h.itemImage.width;N=h.itemImage.height}else{M=e.itemheight*4/3;N=e.itemheight}_css(K,{height:N,width:M,"float":"left",styleFloat:"left",cssFloat:"left",margin:"0 5px 0 0",background:"black",overflow:"hidden",margin:L+"px",position:"relative"});_css(G,{position:"relative"});K.appendChild(G);G.onload=function(){a.utils.stretch(a.utils.stretching.FILL,G,M,N,this.naturalWidth,this.naturalHeight)};if(Q["playlist.image"]){G.src=Q["playlist.image"]}else{if(Q.image){G.src=Q.image}else{if(h.itemImage){G.src=h.itemImage.src}}}P.appendChild(K)}var F=l-M-L*2;if(D<e.itemheight*d.length){F-=15}var E=document.createElement("div");_css(E,{position:"relative",height:"100%",overflow:"hidden"});var H=document.createElement("span");if(Q.duration>0){H.className="duration";_css(H,{fontSize:(e.fontsize?e.fontsize:11)+"px",fontWeight:(e.fontweight?e.fontweight:"bold"),width:"40px",height:e.fontsize?e.fontsize+10:20,lineHeight:24,"float":"right",styleFloat:"right",cssFloat:"right"});H.innerHTML=_utils.timeFormat(Q.duration);E.appendChild(H)}var O=document.createElement("span");O.className="title";_css(O,{padding:"5px 5px 0 "+(L?0:"5px"),height:e.fontsize?e.fontsize+10:20,lineHeight:e.fontsize?e.fontsize+10:20,overflow:"hidden","float":"left",styleFloat:"left",cssFloat:"left",width:((Q.duration>0)?F-50:F)-10+"px",fontSize:(e.fontsize?e.fontsize:13)+"px",fontWeight:(e.fontweight?e.fontweight:"bold")});O.innerHTML=Q?Q.title:"";E.appendChild(O);if(Q.description){var I=document.createElement("span");I.className="description";_css(I,{display:"block","float":"left",styleFloat:"left",cssFloat:"left",margin:0,paddingLeft:O.style.paddingLeft,paddingRight:O.style.paddingRight,lineHeight:(e.fontsize?e.fontsize+4:16)+"px",overflow:"hidden",position:"relative"});I.innerHTML=Q.description;E.appendChild(I)}P.appendChild(E);return P}function s(F){y.innerHTML="";d=t();if(!d){return}items=[];f=p();for(var G=0;G<d.length;G++){var E=q(G);E.onclick=A(G);f.appendChild(E);items.push(E)}k=x.jwGetPlaylistIndex();o(k)();y.appendChild(f);if(_utils.isIOS()&&window.iScroll){f.style.height=e.itemheight*d.length+"px";var H=new iScroll(y.id)}}function t(){var F=x.jwGetPlaylist();var G=[];for(var E=0;E<F.length;E++){if(!F[E]["ova.hidden"]){G.push(F[E])}}return G}function A(E){return function(){x.jwPlaylistItem(E);x.jwPlay(true)}}function n(){f.scrollTop=x.jwGetPlaylistIndex()*e.itemheight}function w(){return e.thumbs.toString().toLowerCase()=="true"}function v(E){if(k>=0){o(k)();k=E.index}o(E.index)();n()}function m(){if(e.position==a.html5.view.positions.OVER){switch(x.jwGetState()){case a.api.events.state.IDLE:_show(y);break;default:_hide(y);break}}}function B(){for(var E in h){h[E]=u(E)}}function u(E){return x.skin.getSkinElement("playlist",E)}j();return this}})(jwplayer);(function(b){b.html5.playlistitem=function(d){var e={author:"",date:"",description:"",image:"",link:"",mediaid:"",tags:"",title:"",provider:"",file:"",streamer:"",duration:-1,start:0,currentLevel:-1,levels:[]};var c=b.utils.extend({},e,d);if(c.type){c.provider=c.type;delete c.type}if(c.levels.length===0){c.levels[0]=new b.html5.playlistitemlevel(c)}if(!c.provider){c.provider=a(c.levels[0])}else{c.provider=c.provider.toLowerCase()}return c};function a(e){if(b.utils.isYouTube(e.file)){return"youtube"}else{var f=b.utils.extension(e.file);var c;if(f&&b.utils.extensionmap[f]){if(f=="m3u8"){return"video"}c=b.utils.extensionmap[f].html5}else{if(e.type){c=e.type}}if(c){var d=c.split("/")[0];if(d=="audio"){return"sound"}else{if(d=="video"){return d}}}}return""}})(jwplayer);(function(a){a.html5.playlistitemlevel=function(b){var d={file:"",streamer:"",bitrate:0,width:0};for(var c in d){if(a.utils.exists(b[c])){d[c]=b[c]}}return d}})(jwplayer);(function(a){a.html5.playlistloader=function(){var c=new a.html5.eventdispatcher();a.utils.extend(this,c);this.load=function(e){a.utils.ajax(e,d,b)};function d(g){var f=[];try{var f=a.utils.parsers.rssparser.parse(g.responseXML.firstChild);c.sendEvent(a.api.events.JWPLAYER_PLAYLIST_LOADED,{playlist:new a.html5.playlist({playlist:f})})}catch(h){b("Could not parse the playlist")}}function b(e){c.sendEvent(a.api.events.JWPLAYER_ERROR,{message:e?e:"Could not load playlist an unknown reason."})}}})(jwplayer);(function(a){a.html5.skin=function(){var b={};var c=false;this.load=function(d,e){new a.html5.skinloader(d,function(f){c=true;b=f;e()},function(){new a.html5.skinloader("",function(f){c=true;b=f;e()})})};this.getSkinElement=function(d,e){if(c){try{return b[d].elements[e]}catch(f){a.utils.log("No such skin component / element: ",[d,e])}}return null};this.getComponentSettings=function(d){if(c&&b&&b[d]){return b[d].settings}return null};this.getComponentLayout=function(d){if(c){return b[d].layout}return null}}})(jwplayer);(function(a){a.html5.skinloader=function(f,p,k){var o={};var c=p;var l=k;var e=true;var j;var n=f;var s=false;function m(){if(typeof n!="string"||n===""){d(a.html5.defaultSkin().xml)}else{a.utils.ajax(a.utils.getAbsolutePath(n),function(t){try{if(a.utils.exists(t.responseXML)){d(t.responseXML);return}}catch(u){h()}d(a.html5.defaultSkin().xml)},function(t){d(a.html5.defaultSkin().xml)})}}function d(y){var E=y.getElementsByTagName("component");if(E.length===0){return}for(var H=0;H<E.length;H++){var C=E[H].getAttribute("name");var B={settings:{},elements:{},layout:{}};o[C]=B;var G=E[H].getElementsByTagName("elements")[0].getElementsByTagName("element");for(var F=0;F<G.length;F++){b(G[F],C)}var z=E[H].getElementsByTagName("settings")[0];if(z&&z.childNodes.length>0){var K=z.getElementsByTagName("setting");for(var P=0;P<K.length;P++){var Q=K[P].getAttribute("name");var I=K[P].getAttribute("value");var x=/color$/.test(Q)?"color":null;o[C].settings[Q]=a.utils.typechecker(I,x)}}var L=E[H].getElementsByTagName("layout")[0];if(L&&L.childNodes.length>0){var M=L.getElementsByTagName("group");for(var w=0;w<M.length;w++){var A=M[w];o[C].layout[A.getAttribute("position")]={elements:[]};for(var O=0;O<A.attributes.length;O++){var D=A.attributes[O];o[C].layout[A.getAttribute("position")][D.name]=D.value}var N=A.getElementsByTagName("*");for(var v=0;v<N.length;v++){var t=N[v];o[C].layout[A.getAttribute("position")].elements.push({type:t.tagName});for(var u=0;u<t.attributes.length;u++){var J=t.attributes[u];o[C].layout[A.getAttribute("position")].elements[v][J.name]=J.value}if(!a.utils.exists(o[C].layout[A.getAttribute("position")].elements[v].name)){o[C].layout[A.getAttribute("position")].elements[v].name=t.tagName}}}}e=false;r()}}function r(){clearInterval(j);if(!s){j=setInterval(function(){q()},100)}}function b(y,x){var w=new Image();var t=y.getAttribute("name");var v=y.getAttribute("src");var A;if(v.indexOf("data:image/png;base64,")===0){A=v}else{var u=a.utils.getAbsolutePath(n);var z=u.substr(0,u.lastIndexOf("/"));A=[z,x,v].join("/")}o[x].elements[t]={height:0,width:0,src:"",ready:false,image:w};w.onload=function(B){g(w,t,x)};w.onerror=function(B){s=true;r();l()};w.src=A}function h(){for(var u in o){var w=o[u];for(var t in w.elements){var x=w.elements[t];var v=x.image;v.onload=null;v.onerror=null;delete x.image;delete w.elements[t]}delete o[u]}}function q(){for(var t in o){if(t!="properties"){for(var u in o[t].elements){if(!o[t].elements[u].ready){return}}}}if(e===false){clearInterval(j);c(o)}}function g(t,v,u){if(o[u]&&o[u].elements[v]){o[u].elements[v].height=t.height;o[u].elements[v].width=t.width;o[u].elements[v].src=t.src;o[u].elements[v].ready=true;r()}else{a.utils.log("Loaded an image for a missing element: "+u+"."+v)}}m()}})(jwplayer);(function(a){a.html5.api=function(c,p){var n={};var g=document.createElement("div");c.parentNode.replaceChild(g,c);g.id=c.id;n.version=a.version;n.id=g.id;var m=new a.html5.model(n,g,p);var k=new a.html5.view(n,g,m);var l=new a.html5.controller(n,g,m,k);n.skin=new a.html5.skin();n.jwPlay=function(q){if(typeof q=="undefined"){f()}else{if(q.toString().toLowerCase()=="true"){l.play()}else{l.pause()}}};n.jwPause=function(q){if(typeof q=="undefined"){f()}else{if(q.toString().toLowerCase()=="true"){l.pause()}else{l.play()}}};function f(){if(m.state==a.api.events.state.PLAYING||m.state==a.api.events.state.BUFFERING){l.pause()}else{l.play()}}n.jwStop=l.stop;n.jwSeek=l.seek;n.jwPlaylistItem=function(q){if(d){if(d.playlistClickable()){d.jwInstreamDestroy();return l.item(q)}}else{return l.item(q)}};n.jwPlaylistNext=l.next;n.jwPlaylistPrev=l.prev;n.jwResize=l.resize;n.jwLoad=l.load;n.jwDetachMedia=l.detachMedia;n.jwAttachMedia=l.attachMedia;function j(q){return function(){return m[q]}}function e(q,s,r){return function(){var t=m.plugins.object[q];if(t&&t[s]&&typeof t[s]=="function"){t[s].apply(t,r)}}}n.jwGetPlaylistIndex=j("item");n.jwGetPosition=j("position");n.jwGetDuration=j("duration");n.jwGetBuffer=j("buffer");n.jwGetWidth=j("width");n.jwGetHeight=j("height");n.jwGetFullscreen=j("fullscreen");n.jwSetFullscreen=l.setFullscreen;n.jwGetVolume=j("volume");n.jwSetVolume=l.setVolume;n.jwGetMute=j("mute");n.jwSetMute=l.setMute;n.jwGetStretching=function(){return m.stretching.toUpperCase()};n.jwGetState=j("state");n.jwGetVersion=function(){return n.version};n.jwGetPlaylist=function(){return m.playlist};n.jwAddEventListener=l.addEventListener;n.jwRemoveEventListener=l.removeEventListener;n.jwSendEvent=l.sendEvent;n.jwDockSetButton=function(t,q,r,s){if(m.plugins.object.dock&&m.plugins.object.dock.setButton){m.plugins.object.dock.setButton(t,q,r,s)}};n.jwControlbarShow=e("controlbar","show");n.jwControlbarHide=e("controlbar","hide");n.jwDockShow=e("dock","show");n.jwDockHide=e("dock","hide");n.jwDisplayShow=e("display","show");n.jwDisplayHide=e("display","hide");var d;n.jwLoadInstream=function(r,q){if(!d){d=new a.html5.instream(n,m,k,l)}setTimeout(function(){d.load(r,q)},10)};n.jwInstreamDestroy=function(){if(d){d.jwInstreamDestroy()}};n.jwInstreamAddEventListener=o("jwInstreamAddEventListener");n.jwInstreamRemoveEventListener=o("jwInstreamRemoveEventListener");n.jwInstreamGetState=o("jwInstreamGetState");n.jwInstreamGetDuration=o("jwInstreamGetDuration");n.jwInstreamGetPosition=o("jwInstreamGetPosition");n.jwInstreamPlay=o("jwInstreamPlay");n.jwInstreamPause=o("jwInstreamPause");n.jwInstreamSeek=o("jwInstreamSeek");function o(q){return function(){if(d&&typeof d[q]=="function"){return d[q].apply(this,arguments)}else{_utils.log("Could not call instream method - instream API not initialized")}}}n.jwGetLevel=function(){};n.jwGetBandwidth=function(){};n.jwGetLockState=function(){};n.jwLock=function(){};n.jwUnlock=function(){};function b(){if(m.config.playlistfile){m.addEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,h);m.loadPlaylist(m.config.playlistfile)}else{if(typeof m.config.playlist=="string"){m.addEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,h);m.loadPlaylist(m.config.playlist)}else{m.loadPlaylist(m.config);setTimeout(h,25)}}}function h(q){m.removeEventListener(a.api.events.JWPLAYER_PLAYLIST_LOADED,h);m.setupPlugins();k.setup();var q={id:n.id,version:n.version};l.playerReady(q)}if(m.config.chromeless&&!a.utils.isIOS()){b()}else{n.skin.load(m.config.skin,b)}return n}})(jwplayer)}; \ No newline at end of file diff --git a/bbb-screenshare/jws/player/jw-player/license.txt b/bbb-screenshare/jws/player/jw-player/license.txt new file mode 100755 index 0000000000000000000000000000000000000000..3e8047b04f92e963d4631661ae9548df0a08ac6f --- /dev/null +++ b/bbb-screenshare/jws/player/jw-player/license.txt @@ -0,0 +1 @@ +This work is licensed under the Creative Commons Attribution-NonCommercial-ShareAlike 3.0 Unported License. To view a copy of this license, visit http://creativecommons.org/licenses/by-nc-sa/3.0/ or send a letter to Creative Commons, 171 Second Street, Suite 300, San Francisco, California, 94105, USA. \ No newline at end of file diff --git a/bbb-screenshare/jws/player/jw-player/player.swf b/bbb-screenshare/jws/player/jw-player/player.swf new file mode 100755 index 0000000000000000000000000000000000000000..be4c9d958311e1019a5dd206ba1341c6f73dccb3 Binary files /dev/null and b/bbb-screenshare/jws/player/jw-player/player.swf differ diff --git a/bbb-screenshare/jws/player/screenshot.png b/bbb-screenshare/jws/player/screenshot.png new file mode 100755 index 0000000000000000000000000000000000000000..ab49abd5756bcd044fcaa5887cf4fc4aaf49a555 Binary files /dev/null and b/bbb-screenshare/jws/player/screenshot.png differ diff --git a/bbb-screenshare/jws/player/style.css b/bbb-screenshare/jws/player/style.css new file mode 100755 index 0000000000000000000000000000000000000000..da15278e751749190f31ffdfda3c9bae3c2866fa --- /dev/null +++ b/bbb-screenshare/jws/player/style.css @@ -0,0 +1,53 @@ +body { + margin: 0; + padding: 0; +} + +#menu { + border-bottom: 1px solid #bbb; + background: #eee; + padding: 20px; +} + +#menu form { + padding: 0; + margin: 0; +} + +.field { + clear: both; + display: block; + margin-bottom: 5px; +} + +.field2 { + clear: both; + display: block; + margin-bottom: 5px; + margin-left: 160px; +} + +.field .label { + width: 80px; + float: left; + text-align: right; + margin-right: 10px; + margin-top: 5px; +} + +.field input[type=text] { + width: 450px; +} + +.field button { + margin-left: 210px; +} + +#content { + padding: 30px; +} + +#wrapper { + border: 1px solid red; + padding: 5px; +} diff --git a/bbb-screenshare/jws/webstart/build.gradle b/bbb-screenshare/jws/webstart/build.gradle new file mode 100755 index 0000000000000000000000000000000000000000..f3273335b168e7f974920588c473a48452d0691f --- /dev/null +++ b/bbb-screenshare/jws/webstart/build.gradle @@ -0,0 +1,26 @@ +apply plugin: 'java' +apply plugin: 'eclipse' + +sourceCompatibility=1.6 +targetCompatibility=1.6 + +version = '0.0.1' +archivesBaseName = 'javacv-screenshare' + +repositories { + flatDir name: 'localRepo', dirs: "./lib" +} + +dependencies { + compile ":javacpp:@jar" + compile ":ffmpeg:@jar" +} + +jar { + manifest.mainAttributes("Permissions": "all-permissions") + manifest.mainAttributes("Codebase": "*") + manifest.mainAttributes("Application-Name": "BigBlueButton Screenshare") + manifest.mainAttributes("Application-Library-Allowable-Codebase": "*") + manifest.mainAttributes("Caller-Allowable-Codebase": "*") + manifest.mainAttributes("Trusted-Only": "true") +} diff --git a/bbb-screenshare/jws/webstart/build.sh b/bbb-screenshare/jws/webstart/build.sh new file mode 100755 index 0000000000000000000000000000000000000000..2be4ae9983076cf39cb935532122f6b308ba908e --- /dev/null +++ b/bbb-screenshare/jws/webstart/build.sh @@ -0,0 +1,5 @@ +cp ../../app/jws/lib/*.jar lib +gradle jar +ant sign-jar +cp build/libs/javacv-screenshare-0.0.1.jar ../../app/jws/lib/ + diff --git a/bbb-screenshare/jws/webstart/build.xml b/bbb-screenshare/jws/webstart/build.xml new file mode 100755 index 0000000000000000000000000000000000000000..6c1e3d55e48daf60090965ae984138b5be6576b2 --- /dev/null +++ b/bbb-screenshare/jws/webstart/build.xml @@ -0,0 +1,64 @@ +<?xml version="1.0" ?> +<project name="bbb-deskshare-applet" basedir="."> + <!-- How to sign the applet. From Ant in Action book --> + + <target name="get-password" depends="init-security" description="Prompts for password for keystore"> + <input addproperty="keystore.password" >password for keystore:</input> + <echo level="verbose">password = ${keystore.password}</echo> + </target> + + <target name="init-security"> + <property name="keystore.dir" location="${user.home}/.secret" /> + <mkdir dir="${keystore.dir}" /> + <chmod file="${keystore.dir}" perm="700"/> + <property name="keystore" + location="${keystore.dir}/local.keystore" /> + <property file="${keystore.dir}/keystore.properties" /> + <property name="keystore.alias" value="code.signer"/> + </target> + + <target name="create-signing-key" depends="get-password"> + <genkey alias="${keystore.alias}" keystore="${keystore}" storepass="${keystore.password}" validity="366" > + <dname> + <param name="CN" value="BigBlueButton"/> + <param name="OU" value="BigBlueButton Project"/> + <param name="O" value="BigBlueButton"/> + <param name="C" value="CA"/> + </dname> + </genkey> + </target> + + <!-- Sign jar with Certificate using pkcs12 file --> + <target name="check-certificate"> + <input message="Enter cetificate filename:" addproperty="cert.name" /> + <input message="Enter cetificate password:" addproperty="cert.password" /> + <exec executable="/usr/bin/keytool" outputproperty="cert.info"> + <arg line="-list" /> + <arg line="-storetype pkcs12" /> + <arg line="-keystore ${cert.name}" /> + <arg line="-storepass ${cert.password}" /> + <arg line="-v" /> + </exec> + </target> + + <target name="get-alias-name" depends="check-certificate"> + <script language="javascript"> + <![CDATA[ + // getting the value + info = project.getProperty("cert.info"); + alias = (info.match(/Alias name:(.*)/)[0]).replace("Alias name: ",""); + project.setProperty("cert.alias",alias); + ]]> + </script> + </target> + + + <target name="sign-jar" depends="get-alias-name"> + <signjar jar="build/libs/javacv-screenshare-0.0.1.jar" + storetype="pkcs12" + keystore="${cert.name}" + storepass="${cert.password}" + alias="${cert.alias}" /> + </target> + +</project> diff --git a/bbb-screenshare/jws/webstart/src/main/java/com/myjavatools/web/ClientHttpRequest.java b/bbb-screenshare/jws/webstart/src/main/java/com/myjavatools/web/ClientHttpRequest.java new file mode 100755 index 0000000000000000000000000000000000000000..bc553e209a47a59794d83b6e77c474f0b30a7a26 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/com/myjavatools/web/ClientHttpRequest.java @@ -0,0 +1,492 @@ +package com.myjavatools.web; + +import java.net.URLConnection; +import java.net.URL; +import java.io.IOException; +import java.util.HashMap; +import java.util.Map; +import java.io.File; +import java.io.InputStream; +import java.util.Random; +import java.io.OutputStream; +import java.io.FileInputStream; +import java.util.Iterator; + +/** + * <p>Title: Client HTTP Request class</p> + * <p>Description: this class helps to send POST HTTP requests with various form data, + * including files. Cookies can be added to be included in the request.</p> + * + * @author Vlad Patryshev + * @version 1.0 + */ +public class ClientHttpRequest { + public URLConnection connection; + OutputStream os = null; + Map cookies = new HashMap(); + + protected void connect() throws IOException { + if (os == null) os = connection.getOutputStream(); + } + + protected void write(char c) throws IOException { + connect(); + os.write(c); + } + + protected void write(String s) throws IOException { + connect(); + os.write(s.getBytes()); + } + + protected void newline() throws IOException { + connect(); + write("\r\n"); + } + + protected void writeln(String s) throws IOException { + connect(); + write(s); + newline(); + } + + private static Random random = new Random(); + + protected static String randomString() { + return Long.toString(random.nextLong(), 36); + } + + String boundary = "---------------------------" + randomString() + randomString() + randomString(); + + private void boundary() throws IOException { + write("--"); + write(boundary); + } + + /** + * Creates a new multipart POST HTTP request on a freshly opened URLConnection + * + * @param connection an already open URL connection + * @throws IOException + */ + public ClientHttpRequest(URLConnection connection) throws IOException { + this.connection = connection; + connection.setDoOutput(true); + connection.setRequestProperty("Content-Type", + "multipart/form-data; boundary=" + boundary); + } + + /** + * Creates a new multipart POST HTTP request for a specified URL + * + * @param url the URL to send request to + * @throws IOException + */ + public ClientHttpRequest(URL url) throws IOException { + this(url.openConnection()); + } + + /** + * Creates a new multipart POST HTTP request for a specified URL string + * + * @param urlString the string representation of the URL to send request to + * @throws IOException + */ + public ClientHttpRequest(String urlString) throws IOException { + this(new URL(urlString)); + } + + + private void postCookies() { + StringBuffer cookieList = new StringBuffer(); + + for (Iterator i = cookies.entrySet().iterator(); i.hasNext();) { + Map.Entry entry = (Map.Entry)(i.next()); + cookieList.append(entry.getKey().toString() + "=" + entry.getValue()); + + if (i.hasNext()) { + cookieList.append("; "); + } + } + if (cookieList.length() > 0) { + connection.setRequestProperty("Cookie", cookieList.toString()); + } + } + + /** + * adds a cookie to the requst + * @param name cookie name + * @param value cookie value + * @throws IOException + */ + public void setCookie(String name, String value) throws IOException { + cookies.put(name, value); + } + + /** + * adds cookies to the request + * @param cookies the cookie "name-to-value" map + * @throws IOException + */ + public void setCookies(Map cookies) throws IOException { + if (cookies == null) return; + this.cookies.putAll(cookies); + } + + /** + * adds cookies to the request + * @param cookies array of cookie names and values (cookies[2*i] is a name, cookies[2*i + 1] is a value) + * @throws IOException + */ + public void setCookies(String[] cookies) throws IOException { + if (cookies == null) return; + for (int i = 0; i < cookies.length - 1; i+=2) { + setCookie(cookies[i], cookies[i+1]); + } + } + + private void writeName(String name) throws IOException { + newline(); + write("Content-Disposition: form-data; name=\""); + write(name); + write('"'); + } + + /** + * adds a string parameter to the request + * @param name parameter name + * @param value parameter value + * @throws IOException + */ + public void setParameter(String name, String value) throws IOException { + boundary(); + writeName(name); + newline(); newline(); + writeln(value); + } + + private static void pipe(InputStream in, OutputStream out) throws IOException { + byte[] buf = new byte[500000]; + int nread; + int navailable; + int total = 0; + synchronized (in) { + while((nread = in.read(buf, 0, buf.length)) >= 0) { + out.write(buf, 0, nread); + total += nread; + } + } + out.flush(); + buf = null; + } + + /** + * adds a file parameter to the request + * @param name parameter name + * @param filename the name of the file + * @param is input stream to read the contents of the file from + * @throws IOException + */ + public void setParameter(String name, String filename, InputStream is) throws IOException { + boundary(); + writeName(name); + write("; filename=\""); + write(filename); + write('"'); + newline(); + write("Content-Type: "); + String type = connection.guessContentTypeFromName(filename); + if (type == null) type = "application/octet-stream"; + writeln(type); + newline(); + pipe(is, os); + newline(); + } + + /** + * adds a file parameter to the request + * @param name parameter name + * @param file the file to upload + * @throws IOException + */ + public void setParameter(String name, File file) throws IOException { + setParameter(name, file.getPath(), new FileInputStream(file)); + } + + /** + * adds a parameter to the request; if the parameter is a File, the file is uploaded, otherwise the string value of the parameter is passed in the request + * @param name parameter name + * @param object parameter value, a File or anything else that can be stringified + * @throws IOException + */ + public void setParameter(String name, Object object) throws IOException { + if (object instanceof File) { + setParameter(name, (File) object); + } else { + setParameter(name, object.toString()); + } + } + + /** + * adds parameters to the request + * @param parameters "name-to-value" map of parameters; if a value is a file, the file is uploaded, otherwise it is stringified and sent in the request + * @throws IOException + */ + public void setParameters(Map parameters) throws IOException { + if (parameters == null) return; + for (Iterator i = parameters.entrySet().iterator(); i.hasNext();) { + Map.Entry entry = (Map.Entry)i.next(); + setParameter(entry.getKey().toString(), entry.getValue()); + } + } + + /** + * adds parameters to the request + * @param parameters array of parameter names and values (parameters[2*i] is a name, parameters[2*i + 1] is a value); if a value is a file, the file is uploaded, otherwise it is stringified and sent in the request + * @throws IOException + */ + public void setParameters(Object[] parameters) throws IOException { + if (parameters == null) return; + for (int i = 0; i < parameters.length - 1; i+=2) { + setParameter(parameters[i].toString(), parameters[i+1]); + } + } + + /** + * posts the requests to the server, with all the cookies and parameters that were added + * @return input stream with the server response + * @throws IOException + */ + public InputStream post() throws IOException { + boundary(); + writeln("--"); + os.close(); + return connection.getInputStream(); + } + + /** + * posts the requests to the server, with all the cookies and parameters that were added before (if any), and with parameters that are passed in the argument + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + */ + public InputStream post(Map parameters) throws IOException { + setParameters(parameters); + return post(); + } + + /** + * posts the requests to the server, with all the cookies and parameters that were added before (if any), and with parameters that are passed in the argument + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + */ + public InputStream post(Object[] parameters) throws IOException { + setParameters(parameters); + return post(); + } + + /** + * posts the requests to the server, with all the cookies and parameters that were added before (if any), and with cookies and parameters that are passed in the arguments + * @param cookies request cookies + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + * @see setCookies + */ + public InputStream post(Map cookies, Map parameters) throws IOException { + setCookies(cookies); + setParameters(parameters); + return post(); + } + + /** + * posts the requests to the server, with all the cookies and parameters that were added before (if any), and with cookies and parameters that are passed in the arguments + * @param cookies request cookies + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + * @see setCookies + */ + public InputStream post(String[] cookies, Object[] parameters) throws IOException { + setCookies(cookies); + setParameters(parameters); + return post(); + } + + /** + * post the POST request to the server, with the specified parameter + * @param name parameter name + * @param value parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public InputStream post(String name, Object value) throws IOException { + setParameter(name, value); + return post(); + } + + /** + * post the POST request to the server, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public InputStream post(String name1, Object value1, String name2, Object value2) throws IOException { + setParameter(name1, value1); + return post(name2, value2); + } + + /** + * post the POST request to the server, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @param name3 third parameter name + * @param value3 third parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public InputStream post(String name1, Object value1, String name2, Object value2, String name3, Object value3) throws IOException { + setParameter(name1, value1); + return post(name2, value2, name3, value3); + } + + /** + * post the POST request to the server, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @param name3 third parameter name + * @param value3 third parameter value + * @param name4 fourth parameter name + * @param value4 fourth parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public InputStream post(String name1, Object value1, String name2, Object value2, String name3, Object value3, String name4, Object value4) throws IOException { + setParameter(name1, value1); + return post(name2, value2, name3, value3, name4, value4); + } + + /** + * posts a new request to specified URL, with parameters that are passed in the argument + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + */ + public static InputStream post(URL url, Map parameters) throws IOException { + return new ClientHttpRequest(url).post(parameters); + } + + /** + * posts a new request to specified URL, with parameters that are passed in the argument + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setParameters + */ + public static InputStream post(URL url, Object[] parameters) throws IOException { + return new ClientHttpRequest(url).post(parameters); + } + + /** + * posts a new request to specified URL, with cookies and parameters that are passed in the argument + * @param cookies request cookies + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setCookies + * @see setParameters + */ + public static InputStream post(URL url, Map cookies, Map parameters) throws IOException { + return new ClientHttpRequest(url).post(cookies, parameters); + } + + /** + * posts a new request to specified URL, with cookies and parameters that are passed in the argument + * @param cookies request cookies + * @param parameters request parameters + * @return input stream with the server response + * @throws IOException + * @see setCookies + * @see setParameters + */ + public static InputStream post(URL url, String[] cookies, Object[] parameters) throws IOException { + return new ClientHttpRequest(url).post(cookies, parameters); + } + + /** + * post the POST request specified URL, with the specified parameter + * @param name parameter name + * @param value parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public static InputStream post(URL url, String name1, Object value1) throws IOException { + return new ClientHttpRequest(url).post(name1, value1); + } + + /** + * post the POST request to specified URL, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public static InputStream post(URL url, String name1, Object value1, String name2, Object value2) throws IOException { + return new ClientHttpRequest(url).post(name1, value1, name2, value2); + } + + /** + * post the POST request to specified URL, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @param name3 third parameter name + * @param value3 third parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public static InputStream post(URL url, String name1, Object value1, String name2, Object value2, String name3, Object value3) throws IOException { + return new ClientHttpRequest(url).post(name1, value1, name2, value2, name3, value3); + } + + /** + * post the POST request to specified URL, with the specified parameters + * @param name1 first parameter name + * @param value1 first parameter value + * @param name2 second parameter name + * @param value2 second parameter value + * @param name3 third parameter name + * @param value3 third parameter value + * @param name4 fourth parameter name + * @param value4 fourth parameter value + * @return input stream with the server response + * @throws IOException + * @see setParameter + */ + public static InputStream post(URL url, String name1, Object value1, String name2, Object value2, String name3, Object value3, String name4, Object value4) throws IOException { + return new ClientHttpRequest(url).post(name1, value1, name2, value2, name3, value3, name4, value4); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/jargs/gnu/CmdLineParser.java b/bbb-screenshare/jws/webstart/src/main/java/jargs/gnu/CmdLineParser.java new file mode 100755 index 0000000000000000000000000000000000000000..cd429dd8156510d76da643411a13848a4eeb984d --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/jargs/gnu/CmdLineParser.java @@ -0,0 +1,525 @@ +package jargs.gnu; + +import java.text.NumberFormat; +import java.text.ParseException; +import java.util.Hashtable; +import java.util.Vector; +import java.util.Locale; + +/** + * Largely GNU-compatible command-line options parser. Has short (-v) and + * long-form (--verbose) option support, and also allows options with + * associated values (-d 2, --debug 2, --debug=2). Option processing + * can be explicitly terminated by the argument '--'. + * + * @author Steve Purcell + * @version $Revision$ + * @see jargs.examples.gnu.OptionTest + */ +public class CmdLineParser { + + /** + * Base class for exceptions that may be thrown when options are parsed + */ + public static abstract class OptionException extends Exception { + OptionException(String msg) { super(msg); } + } + + /** + * Thrown when the parsed command-line contains an option that is not + * recognised. <code>getMessage()</code> returns + * an error string suitable for reporting the error to the user (in + * English). + */ + public static class UnknownOptionException extends OptionException { + UnknownOptionException( String optionName ) { + this(optionName, "Unknown option '" + optionName + "'"); + } + + UnknownOptionException( String optionName, String msg ) { + super(msg); + this.optionName = optionName; + } + + /** + * @return the name of the option that was unknown (e.g. "-u") + */ + public String getOptionName() { return this.optionName; } + private String optionName = null; + } + + /** + * Thrown when the parsed commandline contains multiple concatenated + * short options, such as -abcd, where one is unknown. + * <code>getMessage()</code> returns an english human-readable error + * string. + * @author Vidar Holen + */ + public static class UnknownSuboptionException + extends UnknownOptionException { + private char suboption; + + UnknownSuboptionException( String option, char suboption ) { + super(option, "Illegal option: '"+suboption+"' in '"+option+"'"); + this.suboption=suboption; + } + public char getSuboption() { return suboption; } + } + + /** + * Thrown when the parsed commandline contains multiple concatenated + * short options, such as -abcd, where one or more requires a value. + * <code>getMessage()</code> returns an english human-readable error + * string. + * @author Vidar Holen + */ + public static class NotFlagException extends UnknownOptionException { + private char notflag; + + NotFlagException( String option, char unflaggish ) { + super(option, "Illegal option: '"+option+"', '"+ + unflaggish+"' requires a value"); + notflag=unflaggish; + } + + /** + * @return the first character which wasn't a boolean (e.g 'c') + */ + public char getOptionChar() { return notflag; } + } + + /** + * Thrown when an illegal or missing value is given by the user for + * an option that takes a value. <code>getMessage()</code> returns + * an error string suitable for reporting the error to the user (in + * English). + */ + public static class IllegalOptionValueException extends OptionException { + public IllegalOptionValueException( Option opt, String value ) { + super("Illegal value '" + value + "' for option " + + (opt.shortForm() != null ? "-" + opt.shortForm() + "/" : "") + + "--" + opt.longForm()); + this.option = opt; + this.value = value; + } + + /** + * @return the name of the option whose value was illegal (e.g. "-u") + */ + public Option getOption() { return this.option; } + + /** + * @return the illegal value + */ + public String getValue() { return this.value; } + private Option option; + private String value; + } + + /** + * Representation of a command-line option + */ + public static abstract class Option { + + protected Option( String longForm, boolean wantsValue ) { + this(null, longForm, wantsValue); + } + + protected Option( char shortForm, String longForm, + boolean wantsValue ) { + this(new String(new char[]{shortForm}), longForm, wantsValue); + } + + private Option( String shortForm, String longForm, boolean wantsValue ) { + if ( longForm == null ) + throw new IllegalArgumentException("Null longForm not allowed"); + this.shortForm = shortForm; + this.longForm = longForm; + this.wantsValue = wantsValue; + } + + public String shortForm() { return this.shortForm; } + + public String longForm() { return this.longForm; } + + /** + * Tells whether or not this option wants a value + */ + public boolean wantsValue() { return this.wantsValue; } + + public final Object getValue( String arg, Locale locale ) + throws IllegalOptionValueException { + if ( this.wantsValue ) { + if ( arg == null ) { + throw new IllegalOptionValueException(this, ""); + } + return this.parseValue(arg, locale); + } + else { + return Boolean.TRUE; + } + } + + /** + * Override to extract and convert an option value passed on the + * command-line + */ + protected Object parseValue( String arg, Locale locale ) + throws IllegalOptionValueException { + return null; + } + + private String shortForm = null; + private String longForm = null; + private boolean wantsValue = false; + + public static class BooleanOption extends Option { + public BooleanOption( char shortForm, String longForm ) { + super(shortForm, longForm, false); + } + public BooleanOption( String longForm ) { + super(longForm, false); + } + } + + /** + * An option that expects an integer value + */ + public static class IntegerOption extends Option { + public IntegerOption( char shortForm, String longForm ) { + super(shortForm, longForm, true); + } + public IntegerOption( String longForm ) { + super(longForm, true); + } + protected Object parseValue( String arg, Locale locale ) + throws IllegalOptionValueException { + try { + return new Integer(arg); + } + catch (NumberFormatException e) { + throw new IllegalOptionValueException(this, arg); + } + } + } + + /** + * An option that expects a long integer value + */ + public static class LongOption extends Option { + public LongOption( char shortForm, String longForm ) { + super(shortForm, longForm, true); + } + public LongOption( String longForm ) { + super(longForm, true); + } + protected Object parseValue( String arg, Locale locale ) + throws IllegalOptionValueException { + try { + return new Long(arg); + } + catch (NumberFormatException e) { + throw new IllegalOptionValueException(this, arg); + } + } + } + + /** + * An option that expects a floating-point value + */ + public static class DoubleOption extends Option { + public DoubleOption( char shortForm, String longForm ) { + super(shortForm, longForm, true); + } + public DoubleOption( String longForm ) { + super(longForm, true); + } + protected Object parseValue( String arg, Locale locale ) + throws IllegalOptionValueException { + try { + NumberFormat format = NumberFormat.getNumberInstance(locale); + Number num = (Number)format.parse(arg); + return new Double(num.doubleValue()); + } + catch (ParseException e) { + throw new IllegalOptionValueException(this, arg); + } + } + } + + /** + * An option that expects a string value + */ + public static class StringOption extends Option { + public StringOption( char shortForm, String longForm ) { + super(shortForm, longForm, true); + } + public StringOption( String longForm ) { + super(longForm, true); + } + protected Object parseValue( String arg, Locale locale ) { + return arg; + } + } + } + + /** + * Add the specified Option to the list of accepted options + */ + public final Option addOption( Option opt ) { + if ( opt.shortForm() != null ) + this.options.put("-" + opt.shortForm(), opt); + this.options.put("--" + opt.longForm(), opt); + return opt; + } + + /** + * Convenience method for adding a string option. + * @return the new Option + */ + public final Option addStringOption( char shortForm, String longForm ) { + return addOption(new Option.StringOption(shortForm, longForm)); + } + + /** + * Convenience method for adding a string option. + * @return the new Option + */ + public final Option addStringOption( String longForm ) { + return addOption(new Option.StringOption(longForm)); + } + + /** + * Convenience method for adding an integer option. + * @return the new Option + */ + public final Option addIntegerOption( char shortForm, String longForm ) { + return addOption(new Option.IntegerOption(shortForm, longForm)); + } + + /** + * Convenience method for adding an integer option. + * @return the new Option + */ + public final Option addIntegerOption( String longForm ) { + return addOption(new Option.IntegerOption(longForm)); + } + + /** + * Convenience method for adding a long integer option. + * @return the new Option + */ + public final Option addLongOption( char shortForm, String longForm ) { + return addOption(new Option.LongOption(shortForm, longForm)); + } + + /** + * Convenience method for adding a long integer option. + * @return the new Option + */ + public final Option addLongOption( String longForm ) { + return addOption(new Option.LongOption(longForm)); + } + + /** + * Convenience method for adding a double option. + * @return the new Option + */ + public final Option addDoubleOption( char shortForm, String longForm ) { + return addOption(new Option.DoubleOption(shortForm, longForm)); + } + + /** + * Convenience method for adding a double option. + * @return the new Option + */ + public final Option addDoubleOption( String longForm ) { + return addOption(new Option.DoubleOption(longForm)); + } + + /** + * Convenience method for adding a boolean option. + * @return the new Option + */ + public final Option addBooleanOption( char shortForm, String longForm ) { + return addOption(new Option.BooleanOption(shortForm, longForm)); + } + + /** + * Convenience method for adding a boolean option. + * @return the new Option + */ + public final Option addBooleanOption( String longForm ) { + return addOption(new Option.BooleanOption(longForm)); + } + + /** + * Equivalent to {@link #getOptionValue(Option, Object) getOptionValue(o, + * null)}. + */ + public final Object getOptionValue( Option o ) { + return getOptionValue(o, null); + } + + + /** + * @return the parsed value of the given Option, or the given default 'def' + * if the option was not set + */ + public final Object getOptionValue( Option o, Object def ) { + Vector v = (Vector)values.get(o.longForm()); + + if (v == null) { + return def; + } + else if (v.isEmpty()) { + return null; + } + else { + Object result = v.elementAt(0); + v.removeElementAt(0); + return result; + } + } + + + /** + * @return A Vector giving the parsed values of all the occurrences of the + * given Option, or an empty Vector if the option was not set. + */ + public final Vector getOptionValues( Option option ) { + Vector result = new Vector(); + + while (true) { + Object o = getOptionValue(option, null); + + if (o == null) { + return result; + } + else { + result.addElement(o); + } + } + } + + + /** + * @return the non-option arguments + */ + public final String[] getRemainingArgs() { + return this.remainingArgs; + } + + /** + * Extract the options and non-option arguments from the given + * list of command-line arguments. The default locale is used for + * parsing options whose values might be locale-specific. + */ + public final void parse( String[] argv ) + throws IllegalOptionValueException, UnknownOptionException { + + // It would be best if this method only threw OptionException, but for + // backwards compatibility with old user code we throw the two + // exceptions above instead. + + parse(argv, Locale.getDefault()); + } + + /** + * Extract the options and non-option arguments from the given + * list of command-line arguments. The specified locale is used for + * parsing options whose values might be locale-specific. + */ + public final void parse( String[] argv, Locale locale ) + throws IllegalOptionValueException, UnknownOptionException { + + // It would be best if this method only threw OptionException, but for + // backwards compatibility with old user code we throw the two + // exceptions above instead. + + Vector otherArgs = new Vector(); + int position = 0; + this.values = new Hashtable(10); + while ( position < argv.length ) { + String curArg = argv[position]; + if ( curArg.startsWith("-") ) { + if ( curArg.equals("--") ) { // end of options + position += 1; + break; + } + String valueArg = null; + if ( curArg.startsWith("--") ) { // handle --arg=value + int equalsPos = curArg.indexOf("="); + if ( equalsPos != -1 ) { + valueArg = curArg.substring(equalsPos+1); + curArg = curArg.substring(0,equalsPos); + } + } else if(curArg.length() > 2) { // handle -abcd + for(int i=1; i<curArg.length(); i++) { + Option opt=(Option)this.options.get + ("-"+curArg.charAt(i)); + if(opt==null) throw new + UnknownSuboptionException(curArg,curArg.charAt(i)); + if(opt.wantsValue()) throw new + NotFlagException(curArg,curArg.charAt(i)); + addValue(opt, opt.getValue(null,locale)); + + } + position++; + continue; + } + + Option opt = (Option)this.options.get(curArg); + if ( opt == null ) { + throw new UnknownOptionException(curArg); + } + Object value = null; + if ( opt.wantsValue() ) { + if ( valueArg == null ) { + position += 1; + if ( position < argv.length ) { + valueArg = argv[position]; + } + } + value = opt.getValue(valueArg, locale); + } + else { + value = opt.getValue(null, locale); + } + + addValue(opt, value); + + position += 1; + } + else { + otherArgs.addElement(curArg); + position += 1; + } + } + for ( ; position < argv.length; ++position ) { + otherArgs.addElement(argv[position]); + } + + this.remainingArgs = new String[otherArgs.size()]; + otherArgs.copyInto(remainingArgs); + } + + + private void addValue(Option opt, Object value) { + String lf = opt.longForm(); + + Vector v = (Vector)values.get(lf); + + if (v == null) { + v = new Vector(); + values.put(lf, v); + } + + v.addElement(value); + } + + + private String[] remainingArgs = null; + private Hashtable options = new Hashtable(10); + private Hashtable values = new Hashtable(10); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ClientListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ClientListener.java new file mode 100755 index 0000000000000000000000000000000000000000..aa216809a1b0c1775f1105c8999667e2b2238490 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ClientListener.java @@ -0,0 +1,23 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public interface ClientListener { + public void onClientStop(ExitCode reason); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskShareApplet.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskShareApplet.java new file mode 100755 index 0000000000000000000000000000000000000000..f76e5c824a466b96b47a1c7a11f5807010f78683 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskShareApplet.java @@ -0,0 +1,253 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import static org.bytedeco.javacpp.avformat.av_register_all; +import static org.bytedeco.javacpp.avformat.avformat_network_init; + +import javax.imageio.ImageIO; +import javax.swing.JApplet; +import javax.swing.JFrame; +import javax.swing.JOptionPane; + +import org.bigbluebutton.screenshare.client.javacv.BBBFFmpegFrameRecorder; +import org.bigbluebutton.screenshare.client.javacv.BBBFrameRecorder.Exception; +import org.bytedeco.javacpp.Loader; + +import java.io.IOException; +import java.net.URL; +import java.security.*; +import java.awt.Image; + +public class DeskShareApplet extends JApplet implements ClientListener { + public static final String NAME = "DESKSHAREAPPLET: "; + + private static final long serialVersionUID = 1L; + + String hostValue = "localhost"; + String minJreVersion = "1.7.0_51"; + Integer portValue = new Integer(9123); + String roomValue = "85115"; + Integer cWidthValue = new Integer(800); + Integer cHeightValue = new Integer(600); + Integer sWidthValue = new Integer(800); + Integer sHeightValue = new Integer(600); + Double scale = new Double(0.8); + Boolean qualityValue = false; + Boolean aspectRatioValue = false; + Integer xValue = new Integer(0); + Integer yValue = new Integer(0); + Boolean tunnelValue = true; + Boolean fullScreenValue = false; + String url = "rtmp://192.168.23.23/live/foo/room2"; + DeskshareClient client; + Image icon; + + public boolean isSharing = false; + private volatile boolean clientStarted = false; + private final static String VERSION_ERROR_MSG = "You have an unsupported Java version."; + + private static Exception loadingException = null; + + public static void tryLoad() throws Exception { + if (loadingException != null) { + throw loadingException; + } else { + try { + Loader.load(org.bytedeco.javacpp.avutil.class); + Loader.load(org.bytedeco.javacpp.swresample.class); + Loader.load(org.bytedeco.javacpp.avcodec.class); + Loader.load(org.bytedeco.javacpp.avformat.class); + Loader.load(org.bytedeco.javacpp.swscale.class); + + /* initialize libavcodec, and register all codecs and formats */ + av_register_all(); + avformat_network_init(); + } catch (Throwable t) { + if (t instanceof Exception) { + throw loadingException = (Exception)t; + } else { + throw loadingException = new Exception("Failed to load " + BBBFFmpegFrameRecorder.class, t); + } + } + } + } + + static { + try { + tryLoad(); + } catch (Exception ex) { } + } + + private class DestroyJob implements PrivilegedExceptionAction { + public Object run() throws Exception { + System.out.println("Desktop Sharing Applet Destroy"); + if (clientStarted) { + client.stop(); + } + return null; + } + } + + @Override + public void init() { + + System.out.println("Desktop Sharing Applet Initializing"); + + String javaVersion = getParameter("JavaVersion"); + if (javaVersion != null && javaVersion != "") minJreVersion = javaVersion; + + hostValue = getParameter("IP"); + String port = getParameter("PORT"); + if (port != null) portValue = Integer.parseInt(port); + roomValue = getParameter("ROOM"); + + String scaleValue = getParameter("SCALE"); + if (scaleValue != null) scale = Double.parseDouble(scaleValue); + + + String captureFullScreen = getParameter("FULL_SCREEN"); + if (captureFullScreen != null) fullScreenValue = Boolean.parseBoolean(captureFullScreen); + + String rtmpURL = getParameter("rtmpURL"); + if (rtmpURL != null) url = rtmpURL; + + String tunnel = getParameter("HTTP_TUNNEL"); + if (tunnel != null) tunnelValue = Boolean.parseBoolean(tunnel); + try { + URL iconUrl = new URL(getCodeBase(), "bbb.gif"); + icon = ImageIO.read(iconUrl); + } catch (IOException e) { + } + + displaySystemProperties(); + + + } + + + private void displaySystemProperties() { + System.out.println("Java temp dir : " + System.getProperty("java.io.tmpdir")); + System.out.println("Java name : " + System.getProperty("java.vm.name")); + System.out.println("OS name : " + System.getProperty("os.name")); + System.out.println("OS arch : " + System.getProperty("os.arch")); + System.out.println("JNA Path : " + System.getProperty("jna.library.path")); + } + + private String getJavaVersionRuntime() { + return System.getProperty("java.version"); + } + + /** + * Create the GUI and show it. For thread safety, + * this method should be invoked from the + * event-dispatching thread. + */ + private void createAndShowGUI(final String warning) { + JOptionPane.showMessageDialog(null, + warning, + "Java Version Error", + JOptionPane.ERROR_MESSAGE); + stop(); + } + + private void displayJavaWarning(final String warning) { + //Schedule a job for the event-dispatching thread: + //creating and showing this application's GUI. + javax.swing.SwingUtilities.invokeLater(new Runnable() { + public void run() { + createAndShowGUI(warning); + } + }); + } + + @Override + public void start() { + System.out.println("Desktop Sharing Applet Starting"); + super.start(); + String javaRuntimeVersion = getJavaVersionRuntime(); + System.out.println("**** JAVA VERSION = [" + javaRuntimeVersion + "]"); + + allowDesktopSharing(); + } + + + private void allowDesktopSharing() { + client = new DeskshareClient.NewBuilder().host(hostValue).port(portValue) + .meetingId(roomValue).captureWidth(cWidthValue) + .captureHeight(cHeightValue).scaleWidth(sWidthValue).scaleHeight(sHeightValue) + .quality(qualityValue).autoScale(scale) + .x(xValue).y(yValue).fullScreen(fullScreenValue).withURL(url) + .httpTunnel(tunnelValue).trayIcon(icon).enableTrayIconActions(false).build(); + client.addClientListener(this); + + clientStarted = true; + + client.start(); + } + + + @Override + public void destroy() { + /* We make this a privileged job. + * The privileges of the javascript code are 'anded' with the + * java privs. Sometimes (depending on jre version, browser, etc.) + * javascript will not have the privs to do some of the operations + * required for destroy, particularly network related activities, + * but java does. So we make sure here that we run only considering + * java privs, not javascript's. This should be 'security safe', since + * we are only shutting things down. + */ + try { + AccessController.doPrivileged( this.new DestroyJob() ); + } catch ( PrivilegedActionException e) { + System.out.println("Exception during Desktop Sharing Applet Stopping"+e.toString()); + UncheckedExceptions.spit((Exception) e.getException()); + } + super.destroy(); + } + + @Override + public void stop() { + System.out.println("Desktop Sharing Applet Stopping"); + if (clientStarted) { + client.stop(); + } + + super.stop(); + } + + public void onClientStop(ExitCode reason) { + // determine if client is disconnected _PTS_272_ + if ( ExitCode.CONNECTION_TO_DESKSHARE_SERVER_DROPPED == reason ){ + JFrame pframe = new JFrame("Desktop Sharing Disconneted"); + if ( null != pframe ){ + client.disconnected(); + JOptionPane.showMessageDialog(pframe, + "Disconnected. Reason: Lost connection to the server." + reason , + "Disconnected" ,JOptionPane.ERROR_MESSAGE ); + }else{ + System.out.println("Desktop sharing allocate memory failed."); + } + }else{ + client.stop(); + } + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareClient.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareClient.java new file mode 100755 index 0000000000000000000000000000000000000000..1b4c3cb217c0ff37e15490d8e8aa2f0d8eccecb0 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareClient.java @@ -0,0 +1,266 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.Image; +import java.awt.Toolkit; + +public class DeskshareClient { + public static final String NAME = "DESKSHARECLIENT: "; + + private ScreenShareInfo ssi; + private ClientListener listener; + private ScreenSharer screenSharer; + + public void addClientListener(ClientListener l) { + listener = l; + } + + public void start() { + if (ssi.fullScreen) { + System.out.println(NAME + "Sharing full screen."); + shareFullScreen(); + } else { + System.out.println(NAME + "Sharing region of screen."); + shareWithFrame(); + } + } + + private void shareWithFrame() { + screenSharer = new ScreenRegionSharer(ssi); + screenSharer.addClientListener(listener); + screenSharer.start(false); + } + + private void shareFullScreen() { + screenSharer = new ScreenRegionSharer(ssi); + screenSharer.addClientListener(listener); + screenSharer.start(true); + } + + public void disconnected(){ + System.out.println(NAME + "Disconneted"); + screenSharer.disconnected(); + } + + public void stop() { + System.out.println(NAME + "Stop"); + screenSharer.stop(); + } + + private DeskshareClient(ScreenShareInfo ssi) { + this.ssi = ssi; + } + + + /******************************************** + * Helper class + ********************************************/ + + /** + * Builds the Deskstop Sharing Client. + */ + public static class NewBuilder { + private String host = "localhost"; + private int port = 9123; + private String meetingId = "default-room"; + private String streamId = ""; + private String codecOptions = ""; + private int captureWidth = 0; + private int captureHeight = 0; + private int scaleWidth = 0; + private int scaleHeight = 0; + private boolean quality = false; + private double scale = 1; + private int x = -1; + private int y = -1; + private boolean httpTunnel = true; + private Image sysTrayIcon; + private boolean enableTrayActions = false; + private boolean fullScreen = false; + private String URL = "rtmp://192.168.23.23/live/foo/room2"; + + public NewBuilder host(String host) { + this.host = host; + return this; + } + + public NewBuilder port(int port) { + this.port = port; + return this; + } + + public NewBuilder meetingId(String meetingId) { + this.meetingId = meetingId; + return this; + } + + public NewBuilder streamId(String streamId) { + this.streamId = streamId; + return this; + } + + public NewBuilder codecOptions(String options) { + this.codecOptions = options; + return this; + } + + public NewBuilder captureWidth(int width) { + this.captureWidth = width; + return this; + } + + public NewBuilder captureHeight(int height) { + this.captureHeight = height; + return this; + } + + public NewBuilder scaleWidth(int width) { + this.scaleWidth = width; + return this; + } + + public NewBuilder scaleHeight(int height) { + this.scaleHeight = height; + return this; + } + + public NewBuilder quality(boolean quality) { + this.quality = quality; + return this; + } + + public NewBuilder autoScale(double scaleTo) { + this.scale = scaleTo; + return this; + } + + public NewBuilder x(int x) { + this.x = x; + return this; + } + + public NewBuilder y(int y) { + this.y = y; + return this; + } + + public NewBuilder httpTunnel(boolean httpTunnel) { + this.httpTunnel = httpTunnel; + return this; + } + + public NewBuilder fullScreen(boolean fullScreen) { + this.fullScreen = fullScreen; + return this; + } + + public NewBuilder withURL(String url) { + this.URL = url; + return this; + } + + public NewBuilder trayIcon(Image icon) { + this.sysTrayIcon = icon; + return this; + } + + public NewBuilder enableTrayIconActions(boolean enableActions) { + enableTrayActions = enableActions; + return this; + } + + public DeskshareClient build() { + if (fullScreen) { + System.out.println("Sharing full screen."); + setupFullScreen(); + } else { + System.out.println("Sharing region screen."); + setupCaptureRegion(); + } + + ScreenShareInfo ssi = new ScreenShareInfo(); + ssi.host = host; + ssi.port = port; + ssi.meetingId = meetingId; + ssi.streamId = streamId; + ssi.captureWidth = captureWidth; + ssi.captureHeight = captureHeight; + ssi.scaleWidth = scaleWidth; + ssi.scaleHeight = scaleHeight; + ssi.quality = quality; + ssi.scale = scale; + ssi.x = x; + ssi.y = y; + ssi.httpTunnel = httpTunnel; + ssi.fullScreen = fullScreen; + ssi.URL = URL; + ssi.codecOptions = codecOptions; + ssi.sysTrayIcon = sysTrayIcon; + ssi.enableTrayActions = enableTrayActions; + + System.out.println("ScreenShareInfo[captureWidth=" + captureWidth + ",captureHeight=" + captureHeight + "][" + x + "," + y +"]" + + "[scaleWidth=" + scaleWidth + ",scaleHeight=" + scaleHeight + "]"); + + return new DeskshareClient(ssi); + } + + private void setupCaptureRegion() { + if (captureWidth > 0 && captureHeight > 0) { + java.awt.Dimension fullScreenSize = Toolkit.getDefaultToolkit().getScreenSize(); + x = ((int) fullScreenSize.getWidth() - captureWidth) / 2; + y = ((int) fullScreenSize.getHeight() - captureHeight) / 2; + System.out.println("Info[" + captureWidth + "," + captureHeight + "][" + x + "," + y +"]" + + "[" + fullScreenSize.getWidth() + "," + fullScreenSize.getHeight() + "]"); + scaleWidth = captureWidth; + scaleHeight = captureHeight; + } + } + + private void setupFullScreen() { + java.awt.Dimension fullScreenSize = Toolkit.getDefaultToolkit().getScreenSize(); + captureWidth = (int) fullScreenSize.getWidth(); + captureHeight = (int) fullScreenSize.getHeight(); + + x = 0; + y = 0; + + if (scale > 0 && scale <= 0.8) { + scaleWidth = (int)(scale * (double)captureWidth); + scaleHeight = (int)(scale * (double)captureHeight); + } + + System.out.println("Check for scaling[" + captureWidth + "," + captureHeight +"][" + scaleWidth + "," + scaleHeight + "]"); + + if (scale == 1) { + scaleWidth = captureWidth; + scaleHeight = captureHeight; + } else { + if (scaleWidth > 1280) { + scaleWidth = 1280; + double ratio = (double)captureHeight/(double)captureWidth; + scaleHeight = (int)((double)scaleWidth * ratio); + System.out.println("Scaling[" + captureWidth + "," + captureHeight +"][" + scaleWidth + "," + scaleHeight + "]"); + } + } + + } + + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareMain.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareMain.java new file mode 100755 index 0000000000000000000000000000000000000000..9400f79c0ee9c0aa91314813a67b7211476bd740 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareMain.java @@ -0,0 +1,233 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import jargs.gnu.CmdLineParser; +import jargs.gnu.CmdLineParser.Option; + +import java.awt.Image; +import java.awt.Toolkit; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Iterator; +import java.util.List; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.LinkedBlockingQueue; +import javax.imageio.ImageIO; +import javax.swing.JOptionPane; + +public class DeskshareMain implements ClientListener, LifeLineListener { + private final BlockingQueue<ExitCode> exitReasonQ = new LinkedBlockingQueue<ExitCode>(5); + + private List<String> optionHelpStrings = new ArrayList<String>(); + private static LifeLine lifeline; + private static DeskshareClient client; + + private Option addHelp(Option option, String helpString) { + optionHelpStrings.add(" -" + option.shortForm() + ", --" + option.longForm() + ": " + helpString); + return option; + } + + private void printUsage() { + System.err.println("usage: deskshare [options]"); + for (Iterator<String> i = optionHelpStrings.iterator(); i.hasNext(); ) { + System.err.println(i.next()); + } + } + + public static void main(String[] args) { + DeskshareMain dsMain = new DeskshareMain(); + CmdLineParser parser = new CmdLineParser(); + + CmdLineParser.Option host = dsMain.addHelp(parser.addStringOption('s', "server"), serverHelpText); + CmdLineParser.Option port = dsMain.addHelp(parser.addIntegerOption('p', "port"),"The port the application is listening"); + CmdLineParser.Option listenPort = dsMain.addHelp(parser.addIntegerOption('l', "listenPort"),"Port to listen for lifeline"); + CmdLineParser.Option room = dsMain.addHelp(parser.addStringOption('r', "room"),"Room"); + CmdLineParser.Option cWidth = dsMain.addHelp(parser.addIntegerOption('w', "captureWidth"),"Width of the screen capture"); + CmdLineParser.Option cHeight = dsMain.addHelp(parser.addIntegerOption('t', "captureHeight"),"Height of the screen capture"); + CmdLineParser.Option sWidth = dsMain.addHelp(parser.addIntegerOption('d', "scaleWidth"),"Scale capture width"); + CmdLineParser.Option sHeight = dsMain.addHelp(parser.addIntegerOption('g', "scaleHeight"),"Scale capture height"); + CmdLineParser.Option xCoord = dsMain.addHelp(parser.addIntegerOption('x', "x"),"Upper-left x coordinate of the screen capture"); + CmdLineParser.Option yCoord = dsMain.addHelp(parser.addIntegerOption('y', "y"),"Upper-left y coordinate of the screen capture"); + CmdLineParser.Option tryHttpTunnel = dsMain.addHelp(parser.addBooleanOption('n', "httptunnel"),"Http tunnel if direct connection fails"); + CmdLineParser.Option icon = dsMain.addHelp(parser.addStringOption('i', "icon"),"Path to system tray icon file"); + CmdLineParser.Option help = dsMain.addHelp(parser.addBooleanOption('h', "help"),"Show this help message"); + CmdLineParser.Option fullScreen = dsMain.addHelp(parser.addBooleanOption('f', "full-screen"),"Capture the full screen."); + + + try { + parser.parse(args); + } catch (CmdLineParser.OptionException e) { + System.err.println(e.getMessage()); + dsMain.printUsage(); + System.exit(2); + } + + if (Boolean.TRUE.equals(parser.getOptionValue(help))) { + dsMain.printUsage(); + System.exit(0); + } + + // Extract the values entered for the various options -- if the + // options were not specified, the corresponding values will be + // the default. + Integer portValue = (Integer)parser.getOptionValue(port, new Integer(9123)); + Integer listenPortValue = (Integer)parser.getOptionValue(listenPort, new Integer(9125)); + Integer cWidthValue = (Integer)parser.getOptionValue(cWidth, new Integer(801)); + Integer cHeightValue = (Integer)parser.getOptionValue(cHeight, new Integer(601)); + + Integer sWidthValue = (Integer)parser.getOptionValue(sWidth, new Integer(800)); + Integer sHeightValue = (Integer)parser.getOptionValue(sHeight, new Integer(600)); + + Integer xValue = (Integer)parser.getOptionValue(xCoord, new Integer(0)); + Integer yValue = (Integer)parser.getOptionValue(yCoord, new Integer(0)); + Boolean tunnelValue = (Boolean)parser.getOptionValue(tryHttpTunnel, new Boolean(false)); + String iconValue = (String)parser.getOptionValue(icon, ""); + + String url = null; + String meetingId = null; + String streamId = null; + String serverUrl = null; + Boolean captureFullScreen = false; + String codecOptions = null; + + if(args != null && args.length == 7) { + System.out.println("Using passed args: length=[" + args.length + "]"); + url = args[0]; + serverUrl = args[1]; + meetingId = args[2]; + streamId = args[3]; + captureFullScreen = Boolean.parseBoolean(args[4]); + + System.out.println("Using passed args: [" + url + "] meetingId=[" + meetingId + "] streamId=[" + streamId + "] captureFullScreen=" + captureFullScreen); + codecOptions = args[5]; + + String errorMessage = args[6]; + + if (! errorMessage.equalsIgnoreCase("NO_ERRORS")) { + dsMain.displayJavaWarning(errorMessage); + } else { + Image image = null; + if (iconValue.isEmpty()) { + try { + image = ImageIO.read(dsMain.getClass().getResourceAsStream("/images/bbb.gif")); + } catch (IOException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } else { + image = Toolkit.getDefaultToolkit().getImage("bbb.gif"); + } + + + dsMain.displaySystemProperties(); + + lifeline = new LifeLine(listenPortValue.intValue(), dsMain); + lifeline.listen(); + + client = new DeskshareClient.NewBuilder().host(serverUrl).port(portValue) + .meetingId(meetingId).streamId(streamId).captureWidth(cWidthValue) + .captureHeight(cHeightValue).scaleWidth(sWidthValue).scaleHeight(sHeightValue) + .quality(true).autoScale(0).codecOptions(codecOptions) + .x(xValue).y(yValue).fullScreen(captureFullScreen).withURL(url) + .httpTunnel(tunnelValue).trayIcon(image).enableTrayIconActions(true).build(); + + client.addClientListener(dsMain); + client.start(); + + try { + System.out.println("Waiting for trigger to Stop client."); + ExitCode reason = dsMain.exitReasonQ.take(); + System.out.println("Stopping Java Web Start."); + client.stop(); + lifeline.disconnect(); + System.exit(reason.getExitCode()); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + System.exit(500); + } + } + + } else { + System.out.println("Using default args: [" + url + "] width=[" + cWidthValue + "] height=[" + cHeightValue + "]"); + System.out.println("args null =[" + (args == null) + "] args.length=[" + args.length + "]"); + dsMain.displayJavaWarning("Invalid number of arguments."); + } + + } + + + /** + * Create the GUI and show it. For thread safety, + * this method should be invoked from the + * event-dispatching thread. + */ + private void createAndShowGUI(final String warning) { + JOptionPane.showMessageDialog(null, + warning, + "Java Version Error", + JOptionPane.ERROR_MESSAGE); + } + + private void displayJavaWarning(final String warning) { + //Schedule a job for the event-dispatching thread: + //creating and showing this application's GUI. + javax.swing.SwingUtilities.invokeLater(new Runnable() { + public void run() { + createAndShowGUI(warning); + } + }); + } + + private void displaySystemProperties() { + System.out.println("========== SYSTEM PROPERTIES ================ "); + System.out.println("Java temp dir : " + System.getProperty("java.io.tmpdir")); + System.out.println("Java name : " + System.getProperty("java.vm.name")); + System.out.println("OS name : " + System.getProperty("os.name")); + System.out.println("OS arch : " + System.getProperty("os.arch")); + System.out.println("JNA Path : " + System.getProperty("jna.library.path")); + System.out.println("========== END SYSTEM PROPERTIES ================ "); + } + + public void onClientStop(ExitCode reason) { + queueExitCode(reason); + } + + @Override + public void disconnected(ExitCode reason) { + queueExitCode(reason); + } + + private void queueExitCode(ExitCode reason) { + try { + // System.out.println("Trigger stop client ." + exitReasonQ.remainingCapacity()); + exitReasonQ.put(reason); + System.out.println("Triggered stop client."); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + client.stop(); + lifeline.disconnect(); + System.exit(reason.getExitCode()); + } + } + + private static final String serverHelpText = "\n\t The host or IP of the desktop sharing server. Default is localhost."; +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareShell.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareShell.java new file mode 100755 index 0000000000000000000000000000000000000000..4d79276be55b8c89f6492d9c8c0fc72cff79b66b --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareShell.java @@ -0,0 +1,47 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.io.IOException; + +/* + * An application that calls the VBScript to test if the application is returning the exit codes. + */ +public class DeskshareShell { + + public static void main(String[] args) { + String COMMAND = "wscript deskshare.vbs -s 192.168.0.120 -r 6e87dfef-9f08-4f80-993f-c0ef5f7b999b"; + + Process process; + try { + process = Runtime.getRuntime().exec(COMMAND); + // Wait for the process to finish. + int exitValue = process.waitFor(); + System.out.println("Exit Value " + exitValue + " while for " + COMMAND); + if (exitValue != 0) { + System.out.println("Exit Value != 0 while for " + COMMAND); + } + } catch (IOException e) { + System.out.println("IOException while processing " + COMMAND); + } catch (InterruptedException e) { + System.out.println("InterruptedException while processing " + COMMAND); + } + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareSystemTray.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareSystemTray.java new file mode 100755 index 0000000000000000000000000000000000000000..f1dd6bab72fb72683f2377c2d4d7c757cdbd2181 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/DeskshareSystemTray.java @@ -0,0 +1,121 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.*; +import java.awt.event.*; + +public class DeskshareSystemTray { + private SystemTrayListener listener; + private TrayIcon trayIcon = null; + private SystemTray tray = null; + + public void addSystemTrayListener(SystemTrayListener l) { + listener = l; + } + + public void displayIconOnSystemTray(final Image image, final boolean enableActions) { + Runnable runner = new Runnable() { + public void run() { + if (SystemTray.isSupported()) { + tray = SystemTray.getSystemTray(); + + PopupMenu popup = new PopupMenu(); + trayIcon = new TrayIcon(image, "Sharing Desktop", popup); + + if (enableActions) { + MenuItem stopItem = new MenuItem("Stop Sharing"); + stopItem.addActionListener(new StopSharingListener( + trayIcon, "Stop Desktop Sharing", "Stop sharing your desktop", TrayIcon.MessageType.INFO)); + popup.add(stopItem); + } + + + try { + tray.add(trayIcon); + trayIcon.displayMessage("Sharing Desktop", "You are now sharing your desktop", TrayIcon.MessageType.INFO); + } catch (AWTException e) { + System.err.println("Can't add to tray"); + } + } else { + System.err.println("Tray unavailable"); + } + } + }; + EventQueue.invokeLater(runner); + } + + /***************************************************************************** + ; disconnectIconSystemTrayMessage + ;---------------------------------------------------------------------------- + ; DESCRIPTION + ; This routine is used to change icon system tray message string + ; to disconnect. + ; + ; RETURNS : N/A + ; + ; INTERFACE NOTES + ; + ; INPUT : N/A + ; + ; OUTPUT : N/A + ; + ; IMPLEMENTATION + ; + ; HISTORY + ; __date__ : PTS: + ; 2010.11.19 problem 272 + ; + ******************************************************************************/ + public void disconnectIconSystemTrayMessage(){ + trayIcon.setToolTip("Disconnected"); + trayIcon.displayMessage("Deskshare Disconnected" , + "You're disconnected from desktop sharing", + TrayIcon.MessageType.ERROR); + } // END FUNCTION disconnectIconSystemTrayMessage + + public void removeIconFromSystemTray() { + if (tray != null && trayIcon != null) { + tray.remove(trayIcon); + } + } + + class StopSharingListener implements ActionListener { + TrayIcon trayIcon; + String title; + String message; + TrayIcon.MessageType messageType; + + StopSharingListener(TrayIcon trayIcon, String title, + String message, TrayIcon.MessageType messageType) { + this.trayIcon = trayIcon; + this.title = title; + this.message = message; + this.messageType = messageType; + } + + public void actionPerformed(ActionEvent e) { + trayIcon.displayMessage(title, message, messageType); + if (listener != null) { + listener.onStopSharingSysTrayMenuClicked(); + } + } + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ExitCode.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ExitCode.java new file mode 100755 index 0000000000000000000000000000000000000000..c83c6f119aa140e70f99c61126472ab0d892c735 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ExitCode.java @@ -0,0 +1,37 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public enum ExitCode { + NORMAL(0), + BAD_PARAMETERS(400), + CANNOT_CONNECT_TO_LIFELINE(401), + ERROR_ON_LIFELINE_CONNECTION(402), + CONNECTION_TO_DESKSHARE_SERVER_DROPPED(403), + CANNOT_BIND_TO_LIFELINE_PORT(404), + LIFELINE_CONNECTION_CLOSED(405), + INTERNAL_ERROR(500), + DESKSHARE_SERVICE_UNAVAILABLE(503); + + private final int exitValue; + + ExitCode(int code) { this.exitValue = code; } + + public int getExitCode() {return exitValue;} +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLine.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLine.java new file mode 100755 index 0000000000000000000000000000000000000000..c8e6aced7ec56e9da5e10c0c42ec9e888a90f8d5 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLine.java @@ -0,0 +1,100 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.net.*; +import java.io.*; + +public class LifeLine { + + private int port; + private ServerSocket serverSocket = null; + private Socket clientSocket = null; + private boolean connected = false; + private PrintWriter out = null; + private BufferedReader in = null; + private LifeLineListener listener = null; + private Thread lifeLineThread; + private LifeLineServer lifeLineServer; + + public LifeLine(int port, LifeLineListener listener) { + this.port = port; + this.listener = listener; + } + + public void listen() { + + lifeLineServer = new LifeLineServer(); + lifeLineThread = new Thread(lifeLineServer, "LifeLineServer"); + lifeLineThread.start(); + } + + public void disconnect() { + lifeLineServer.close(); + } + + private void notifyListener(ExitCode reason) { + if (listener != null) listener.disconnected(reason); + } + + private class LifeLineServer implements Runnable { + @Override + public void run() { + try { + serverSocket = new ServerSocket(); + serverSocket.bind(new InetSocketAddress("127.0.0.1", port)); + } catch (IOException e) { + System.err.println("Could not listen on port: " + port); + notifyListener(ExitCode.CANNOT_BIND_TO_LIFELINE_PORT); + } + + try { + System.out.println("Starting listener on [" + serverSocket.getInetAddress() + ":" + port + "]"); + clientSocket = serverSocket.accept(); + clientSocket.setKeepAlive(true); + out = new PrintWriter(clientSocket.getOutputStream(), true); + in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream())); + String inputLine; + connected = true; + while ((inputLine = in.readLine()) != null) { + // do nothing + } + } catch (IOException e) { + System.out.println("IOException listener"); + System.err.println("Accept failed."); + notifyListener(ExitCode.ERROR_ON_LIFELINE_CONNECTION); + } + close(); + System.out.println("Stopped listener"); + notifyListener(ExitCode.LIFELINE_CONNECTION_CLOSED); + } + + public void close() { + try { + if (out != null) out.close(); + if (in != null) in.close(); + if (clientSocket != null) clientSocket.close(); + if (serverSocket != null) serverSocket.close(); + } catch (IOException e) { + e.printStackTrace(); + } + } + + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLineListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLineListener.java new file mode 100755 index 0000000000000000000000000000000000000000..9b7ab26130e1b4176b87d98d920194621ea25491 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/LifeLineListener.java @@ -0,0 +1,24 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public interface LifeLineListener { + + public void disconnected(ExitCode reason); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCapture.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCapture.java new file mode 100755 index 0000000000000000000000000000000000000000..a148a8ef2a358a700311fc0b9479e5060fa3dd71 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCapture.java @@ -0,0 +1,159 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.AWTException; +import java.awt.Graphics2D; +import java.awt.HeadlessException; +import java.awt.Image; +import java.awt.MouseInfo; +import java.awt.Point; +import java.awt.PointerInfo; +import java.awt.Rectangle; +import java.awt.Robot; +import java.awt.image.BufferedImage; +import java.io.IOException; + +import javax.imageio.ImageIO; + +/** + * The Capture class uses the java Robot class to capture the screen. + * @author Snap + * + */ +public class ScreenCapture { + private Robot robot; + private Rectangle screenBounds; + private int scaleWidth, scaleHeight, x, y, captureWidth, captureHeight; + private Point curMouseLocation = new Point(Integer.MIN_VALUE, Integer.MIN_VALUE); + + private Image cursor; + + public ScreenCapture(int x, int y, int captureWidth, int captureHeight, int scaleWidth, int scaleHeight) { + this.x = x; + this.y = y; + this.captureWidth = captureWidth; + this.captureHeight = captureHeight; + + try{ + robot = new Robot(); + }catch (AWTException e){ + System.out.println(e.getMessage()); + } + + this.screenBounds = new Rectangle(x, y, this.captureWidth, this.captureHeight); + this.scaleWidth = scaleWidth; + this.scaleHeight = scaleHeight; + + try { + cursor = ImageIO.read(getClass().getResourceAsStream("/images/Cursor.png")); + } catch (IOException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + + public BufferedImage takeSingleSnapshot() { + BufferedImage capturedImage = robot.createScreenCapture(this.screenBounds); + +// System.out.println("ScreenCapture snap: [cw=" + captureWidth + ",ch=" + captureHeight + "] at [x=" + x + ",y=" + y +"]" +// + "[sw==" + scaleWidth + ",sh=" + scaleHeight + "]"); + + BufferedImage currentScreenshot = new BufferedImage(capturedImage.getWidth(), capturedImage.getHeight(), BufferedImage.TYPE_3BYTE_BGR); + currentScreenshot.getGraphics().drawImage(capturedImage, 0, 0, null); + + Point mouseLoc = takeMouseLocation(); + int x = mouseLoc.x; + int y = mouseLoc.y; + + Graphics2D graphics2D = currentScreenshot.createGraphics(); + graphics2D.drawImage(cursor, x, y, 16, 16, null); // cursor.gif is 16x16 size. + + return currentScreenshot; + } + + public void setX(int x) { + this.x = x; + updateBounds(); + } + + public void setY(int y) { + this.y = y; + updateBounds(); + } + + public void setWidth(int width) { + this.captureWidth = width; + updateBounds(); + } + + public void setHeight(int height) { + this.captureHeight = height; + updateBounds(); + } + + public void updateBounds() { + this.screenBounds = new Rectangle(x, y, captureWidth, captureHeight); + } + + private Point getMouseLocation() { + PointerInfo pInfo; + Point pointerLocation = new Point(0,0); + + try { + pInfo = MouseInfo.getPointerInfo(); + } catch (HeadlessException e) { + pInfo = null; + } catch (SecurityException e) { + pInfo = null; + } + + if (pInfo == null) return pointerLocation; + + return pInfo.getLocation(); + } + + private Point calculatePointerLocation(Point p) { + // System.out.println("Mouse Tracker:: Image=[" + captureWidth + "," + captureHeight + "] scale=[" + scaleWidth + "," + scaleHeight + "]"); + + int mouseXInCapturedRegion = p.x - x; + int mouseYInCapturedRegion = p.y - y; + + double scaledMouseX = mouseXInCapturedRegion * (double)((double)scaleWidth / (double)captureWidth); + double scaledMouseY = mouseYInCapturedRegion * (double)((double)scaleHeight / (double)captureHeight); + + return new Point((int)scaledMouseX, (int)scaledMouseY); + } + + private Point takeMouseLocation() { + Point mouseLocation = getMouseLocation(); + if (isMouseInsideCapturedRegion(mouseLocation)) { + // System.out.println("Mouse is inside captured region [" + mouseLocation.x + "," + mouseLocation.y + "]"); + curMouseLocation = calculatePointerLocation(mouseLocation); + } + + return curMouseLocation; + } + + private boolean isMouseInsideCapturedRegion(Point p) { + return true; + // return ( ( (p.x > captureX) && (p.x < (captureX + captureWidth) ) ) + // && (p.y > captureY && p.y < captureY + captureHeight)); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureListener.java new file mode 100755 index 0000000000000000000000000000000000000000..8e41b4458d0aba4d630ad3f9d273c60851f62b13 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureListener.java @@ -0,0 +1,26 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.image.BufferedImage; + +public interface ScreenCaptureListener { + + void onScreenCaptured(BufferedImage screen); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureTaker.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureTaker.java new file mode 100755 index 0000000000000000000000000000000000000000..8804940b245d59c773d980146d29d2b8823270d6 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenCaptureTaker.java @@ -0,0 +1,45 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.image.BufferedImage; + +public class ScreenCaptureTaker { + private ScreenCapture capture; + + public ScreenCaptureTaker(int x, int y, int captureWidth, int captureHeight, int scaleWidth, int scaleHeight){ + capture = new ScreenCapture(x, y, captureWidth, captureHeight, scaleWidth, scaleHeight); + } + + public void setCaptureCoordinates(int x, int y){ + capture.setX(x); + capture.setY(y); + } + + public BufferedImage captureScreen() { + // System.out.println("----- Taking screen capture -----"); + long start = System.currentTimeMillis(); + BufferedImage image = capture.takeSingleSnapshot(); + long end = System.currentTimeMillis(); + // System.out.println("Capture took " + (end - start) + " millis"); + return image; + } + + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenRegionSharer.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenRegionSharer.java new file mode 100755 index 0000000000000000000000000000000000000000..e7170da67b1a5d674fd61a3f2ba6a0053e034034 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenRegionSharer.java @@ -0,0 +1,120 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import org.bigbluebutton.screenshare.client.frame.CaptureRegionFrame; +import org.bigbluebutton.screenshare.client.frame.CaptureRegionListener; +import org.bigbluebutton.screenshare.client.net.NetworkConnectionListener; +import org.bigbluebutton.screenshare.client.net.NetworkStreamSender; + +public class ScreenRegionSharer implements ScreenSharer, NetworkConnectionListener { + public static final String NAME = "SCREENREGIONSHARER: "; + + private final ScreenShareInfo ssi; + private ScreenSharerRunner sharer; + private CaptureRegionFrame frame; + private NetworkStreamSender signalChannel; + private DeskshareSystemTray tray = new DeskshareSystemTray(); + private ClientListener listener; + + public ScreenRegionSharer(ScreenShareInfo ssi) { + signalChannel = new NetworkStreamSender(ssi.host, ssi.meetingId, ssi.streamId); + signalChannel.addNetworkConnectionListener(this); + signalChannel.start(); + this.ssi = ssi; + sharer = new ScreenSharerRunner(ssi); + } + + public void start(boolean autoStart) { + CaptureRegionListener crl = new CaptureRegionListenerImp(this); + frame = new CaptureRegionFrame(crl, 5); + frame.setHeight(ssi.captureHeight); + frame.setWidth(ssi.captureWidth); + frame.setLocation(ssi.x, ssi.y); + System.out.println(NAME + "Launching Screen Capture Frame"); + frame.start(autoStart); + } + + public void addClientListener(ClientListener l) { + listener = l; + SystemTrayListener systrayListener = new SystemTrayListenerImp(listener); + tray.addSystemTrayListener(systrayListener); + tray.displayIconOnSystemTray(ssi.sysTrayIcon, ssi.enableTrayActions); + } + + public void disconnected(){ + frame.setVisible(false); + sharer.disconnectSharing(); + System.out.println(NAME + "Change system tray icon message"); + tray.disconnectIconSystemTrayMessage(); + System.out.println(NAME + "Desktop sharing disconneted"); + } + + public void stop() { + frame.setVisible(false); + sharer.stopSharing(); + signalChannel.stopSharing(); + tray.removeIconFromSystemTray(); + System.out.println(NAME + "Closing Screen Capture Frame"); + } + + @Override + public void networkConnectionException(ExitCode reason) { + if (listener != null) listener.onClientStop(reason); + } + + private class CaptureRegionListenerImp implements CaptureRegionListener { + private final ScreenRegionSharer srs; + + public CaptureRegionListenerImp(ScreenRegionSharer srs) { + this.srs = srs; + } + + @Override + public void onCaptureRegionMoved(int x, int y) { + ssi.x = x; + ssi.y = y; + if (sharer != null) + sharer.setCaptureCoordinates(x, y); + } + + @Override + public void onStartCapture(int x, int y, int width, int height) { + ssi.x = x; + ssi.y = y; + ssi.captureWidth = width; + ssi.captureHeight = height; + ssi.scaleWidth = width; + ssi.scaleHeight = height; + sharer.updateScreenShareInfo(x, y, width, height); + + signalChannel.startSharing(width, height); + sharer.addClientListener(listener); + + sharer.startSharing(); + } + + @Override + public void onStopCapture() { + srs.stop(); + } + } + + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenShareInfo.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenShareInfo.java new file mode 100755 index 0000000000000000000000000000000000000000..b07ebd5ba90eb59d7841e822fd5bb5f14b364e93 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenShareInfo.java @@ -0,0 +1,42 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.Image; + +public class ScreenShareInfo { + public String host; + public int port; + public String meetingId; + public String streamId; + public String codecOptions; + public int captureWidth; + public int captureHeight; + public int scaleWidth; + public int scaleHeight; + public boolean quality; + public double scale; + public int x; + public int y; + public boolean httpTunnel; + public boolean fullScreen; + public Image sysTrayIcon; + public boolean enableTrayActions; + public String URL; +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharer.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharer.java new file mode 100755 index 0000000000000000000000000000000000000000..db54a52d7e8e04465498c964a158ce61cf4dc3c9 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharer.java @@ -0,0 +1,26 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public interface ScreenSharer { + void start(boolean autoStart); + void disconnected(); + void stop(); + void addClientListener(ClientListener l); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharerRunner.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharerRunner.java new file mode 100755 index 0000000000000000000000000000000000000000..a6266c981434a4cd8ba0166511fe66a8022c170e --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/ScreenSharerRunner.java @@ -0,0 +1,120 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +import java.awt.AWTException; +import java.io.IOException; +import org.bigbluebutton.screenshare.client.javacv.JavaCVScreenshare; + +public class ScreenSharerRunner { + public static final String NAME = "SCREENSHARERUNNER: "; + + boolean connected = false; + private boolean started = false; + private ScreenShareInfo ssi; + private int x, y, width, height; + private final JavaCVScreenshare jcs; + + public ScreenSharerRunner(ScreenShareInfo ssi) { + this.ssi = ssi; + +// System.out.println("ScreenSharerRunner[captureWidth=" + ssi.captureWidth + ",captureHeight=" + ssi.captureHeight + "][" + ssi.x + "," + ssi.y +"]" +// + "[scaleWidth=" + ssi.scaleWidth + ",scaleHeight=" + ssi.scaleHeight + "]"); + + jcs = new JavaCVScreenshare(ssi); + } + + public void updateScreenShareInfo(int x, int y, int width, int height) { + this.x = x; + this.y = y; + this.width = width; + this.height = height; + } + + public void startSharing() { +// printHeader(); + + try { + jcs.go(ssi.URL, x, y, width, height); + jcs.start(); + } catch (IOException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } catch (AWTException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } catch (org.bigbluebutton.screenshare.client.javacv.BBBFrameRecorder.Exception e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + + public void disconnectSharing(){ + System.out.println(NAME + "Disconneted"); + + jcs.stop(); + } // END FUNCTION disconnectSharing + + public void stopSharing() { + System.out.println(NAME + "Stopping"); + System.out.println(NAME + "Removing icon from system tray."); + + jcs.stop(); + } + + public void setCaptureCoordinates(int x, int y) { + jcs.setCaptureCoordinates(x, y); + } + + + public void addClientListener(ClientListener l) { + // NetworkConnectionListener netConnListener = new NetworkConnectionListenerImp(listener); + // if (sender != null) + // sender.addNetworkConnectionListener(netConnListener); + // else + // System.out.println(NAME + "ERROR - Cannot add listener to network connection."); + } + + private void printHeader() { + System.out.println("-----------------------------------------------------------------------"); + System.out.println(LICENSE_HEADER); + System.out.println("-----------------------------------------------------------------------\n\n"); + System.out.println("Desktop Sharing v0.9.0"); + System.out.println("Start"); + System.out.println("Connecting to " + ssi.host + ":" + ssi.port + " meetingId " + ssi.meetingId); + System.out.println("Sharing " + ssi.captureWidth + "x" + ssi.captureHeight + " at " + ssi.x + "," + ssi.y); + System.out.println("Scale to " + ssi.scaleWidth + "x" + ssi.scaleHeight + " with quality = " + ssi.quality); + // System.out.println("Http Tunnel: " + ssi.httpTunnel); + } + + private static final String LICENSE_HEADER = "This program is free software: you can redistribute it and/or modify\n" + + "it under the terms of the GNU Lesser General Public License as published by\n" + + "the Free Software Foundation, either version 3 of the License, or\n" + + "(at your option) any later version.\n\n" + + "This program is distributed in the hope that it will be useful,\n" + + "but WITHOUT ANY WARRANTY; without even the implied warranty of\n" + + "MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n" + + "GNU General Public License for more details.\n\n" + + "You should have received a copy of the GNU Lesser General Public License\n" + + "along with this program. If not, see <http://www.gnu.org/licenses/>.\n\n" + + "Copyright 2010 BigBlueButton. All Rights Reserved.\n\n"; +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListener.java new file mode 100755 index 0000000000000000000000000000000000000000..c606296ea4ebeed94082d023ddadadaf6a73a056 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListener.java @@ -0,0 +1,24 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public interface SystemTrayListener { + + public void onStopSharingSysTrayMenuClicked(); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListenerImp.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListenerImp.java new file mode 100755 index 0000000000000000000000000000000000000000..fc094e9ab0f73a67b7ce289167bb092bb717d409 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/SystemTrayListenerImp.java @@ -0,0 +1,34 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public class SystemTrayListenerImp implements SystemTrayListener { + + private final ClientListener listener; + + public SystemTrayListenerImp(ClientListener listener) { + this.listener = listener; + } + + @Override + public void onStopSharingSysTrayMenuClicked() { + listener.onClientStop(ExitCode.NORMAL); + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/UncheckedExceptions.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/UncheckedExceptions.java new file mode 100755 index 0000000000000000000000000000000000000000..449f1b9b086ba918f075cdb202b67d8f942d00c0 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/UncheckedExceptions.java @@ -0,0 +1,38 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; +class UncheckedExceptions { + private static Throwable throwable; + + private UncheckedExceptions() throws Throwable { + throw throwable; + } + + public static synchronized void spit(Throwable throwable) { + UncheckedExceptions.throwable = throwable; + try { + UncheckedExceptions.class.newInstance(); + } catch(InstantiationException e) { + } catch(IllegalAccessException e) { + } finally { + UncheckedExceptions.throwable = null; + } + } +} + diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/VersionCheckUtil.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/VersionCheckUtil.java new file mode 100755 index 0000000000000000000000000000000000000000..3586ef9faf2c87fac52663deb84a3c48869b7ccc --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/VersionCheckUtil.java @@ -0,0 +1,80 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client; + +public class VersionCheckUtil { + + public static boolean validateMinJREVersion(String runtimeVersion, String minVersion){ + String[] requestedVersioning = minVersion.split("\\."); + String[] clientVersioning = runtimeVersion.split("\\."); + + if (requestedVersioning.length < 3 || clientVersioning.length < 3) + return false; + + // First major update + if (Integer.parseInt(clientVersioning[0]) > Integer.parseInt(requestedVersioning[0])) + return true; + else{ + // Checking Java version + if (Integer.parseInt(clientVersioning[1]) > Integer.parseInt(requestedVersioning[1])) + return true; + + // Checking update + else if (Integer.parseInt(clientVersioning[1]) == Integer.parseInt(requestedVersioning[1])){ + // non-GA or non-FCS release won't be supported + if(clientVersioning[2].indexOf("-") != -1) + return false; + + int rUpdatePart1 = 0; + int rUpdatePart2 = 0; + + int underbar = requestedVersioning[2].indexOf("_"); + if ( underbar == -1){ + rUpdatePart1 = Integer.parseInt(requestedVersioning[2]); + } else { + rUpdatePart1 = Integer.parseInt(requestedVersioning[2].substring(0, underbar)); + rUpdatePart2 = Integer.parseInt(requestedVersioning[2].substring(underbar + 1, requestedVersioning[2].length())); + } + + int cUpdatePart1 = 0; + int cUpdatePart2 = 0; + + underbar = clientVersioning[2].indexOf("_"); + if ( underbar == -1) { + cUpdatePart1 = Integer.parseInt(clientVersioning[2]); + } else { + cUpdatePart1 = Integer.parseInt(clientVersioning[2].substring(0, underbar)); + cUpdatePart2 = Integer.parseInt(clientVersioning[2].substring(underbar + 1, clientVersioning[2].length())); + } + + if (cUpdatePart1 > rUpdatePart1) + return true; + else if (cUpdatePart1 == rUpdatePart1) { + if (cUpdatePart2 > rUpdatePart2 || cUpdatePart2 == rUpdatePart2) + return true; + else + return false; + } else + return false; + } else + return false; + } + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionFrame.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionFrame.java new file mode 100755 index 0000000000000000000000000000000000000000..dc002f8c4543ab52616a6cf506b0784efb67bd8e --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionFrame.java @@ -0,0 +1,164 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.frame; + +import java.awt.Button; +import java.awt.Color; +import java.awt.Dimension; +import java.awt.FlowLayout; +import java.awt.Graphics; +import java.awt.GridBagLayout; +import java.awt.Rectangle; +import java.awt.event.ActionEvent; +import java.awt.event.ActionListener; +import java.awt.image.BufferedImage; +import java.io.IOException; +import javax.imageio.ImageIO; +import javax.swing.BorderFactory; +import javax.swing.ImageIcon; +import javax.swing.JLabel; +import javax.swing.JPanel; + +public class CaptureRegionFrame { + private Button btnStartStop; + private CaptureRegionListener client; + private boolean capturing = false; + private WindowlessFrame frame; + private static final int RESIZE_BAR_SIZE = 40; + private static final int MOVE_BAR_SIZE = 60; + + private final String CLASS = "CaptureRegionFrame"; + + public CaptureRegionFrame(CaptureRegionListener client, int borderWidth) { + frame = new WindowlessFrame(borderWidth); + this.client = client; + frame.setCaptureRegionListener(client); + } + + public void setHeight(int h) { + frame.setHeight(h); + } + + public void setWidth(int w) { + frame.setWidth(w); + } + + public void setLocation(int x, int y) { + frame.setLocation(x, y); + } + + public void setVisible(boolean visible) { + frame.setVisible(visible); + } + + public void start(boolean autoStart) { + frame.setToolbar(createToolbar()); + frame.setResizeBar(createResizeBar()); + frame.setMoveBar(createMoveBar()); + setVisible(true); + if (autoStart) { + startCapture(); + } + } + + private JPanel createResizeBar(){ + final JPanel resizePanel = new JPanel(); + resizePanel.setPreferredSize(new Dimension(RESIZE_BAR_SIZE,RESIZE_BAR_SIZE)); + resizePanel.setBorder(BorderFactory.createLineBorder(Color.RED)); + resizePanel.setLayout(new GridBagLayout()); + BufferedImage resizeCursorImage = null; + + try { + // Image was taken from http://4.bp.blogspot.com/_fhb-4UuRH50/R1ZLryoIvJI/AAAAAAAAA6U/G3S-XYabULk/s1600/se-resize.gif + resizeCursorImage = ImageIO.read(getClass().getResourceAsStream("/images/resize-cursor.png")); + } catch (IOException e) { + e.printStackTrace(); + } + + JLabel resizePicLabel = new JLabel(new ImageIcon(resizeCursorImage)); + resizePanel.add(resizePicLabel); + return resizePanel; + } + + private JPanel createMoveBar() { + final CirclePanel movePanel = new CirclePanel(); + movePanel.setPreferredSize(new Dimension(MOVE_BAR_SIZE,MOVE_BAR_SIZE)); + movePanel.setLayout(new GridBagLayout()); + BufferedImage moveCursorImage = null; + + try { + // Image was taken from http://www.iconarchive.com/show/oxygen-icons-by-oxygen-icons.org/Actions-transform-move-icon.html + moveCursorImage = ImageIO.read(getClass().getResourceAsStream("/images/move-cursor.png")); + } catch (IOException e) { + e.printStackTrace(); + } + + JLabel movePicLabel = new JLabel(new ImageIcon(moveCursorImage)); + movePanel.add(movePicLabel); + return movePanel; + } + + // Wrap move panel in a circle + public class CirclePanel extends JPanel { + static final long serialVersionUID = 1L; + + @Override + protected void paintComponent(Graphics g) { + g.drawOval(0, 0, g.getClipBounds().width, g.getClipBounds().height); + } + } + + private JPanel createToolbar() { + final JPanel panel = new JPanel(); + panel.setBackground(Color.RED); + panel.setLayout(new FlowLayout()); + capturing = false; + btnStartStop = new Button("Start Sharing"); + btnStartStop.addActionListener(new ActionListener() { + @Override + public void actionPerformed(ActionEvent e) { + // if (capturing) { + // capturing = false; + // btnStartStop.setLabel("Start Capture"); + // stopCapture(); + // } else { + // capturing = true; + // btnStartStop.setLabel("Stop Capture"); + startCapture(); + // } + } + }); + panel.add(btnStartStop); + return panel; + } + + private void startCapture() { + System.out.println(CLASS + " - startCapture" ); + frame.changeBorderToBlue(); + System.out.println(CLASS + " - startCapture:: Change border to blue" ); + frame.removeResizeListeners(); + Rectangle rect = frame.getFramedRectangle(); + client.onStartCapture(rect.x, rect.y, frame.getWidth(), frame.getHeight()); + } + + private void stopCapture() { + frame.changeBorderToRed(); + client.onStopCapture(); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionListener.java new file mode 100755 index 0000000000000000000000000000000000000000..5a4784d8dbd8482bccdcbba45db20e8d70640836 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/CaptureRegionListener.java @@ -0,0 +1,26 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.frame; + +public interface CaptureRegionListener { + + void onStartCapture(int x, int y, int width, int height); + void onStopCapture(); + void onCaptureRegionMoved(int x, int y); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/Corner.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/Corner.java new file mode 100755 index 0000000000000000000000000000000000000000..5ebfa787b77be69ac48caae27139d09dd2510cf6 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/Corner.java @@ -0,0 +1,23 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.frame; + +enum Corner { + NORTHWEST, NORTHEAST, SOUTHEAST, SOUTHWEST +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/WindowlessFrame.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/WindowlessFrame.java new file mode 100755 index 0000000000000000000000000000000000000000..a52760bb624b258161a13cee156754127d83a974 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/frame/WindowlessFrame.java @@ -0,0 +1,889 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.frame; + +import java.awt.BasicStroke; +import java.awt.Color; +import java.awt.Component; +import java.awt.Cursor; +import java.awt.Dimension; +import java.awt.Frame; +import java.awt.GradientPaint; +import java.awt.Graphics; +import java.awt.Graphics2D; +import java.awt.GraphicsConfiguration; +import java.awt.GraphicsDevice; +import java.awt.GraphicsEnvironment; +import java.awt.Insets; +import java.awt.Point; +import java.awt.Rectangle; +import java.awt.Toolkit; +import java.awt.Window; +import java.awt.event.MouseAdapter; +import java.awt.event.MouseEvent; +import java.io.Serializable; +import java.util.concurrent.atomic.AtomicBoolean; + +import javax.swing.JFrame; +import javax.swing.JPanel; + +class WindowlessFrame implements Serializable { + private static final long serialVersionUID = 1L; + + private CaptureRegionListener captureRegionListener; + private MouseAdapter resizingAdapter; + private MouseAdapter movingAdapter; + + private int BORDER_THICKNESS = 5; + + private static interface PropertyChanger { + void changeOn(Component component); + } + + private static interface LocationAndSizeUpdateable { + void updateLocationAndSize(); + } + + private static interface OffsetLocator { + int getLeftOffset(); + int getTopOffset(); + } + + private static class StaticOffsetLocator implements OffsetLocator { + private final int mLeftOffset; + private final int mTopOffset; + + public StaticOffsetLocator(int left, int top) { + mLeftOffset = left; + mTopOffset = top; + } + + @Override + public int getLeftOffset() { + return mLeftOffset; + } + + @Override + public int getTopOffset() { + return mTopOffset; + } + } + + private static final PropertyChanger REPAINTER = new PropertyChanger() { + @Override + public void changeOn(Component component) { + if (component instanceof LocationAndSizeUpdateable) { + ((LocationAndSizeUpdateable) component).updateLocationAndSize(); + } + component.repaint(); + } + }; + + + // properties that change during use + private Point mTopLeft = new Point(); + private Dimension mOverallSize = new Dimension(); + + // properties initialized during construction + private BasicStroke mBorderStroke = new BasicStroke(BORDER_THICKNESS, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND, 0, new float[] { 12, 12 }, 0); + private final BasicStroke borderSolidStroke = new BasicStroke(BORDER_THICKNESS, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND, 0); + + private GradientPaint mGradient = new GradientPaint(0.0f, 0.0f, Color.red, 1.0f, 1.0f, Color.white, true); + private final GradientPaint blueGradient = new GradientPaint(0.0f, 0.0f, Color.blue, 1.0f, 1.0f, Color.blue, true); + private final GradientPaint redGradient = new GradientPaint(0.0f, 0.0f, Color.red, 1.0f, 1.0f, Color.white, true); + + private final int mBorderWidth; + private final JFrame mWindowFrame; + private final BarFrame mTopBorder; + private final BarFrame mRightBorder; + private final BarFrame mBottomBorder; + private final BarFrame mLeftBorder; + private ToolbarFrame mToolbarFrame; + private ResizeBarFrame mResizeBarFrame; + private MoveBarFrame mMoveBarFrame; + + private static final int MIN_WINDOW_SIZE = 200; + + private MultiScreen mScreen = new MultiScreen(); + + private class MultiScreen { + private int minX=0 ; //minimum of x position + private int totalWidth=0 ; // total screen resolution + private int curWidth=0 ; // primary screen width + private GraphicsEnvironment ge ; + private GraphicsDevice[] screenDevice ; + private boolean ismultiscreen=false ; + + private MultiScreen(){ + int i ; + + ge = GraphicsEnvironment.getLocalGraphicsEnvironment() ; + screenDevice = ge.getScreenDevices() ; + + if ( 1 < screenDevice.length ){ + // this is the case for multiple devices. + // set the flag to indicate multiple devices on the system. + ismultiscreen=true ; + for ( i=0; i<screenDevice.length; i++){ + GraphicsConfiguration[] gc = screenDevice[i].getConfigurations() ; + + // determine the minimum x position for the main screen + if ( gc[0].getBounds().x <= minX ){ + minX = gc[0].getBounds().x; + } + + // determine the total screen size + if ( gc[0].getBounds().x >= 0){ + totalWidth = totalWidth + gc[0].getBounds().width; + } + + } + }else{ + // this is the case for one screen only. + ismultiscreen = false ; + } + + // set the main screen width + curWidth = screenDevice[0].getConfigurations()[0].getBounds().width ; + + } // END FUNCTION MultiScreen + + public boolean isMultiScreen(){ + return ismultiscreen ; + } // END FUNCTION isMultiScreen + + } // END CLASS MultiScreen + + private class ToolbarFrame extends Window implements LocationAndSizeUpdateable { + private static final long serialVersionUID = 1L; + + private final OffsetLocator mOffsetLocator; + + public ToolbarFrame(JFrame frame, OffsetLocator ol, JPanel content) { + super(frame); + super.setAlwaysOnTop(true); + frame.setAlwaysOnTop(true); + mOffsetLocator = ol; + //setUndecorated(true); + add(content); + pack(); + } + + public void updateLocationAndSize() { + setLocation(getLocation()); + } + + @Override + public Point getLocation() { + return new Point(mTopLeft.x + mOffsetLocator.getLeftOffset(), mTopLeft.y + mOffsetLocator.getTopOffset()); + } + } + + private class MoveBarFrame extends Window implements LocationAndSizeUpdateable { + private static final long serialVersionUID = 1L; + + private final OffsetLocator mOffsetLocator; + private MouseAdapter moveMouseAdapter = null; + + public MoveBarFrame(JFrame frame, OffsetLocator ol, JPanel content) { + super(frame); + super.setAlwaysOnTop(true); + frame.setAlwaysOnTop(true); + mOffsetLocator = ol; + add(content); + pack(); + + moveMouseAdapter = createMoveBarMovingMouseListener(); + + changeMovingBarFrame(new PropertyChanger() { + @Override + public void changeOn(Component component) { + component.addMouseListener(moveMouseAdapter); + component.addMouseMotionListener(moveMouseAdapter); + } + }); + } + + private void changeMovingBarFrame(PropertyChanger pc) { + pc.changeOn(this); + } + + private MouseAdapter createMoveBarMovingMouseListener() { + return new FrameMovingMouseListener(false); + } + + @Override + public void updateLocationAndSize() { + setLocation(getLocation()); + } + + @Override + public Point getLocation() { + return new Point(mTopLeft.x + mOffsetLocator.getLeftOffset(), mTopLeft.y + mOffsetLocator.getTopOffset()); + } + } + + private class ResizeBarFrame extends Window implements LocationAndSizeUpdateable { + + private static final long serialVersionUID = 1L; + private final OffsetLocator mOffsetLocator; + private MouseAdapter resizeMouseAdapter = null; + + public ResizeBarFrame(JFrame frame, OffsetLocator ol, JPanel content) { + super(frame); + super.setAlwaysOnTop(true); + frame.setAlwaysOnTop(true); + setBackground(new Color(0, 255, 0, 0)); + mOffsetLocator = ol; + add(content); + pack(); + + resizeMouseAdapter = createResizeBarResizingMouseListener(); + + changeResizeBarFrame(new PropertyChanger() { + @Override + public void changeOn(Component component) { + component.addMouseListener(resizeMouseAdapter); + component.addMouseMotionListener(resizeMouseAdapter); + } + }); + } + + private MouseAdapter createResizeBarResizingMouseListener() { + return new FrameResizingMouseListener(false); + } + + public void updateLocationAndSize() { + setLocation(getLocation()); + } + + @Override + public Point getLocation() { + return new Point(mTopLeft.x + mOffsetLocator.getLeftOffset(), mTopLeft.y + mOffsetLocator.getTopOffset()); + } + + private void changeResizeBarFrame(PropertyChanger pc) { + pc.changeOn(this); + } + } + + private class FrameMovingMouseListener extends MouseAdapter { + + private AtomicBoolean mMoving = new AtomicBoolean(false); + private Point mActionOffset = null; + private Boolean isBorder = false; + + public FrameMovingMouseListener(Boolean isBorder) { + this.isBorder = isBorder; + } + + @Override + public void mouseDragged(MouseEvent e) { + int changeInX = e.getLocationOnScreen().x - mActionOffset.x - mTopLeft.x; + int changeInY = e.getLocationOnScreen().y - mActionOffset.y - mTopLeft.y; + Toolkit tk = Toolkit.getDefaultToolkit(); + Dimension d = tk.getScreenSize(); + + // check if multiscreen + if ( false == mScreen.isMultiScreen() ){ + // case one screen only + if (mTopLeft.x < 1 && changeInX < 0) { + mTopLeft.x = 0; + changeInX = 0; + } + if (mTopLeft.y < 1 && changeInY < 0) { + mTopLeft.y = 0; + changeInY = 0; + } + if (mTopLeft.x + mOverallSize.width > (d.width-6) && changeInX > 0) { + mTopLeft.x = d.width - mOverallSize.width-BORDER_THICKNESS; + changeInX = 0; + + } + if (mTopLeft.y + mOverallSize.height > (d.height-6) && changeInY > 0) { + mTopLeft.y = d.height - mOverallSize.height-BORDER_THICKNESS; + changeInY = 0; + } + }else{ + // case multiple screen + if (mTopLeft.x < mScreen.minX+1 && changeInX < 0) { + mTopLeft.x = mScreen.minX; + changeInX = 0; + } + if (mTopLeft.y < 1 && changeInY < 0) { + mTopLeft.y = 0; + changeInY = 0; + } + + if (mTopLeft.x + mOverallSize.width > (mScreen.totalWidth-6) && changeInX > 0) { + mTopLeft.x = mScreen.totalWidth - mOverallSize.width-BORDER_THICKNESS; + changeInX = 0; + } + if (mTopLeft.y + mOverallSize.height > (d.height-6) && changeInY > 0) { + mTopLeft.y = d.height - mOverallSize.height-5; + changeInY = 0; + } + } + if (mMoving.get() && !e.isConsumed()) { + WindowlessFrame.this.setLocation(changeInX + mTopLeft.x, changeInY + mTopLeft.y); + } + } + + @Override + public void mousePressed(MouseEvent e) { + final Point mouse = e.getLocationOnScreen(); + mActionOffset = new Point(mouse.x - mTopLeft.x, mouse.y - mTopLeft.y); + mMoving.set(true); + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); + } + + @Override + public void mouseReleased(MouseEvent e) { + mMoving.set(false); + mActionOffset = null; + e.getComponent().setCursor(Cursor.getDefaultCursor()); + } + + @Override + public void mouseMoved(MouseEvent e) { + if (!isBorder) { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); + } + } + } + + private class FrameResizingMouseListener extends MouseAdapter { + + private static final int CORNER_SIZE = 150; + + private AtomicBoolean mResizing = new AtomicBoolean(false); + + private Point mActionOffset = null; + private Dimension mOriginalSize = null; + private Corner mCorner; + private Boolean isBorder = false; + + public FrameResizingMouseListener(Boolean isBorder) { + this.isBorder = isBorder; + } + + @Override + public void mouseDragged(MouseEvent e) { + int changeInX = e.getLocationOnScreen().x - mActionOffset.x - mTopLeft.x; + final int changeInY = e.getLocationOnScreen().y - mActionOffset.y - mTopLeft.y; + + Toolkit tk = Toolkit.getDefaultToolkit(); + Dimension d = tk.getScreenSize(); + + if (mResizing.get()) { + int newH = mOriginalSize.height; + int newW = mOriginalSize.width; + if (isBorder && Corner.SOUTHEAST != mCorner) { + // doing nothing + } else { + if (e.getLocationOnScreen().x < mTopLeft.x+BORDER_THICKNESS) { + newW = BORDER_THICKNESS; + } else { + newW += changeInX; + } + if (e.getLocationOnScreen().y < mTopLeft.y+BORDER_THICKNESS) { + newH = BORDER_THICKNESS; + } else { + newH += changeInY; + } + } + /*else if (mCorner == Corner.NORTHEAST) { + mTopLeft.y = mTopLeft.y + changeInY; + newH = mOverallSize.height + -changeInY; + newW += changeInX; + } else if (mCorner == Corner.NORTHWEST) { + mTopLeft.y = mTopLeft.y + changeInY; + newH = mOverallSize.height + -changeInY; + mTopLeft.x = mTopLeft.x + changeInX; + newW = mOverallSize.width + -changeInX; + } else if (mCorner == Corner.SOUTHWEST) { + newH += changeInY; + mTopLeft.x = mTopLeft.x + changeInX; + newW = mOverallSize.width + -changeInX; + }*/ + //System.out.println("orig size: " + mOriginalSize + ", newH: " + newH + ", newW: " + newW + ", X: " + changeInX + ", Y: " + changeInY); + + if (newH + mTopLeft.y > d.height-BORDER_THICKNESS){ + newH = d.height - mTopLeft.y-BORDER_THICKNESS; + } + + // check if multiple screen _PTS_644_ _PTS_647_ + if ( false == mScreen.isMultiScreen() ){ + // one screen only + if (newW + mTopLeft.x > d.width-BORDER_THICKNESS){ + newW = d.width - mTopLeft.x-BORDER_THICKNESS; + } + }else{ + int mWidth=0 ; + if ( mTopLeft.x > mScreen.curWidth ){ + mWidth = mScreen.totalWidth ; + }else{ + mWidth = d.width ; + } + if (newW + mTopLeft.x > mWidth-BORDER_THICKNESS && mTopLeft.x >= 0){ + newW = mWidth - mTopLeft.x-BORDER_THICKNESS; + }else if (mTopLeft.x<0 && mTopLeft.x + newW > -BORDER_THICKNESS){ + newW = - mTopLeft.x-BORDER_THICKNESS; + } + } + + // set minimum window size + if (newH < MIN_WINDOW_SIZE) { + newH = MIN_WINDOW_SIZE; + } + if (newW < MIN_WINDOW_SIZE) { + newW = MIN_WINDOW_SIZE; + } + +// if (newW % 2 != 0) newW -= 1; +// if (newH % 2 != 0) newH -= 1; + +// int newWidth = makeLengthEven(newW); +// int newHeight = makeLengthEven(newH); + + WindowlessFrame.this.setSize(newW, newH); + e.consume(); + } + } + + @Override + public void mousePressed(MouseEvent e) { + final Point mouse = e.getLocationOnScreen(); + mActionOffset = new Point(mouse.x - mTopLeft.x, mouse.y - mTopLeft.y); + mOriginalSize = new Dimension(mOverallSize); + + if (isBorder) { + mCorner = nearCorner(mouse); + if (mCorner != null ) { + mResizing.set(true); + } + } + else { + mResizing.set(true); + } + } + + @Override + public void mouseReleased(MouseEvent e) { + mResizing.set(false); + mActionOffset = null; + mOriginalSize = null; + mCorner = null; + } + + private Corner nearCorner(Point mouse) { + if (isNearBottomRightCorner(mouse)) { + return Corner.SOUTHEAST; + } /* else if (isNearTopRightCorner(mouse)) { + return Corner.NORTHEAST; + } else if (isNearTopLeftCorner(mouse)) { + return Corner.NORTHWEST; + } else if (isNearBottomLeftCorner(mouse)) { + return Corner.SOUTHWEST; + } + */ + return null; + } + + private boolean isNearBottomRightCorner(Point mouse) { + int xToBotLeft = Math.abs(mTopLeft.x + (int) mOverallSize.getWidth() - mouse.x); + int yToBotLeft = Math.abs(mTopLeft.y + (int) mOverallSize.getHeight() - mouse.y);; + return (xToBotLeft < CORNER_SIZE && yToBotLeft < CORNER_SIZE); + } + + /* private boolean isNearTopRightCorner(Point mouse) { + int xToTopRight = Math.abs(mTopLeft.x + (int) mOverallSize.getWidth() - mouse.x); + int yToTopRight = Math.abs(mTopLeft.y - mouse.y); + return xToTopRight < CORNER_SIZE && yToTopRight < CORNER_SIZE; + } + + private boolean isNearBottomLeftCorner(Point mouse) { + int xToBottomLeft = Math.abs(mTopLeft.x - mouse.x); + int yToBottomLeft = Math.abs(mTopLeft.y + (int) mOverallSize.getHeight() - mouse.y); + return xToBottomLeft < CORNER_SIZE && yToBottomLeft < CORNER_SIZE; + } + + private boolean isNearTopLeftCorner(Point mouse) { + int xToTopLeft = Math.abs(mTopLeft.x - mouse.x); + int yToTopLeft = Math.abs(mTopLeft.y - mouse.y); + return xToTopLeft < CORNER_SIZE && yToTopLeft < CORNER_SIZE; + } + */ + + @Override + public void mouseMoved(MouseEvent e) { + final Point mouse = e.getLocationOnScreen(); + + /* + if (isNearTopLeftCorner(mouse)) { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.NW_RESIZE_CURSOR)); + } else if (isNearBottomLeftCorner(mouse)) { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.SW_RESIZE_CURSOR)); + } else if (isNearTopRightCorner(mouse)) { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.NE_RESIZE_CURSOR)); + } else + */ + if (isBorder) { + if (isNearBottomRightCorner(mouse)) { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.SE_RESIZE_CURSOR)); + } else { + e.getComponent().setCursor(Cursor.getDefaultCursor()); + } + } else { + e.getComponent().setCursor(Cursor.getPredefinedCursor(Cursor.SE_RESIZE_CURSOR)); + } + } + } + + private class BarFrame extends Window implements LocationAndSizeUpdateable { + private static final long serialVersionUID = 1L; + + private final OffsetLocator mOffsetLocator; + + public BarFrame(Frame frame, OffsetLocator offsetLocator) { + super(frame); + mOffsetLocator = offsetLocator; + //setUndecorated(true); + } + + @Override + public void paint(Graphics g) { + if (shouldPaintRectangle()) { + Graphics2D g2 = (Graphics2D) g; + g2.setStroke(mBorderStroke); + g2.setPaint(mGradient); + g2.drawRect(0, 0, getWidth(), getHeight()); + } else { + super.paint(g); + } + } + + protected boolean shouldPaintRectangle() { + return true; + } + + public void updateLocationAndSize() { + setSize(getWidth(), getHeight()); + setLocation(getLocation()); + } + + @Override + public Point getLocation() { + return new Point(mTopLeft.x + mOffsetLocator.getLeftOffset(), mTopLeft.y + mOffsetLocator.getTopOffset()); + } + + @Override + public int getHeight() { + return mBorderWidth; + } + + @Override + public int getWidth() { + return mOverallSize.width; + } + } + + private class HorizontalBarFrame extends BarFrame { + private static final long serialVersionUID = 1L; + + public HorizontalBarFrame(JFrame frame, OffsetLocator offsetLocator) { + super(frame, offsetLocator); + super.setAlwaysOnTop(true); + } + } + + private class VerticalBarFrame extends BarFrame { + private static final long serialVersionUID = 1L; + + public VerticalBarFrame(JFrame frame, OffsetLocator offsetLocator) { + super(frame, offsetLocator); + super.setAlwaysOnTop(true); + } + + @Override + public int getWidth() { + return mBorderWidth; + } + + @Override + public int getHeight() { + return mOverallSize.height; + } + } + + public WindowlessFrame(int borderWidth) { + mBorderWidth = borderWidth; + + mWindowFrame = new JFrame("Windowless Frame"); + //mWindowFrame.setAlwaysOnTop(true); + + mTopBorder = new HorizontalBarFrame(mWindowFrame, new StaticOffsetLocator(0, 0)); + mBottomBorder = new HorizontalBarFrame(mWindowFrame, new OffsetLocator() { + + @Override + public int getTopOffset() { + return mOverallSize.height; + } + + @Override + public int getLeftOffset() { + return 0; + } + }); + + mRightBorder = new VerticalBarFrame(mWindowFrame, new OffsetLocator() { + + @Override + public int getTopOffset() { + return 0; + } + + @Override + public int getLeftOffset() { + return mOverallSize.width; + } + }); + mLeftBorder = new VerticalBarFrame(mWindowFrame, new StaticOffsetLocator(0, 0)); + + movingAdapter = createMovingMouseListener(); + resizingAdapter = createResizingMouseListener(); + changeBarFrames(new PropertyChanger() { + @Override + public void changeOn(Component component) { + component.addMouseListener(resizingAdapter); + component.addMouseMotionListener(resizingAdapter); + component.addMouseListener(movingAdapter); + component.addMouseMotionListener(movingAdapter); + } + }, false); + } + + public final MouseAdapter createMovingMouseListener() { + return new FrameMovingMouseListener(true); + } + + public final MouseAdapter createResizingMouseListener() { + return new FrameResizingMouseListener(true); + } + + public void setToolbar(final JPanel toolbar) { + final OffsetLocator toolbarOffsetLocator = new OffsetLocator() { + @Override + public int getTopOffset() { + return (mOverallSize.height + mBorderWidth - toolbar.getHeight()) / 2; + } + + @Override + public int getLeftOffset() { + return (mOverallSize.width + mBorderWidth - toolbar.getWidth()) / 2; + } + }; + mToolbarFrame = new ToolbarFrame(mWindowFrame, toolbarOffsetLocator, toolbar); + } + + public void setResizeBar(final JPanel resizeBar) { + + final OffsetLocator resizeBarOffsetLocator = new OffsetLocator() { + @Override + public int getTopOffset() { + return (mOverallSize.height + mBorderWidth - resizeBar.getHeight()); + } + + @Override + public int getLeftOffset() { + return (mOverallSize.width + mBorderWidth - resizeBar.getWidth()); + } + }; + + mResizeBarFrame = new ResizeBarFrame(mWindowFrame, resizeBarOffsetLocator, resizeBar); + } + + public void setMoveBar(final JPanel moveBar) { + final OffsetLocator moveBarOffsetLocator = new OffsetLocator() { + @Override + public int getTopOffset() { + return (mOverallSize.height + mBorderWidth - (moveBar.getHeight()/2) - moveBar.getHeight())/2 - 50; + } + + @Override + public int getLeftOffset() { + return (mOverallSize.width + mBorderWidth - (moveBar.getWidth()))/2; + } + }; + mMoveBarFrame = new MoveBarFrame(mWindowFrame, moveBarOffsetLocator, moveBar); + } + + public final void setSize(int width, int height) { + setHeight(height); + setWidth(width); + repaint(); + } + + public final void setWidth(int width) { + int newWidth = width; + if (width % 2 != 0) { + // We need to make this an even number as H264 rejects odd length. + newWidth = makeLengthEven(width); +// System.out.println("Capture width is not even [" + width + "]. Changing to [" + newWidth + "]"); + } + mOverallSize.width = newWidth + BORDER_THICKNESS; + } + + public final void setHeight(int height) { + int newHeight = height; + if (height % 2 != 0) { + // We need to make this an even number as H264 rejects odd length. + newHeight = makeLengthEven(height); +// System.out.println("Capture height is not even [" + height + "]. Changing to [" + newHeight + "]"); + } + mOverallSize.height = newHeight + BORDER_THICKNESS; + } + + public final void setLocation(int x, int y) { + mTopLeft.x = x; + mTopLeft.y = y; + repaint(); + + if (captureRegionListener != null) { + Rectangle rect = getFramedRectangle(); + captureRegionListener.onCaptureRegionMoved(rect.x, rect.y); + } + } + + public final int getX(){ + return mTopLeft.x; + } + + public final int getY(){ + return mTopLeft.y; + } + + public final int getWidth(){ + return mOverallSize.width - mBorderWidth; + } + + public final int getHeight(){ + return mOverallSize.height - mBorderWidth; + } + + public final void centerOnScreen() { + Toolkit kit = mLeftBorder.getToolkit(); + GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment(); + GraphicsDevice[] gs = ge.getScreenDevices(); + Insets in = kit.getScreenInsets(gs[0].getDefaultConfiguration()); + + Dimension d = kit.getScreenSize(); + int maxWidth = (d.width - in.left - in.right); + int maxHeight = (d.height - in.top - in.bottom); + setLocation((int) (maxWidth - mOverallSize.width) / 2, (int) (maxHeight - mOverallSize.height) / 2); + } + + public final Rectangle getFramedRectangle() { + return new Rectangle(mTopLeft.x + mBorderWidth, mTopLeft.y + mBorderWidth, mOverallSize.width - mBorderWidth, mOverallSize.height - mBorderWidth); + } + + public final void setVisible(final boolean b) { + changeAll(new PropertyChanger() { + @Override + public void changeOn(Component component) { + component.setVisible(b); + } + }, true); + } + + private void changeBarFrames(PropertyChanger pc, boolean repaint) { + pc.changeOn(mTopBorder); + pc.changeOn(mRightBorder); + pc.changeOn(mBottomBorder); + pc.changeOn(mLeftBorder); + if (repaint) { + repaint(); + } + } + + private void changeAll(PropertyChanger pc, boolean repaint) { + if (mToolbarFrame != null) pc.changeOn(mToolbarFrame); + if (mResizeBarFrame != null) pc.changeOn(mResizeBarFrame); + if (mMoveBarFrame != null) pc.changeOn(mMoveBarFrame); + changeBarFrames(pc, repaint); + } + + public final void repaint() { + changeAll(REPAINTER, false); + } + + public void changeBorderToBlue() { + mBorderStroke = borderSolidStroke; + mGradient = blueGradient; + repaint(); + } + + public void changeBorderToRed() { + mGradient = redGradient; + repaint(); + } + + public static void main(String[] args) { + final WindowlessFrame wf = new WindowlessFrame(5); + wf.setHeight(300); + wf.setWidth(600); + wf.setLocation(100, 200); + wf.setVisible(true); + } + + public void setCaptureRegionListener(CaptureRegionListener listener){ + this.captureRegionListener = listener; + } + + public void removeResizeListeners() { + mRightBorder.removeMouseListener(resizingAdapter); + mRightBorder.removeMouseMotionListener(resizingAdapter); + mLeftBorder.removeMouseListener(resizingAdapter); + mLeftBorder.removeMouseMotionListener(resizingAdapter); + mTopBorder.removeMouseListener(resizingAdapter); + mTopBorder.removeMouseMotionListener(resizingAdapter); + mBottomBorder.removeMouseListener(resizingAdapter); + mBottomBorder.removeMouseMotionListener(resizingAdapter); + mResizeBarFrame.removeMouseListener(resizingAdapter); + mResizeBarFrame.removeMouseMotionListener(resizingAdapter); + mMoveBarFrame.removeMouseListener(movingAdapter); + mMoveBarFrame.removeMouseMotionListener(movingAdapter); + repaint(); + + System.out.println("Removing listeners......................"); + mToolbarFrame.setVisible(false); + mResizeBarFrame.setVisible(false); + mMoveBarFrame.setVisible(false); + } + + public void addResizeListeners() { + System.out.println("Adding listeners......................"); + mWindowFrame.add(mToolbarFrame); + } + + private int makeLengthEven(int length) { + if (length % 2 != 0) return length - 1; + return length; + } +} \ No newline at end of file diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFFmpegFrameRecorder.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFFmpegFrameRecorder.java new file mode 100755 index 0000000000000000000000000000000000000000..119d9e8627587ec7d22370c905fb2a4f15cfa785 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFFmpegFrameRecorder.java @@ -0,0 +1,668 @@ +package org.bigbluebutton.screenshare.client.javacv; + +import java.awt.Rectangle; +import java.awt.image.BufferedImage; +import java.awt.image.ComponentSampleModel; +import java.awt.image.DataBuffer; +import java.awt.image.DataBufferByte; +import java.awt.image.DataBufferDouble; +import java.awt.image.DataBufferFloat; +import java.awt.image.DataBufferInt; +import java.awt.image.DataBufferShort; +import java.awt.image.DataBufferUShort; +import java.awt.image.MultiPixelPackedSampleModel; +import java.awt.image.Raster; +import java.awt.image.SampleModel; +import java.awt.image.SinglePixelPackedSampleModel; +import java.io.File; +import java.util.Map.Entry; +import org.bytedeco.javacpp.BytePointer; +import org.bytedeco.javacpp.DoublePointer; +import org.bytedeco.javacpp.Loader; +import org.bytedeco.javacpp.PointerPointer; +import static org.bytedeco.javacpp.avcodec.*; +import static org.bytedeco.javacpp.avformat.*; +import static org.bytedeco.javacpp.avutil.*; +import static org.bytedeco.javacpp.swresample.*; +import static org.bytedeco.javacpp.swscale.*; + +/** + * + * @author Samuel Audet + */ +public class BBBFFmpegFrameRecorder extends BBBFrameRecorder { + public static BBBFFmpegFrameRecorder createDefault(File f, int w, int h) throws Exception { return new BBBFFmpegFrameRecorder(f, w, h); } + public static BBBFFmpegFrameRecorder createDefault(String f, int w, int h) throws Exception { return new BBBFFmpegFrameRecorder(f, w, h); } + + private static Exception loadingException = null; + + public static void tryLoad() throws Exception { + if (loadingException != null) { + throw loadingException; + } else { + try { + Loader.load(org.bytedeco.javacpp.avutil.class); + Loader.load(org.bytedeco.javacpp.swresample.class); + Loader.load(org.bytedeco.javacpp.avcodec.class); + Loader.load(org.bytedeco.javacpp.avformat.class); + Loader.load(org.bytedeco.javacpp.swscale.class); + + /* initialize libavcodec, and register all codecs and formats */ + av_register_all(); + avformat_network_init(); + } catch (Throwable t) { + if (t instanceof Exception) { + throw loadingException = (Exception)t; + } else { + throw loadingException = new Exception("Failed to load " + BBBFFmpegFrameRecorder.class, t); + } + } + } + } + + static { + try { + tryLoad(); + } catch (Exception ex) { } + } + + public BBBFFmpegFrameRecorder(File file, int audioChannels) { + this(file, 0, 0, audioChannels); + } + + public BBBFFmpegFrameRecorder(String filename, int audioChannels) { + this(filename, 0, 0, audioChannels); + } + + public BBBFFmpegFrameRecorder(File file, int imageWidth, int imageHeight) { + this(file, imageWidth, imageHeight, 0); + } + + public BBBFFmpegFrameRecorder(String filename, int imageWidth, int imageHeight) { + this(filename, imageWidth, imageHeight, 0); + } + + public BBBFFmpegFrameRecorder(File file, int imageWidth, int imageHeight, int audioChannels) { + this(file.getAbsolutePath(), imageWidth, imageHeight, audioChannels); + } + + public BBBFFmpegFrameRecorder(String filename, int imageWidth, int imageHeight, int audioChannels) { + this.filename = filename; + this.imageWidth = imageWidth; + this.imageHeight = imageHeight; + + this.pixelFormat = AV_PIX_FMT_NONE; + this.videoCodec = AV_CODEC_ID_NONE; + this.videoBitrate = 400000; + this.frameRate = 30; + + this.videoPacket = new AVPacket(); + } + + public void release() throws Exception { + synchronized (org.bytedeco.javacpp.avcodec.class) { + releaseUnsafe(); + } + } + + public void releaseUnsafe() throws Exception { + /* close each codec */ + if (avCodecContext != null) { + avcodec_close(avCodecContext); + avCodecContext = null; + } + + if (picture_buf != null) { + av_free(picture_buf); + picture_buf = null; + } + if (picture != null) { + av_frame_free(picture); + picture = null; + } + if (tmp_picture != null) { + av_frame_free(tmp_picture); + tmp_picture = null; + } + if (videoOutBuf != null) { + av_free(videoOutBuf); + videoOutBuf = null; + } + if (frame != null) { + av_frame_free(frame); + frame = null; + } + + videoStream = null; + + if (outFormatContext != null && !outFormatContext.isNull()) { + if ((outFormat.flags() & AVFMT_NOFILE) == 0) { + /* close the output file */ + avio_close(outFormatContext.pb()); + } + + /* free the streams */ + int nb_streams = outFormatContext.nb_streams(); + for(int i = 0; i < nb_streams; i++) { + av_free(outFormatContext.streams(i).codec()); + av_free(outFormatContext.streams(i)); + } + + /* free the stream */ + av_free(outFormatContext); + outFormatContext = null; + } + + if (img_convert_ctx != null) { + sws_freeContext(img_convert_ctx); + img_convert_ctx = null; + } + + if (samples_convert_ctx != null) { + swr_free(samples_convert_ctx); + samples_convert_ctx = null; + } + } + + @Override protected void finalize() throws Throwable { + super.finalize(); + release(); + } + + private String filename; + private AVFrame picture, tmp_picture; + private BytePointer picture_buf; + private BytePointer videoOutBuf; + private int video_outbuf_size; + private AVFrame frame; + private AVOutputFormat outFormat; + private AVFormatContext outFormatContext; + private AVCodec video_codec; + private AVCodecContext avCodecContext; + private AVStream videoStream; + private SwsContext img_convert_ctx; + private SwrContext samples_convert_ctx; + private AVPacket videoPacket; + private int[] got_video_packet; + + @Override public int getFrameNumber() { + return picture == null ? super.getFrameNumber() : (int)picture.pts(); + } + + @Override public void setFrameNumber(int frameNumber) { + if (picture == null) { super.setFrameNumber(frameNumber); } else { picture.pts(frameNumber); } + } + + // best guess for timestamp in microseconds... + @Override public long getTimestamp() { + return Math.round(getFrameNumber() * 1000000L / getFrameRate()); + } + + @Override public void setTimestamp(long timestamp) { + setFrameNumber((int)Math.round(timestamp * getFrameRate() / 1000000L)); + } + + public void start() throws Exception { + synchronized (org.bytedeco.javacpp.avcodec.class) { + startUnsafe(); + } + } + + public void startUnsafe() throws Exception { + int ret; + picture = null; + tmp_picture = null; + picture_buf = null; + frame = null; + videoOutBuf = null; + outFormatContext = null; + avCodecContext = null; + videoStream = null; + got_video_packet = new int[1]; + + /* auto detect the output format from the name. */ + String format_name = format == null || format.length() == 0 ? null : format; + if ((outFormat = av_guess_format(format_name, filename, null)) == null) { + int proto = filename.indexOf("://"); + if (proto > 0) { + format_name = filename.substring(0, proto); + } + if ((outFormat = av_guess_format(format_name, filename, null)) == null) { + throw new Exception("av_guess_format() error: Could not guess output format for \"" + filename + "\" and " + format + " format."); + } + } + format_name = outFormat.name().getString(); + + /* allocate the output media context */ + if ((outFormatContext = avformat_alloc_context()) == null) { + throw new Exception("avformat_alloc_context() error: Could not allocate format context"); + } + + outFormatContext.oformat(outFormat); + outFormatContext.filename().putString(filename); + + /* add the audio and video streams using the format codecs + and initialize the codecs */ + + if (imageWidth > 0 && imageHeight > 0) { + if (videoCodec != AV_CODEC_ID_NONE) { + outFormat.video_codec(videoCodec); + } else if ("flv".equals(format_name)) { + outFormat.video_codec(AV_CODEC_ID_FLV1); + } else if ("mp4".equals(format_name)) { + outFormat.video_codec(AV_CODEC_ID_MPEG4); + } else if ("3gp".equals(format_name)) { + outFormat.video_codec(AV_CODEC_ID_H263); + } else if ("avi".equals(format_name)) { + outFormat.video_codec(AV_CODEC_ID_HUFFYUV); + } + + /* find the video encoder */ + if ((video_codec = avcodec_find_encoder_by_name(videoCodecName)) == null && + (video_codec = avcodec_find_encoder(outFormat.video_codec())) == null) { + release(); + throw new Exception("avcodec_find_encoder() error: Video codec not found."); + } + + AVRational frame_rate = av_d2q(frameRate, 1001000); + AVRational supported_framerates = video_codec.supported_framerates(); + if (supported_framerates != null) { + int idx = av_find_nearest_q_idx(frame_rate, supported_framerates); + frame_rate = supported_framerates.position(idx); + } + + /* add a video output stream */ + if ((videoStream = avformat_new_stream(outFormatContext, video_codec)) == null) { + release(); + throw new Exception("avformat_new_stream() error: Could not allocate video stream."); + } + avCodecContext = videoStream.codec(); + avCodecContext.codec_id(outFormat.video_codec()); + avCodecContext.codec_type(AVMEDIA_TYPE_VIDEO); + + /* put sample parameters */ + avCodecContext.bit_rate(videoBitrate); + /* resolution must be a multiple of two, but round up to 16 as often required */ + avCodecContext.width((imageWidth + 15) / 16 * 16); + avCodecContext.height(imageHeight); + /* time base: this is the fundamental unit of time (in seconds) in terms + of which frame timestamps are represented. for fixed-fps content, + timebase should be 1/framerate and timestamp increments should be + identically 1. */ + avCodecContext.time_base(av_inv_q(frame_rate)); + videoStream.time_base(av_inv_q(frame_rate)); + if (gopSize >= 0) { + avCodecContext.gop_size(gopSize); /* emit one intra frame every gopSize frames at most */ + } + if (videoQuality >= 0) { + avCodecContext.flags(avCodecContext.flags() | CODEC_FLAG_QSCALE); + avCodecContext.global_quality((int)Math.round(FF_QP2LAMBDA * videoQuality)); + } + + if (pixelFormat != AV_PIX_FMT_NONE) { + avCodecContext.pix_fmt(pixelFormat); + } else if (avCodecContext.codec_id() == AV_CODEC_ID_RAWVIDEO || avCodecContext.codec_id() == AV_CODEC_ID_PNG || + avCodecContext.codec_id() == AV_CODEC_ID_HUFFYUV || avCodecContext.codec_id() == AV_CODEC_ID_FFV1) { + avCodecContext.pix_fmt(AV_PIX_FMT_RGB32); // appropriate for common lossless formats + } else { + avCodecContext.pix_fmt(AV_PIX_FMT_YUV420P); // lossy, but works with about everything + } + + if (avCodecContext.codec_id() == AV_CODEC_ID_MPEG2VIDEO) { + /* just for testing, we also add B frames */ + avCodecContext.max_b_frames(2); + } else if (avCodecContext.codec_id() == AV_CODEC_ID_MPEG1VIDEO) { + /* Needed to avoid using macroblocks in which some coeffs overflow. + This does not happen with normal video, it just happens here as + the motion of the chroma plane does not match the luma plane. */ + avCodecContext.mb_decision(2); + } else if (avCodecContext.codec_id() == AV_CODEC_ID_H263) { + // H.263 does not support any other resolution than the following + if (imageWidth <= 128 && imageHeight <= 96) { + avCodecContext.width(128).height(96); + } else if (imageWidth <= 176 && imageHeight <= 144) { + avCodecContext.width(176).height(144); + } else if (imageWidth <= 352 && imageHeight <= 288) { + avCodecContext.width(352).height(288); + } else if (imageWidth <= 704 && imageHeight <= 576) { + avCodecContext.width(704).height(576); + } else { + avCodecContext.width(1408).height(1152); + } + } else if (avCodecContext.codec_id() == AV_CODEC_ID_H264) { + // default to constrained baseline to produce content that plays back on anything, + // without any significant tradeoffs for most use cases + //video_c.profile(AVCodecContext.FF_PROFILE_H264_HIGH); + avCodecContext.profile(AVCodecContext.FF_PROFILE_H264_CONSTRAINED_BASELINE); + + } + + // some formats want stream headers to be separate + if ((outFormat.flags() & AVFMT_GLOBALHEADER) != 0) { + avCodecContext.flags(avCodecContext.flags() | CODEC_FLAG_GLOBAL_HEADER); + } + + if ((video_codec.capabilities() & CODEC_CAP_EXPERIMENTAL) != 0) { + avCodecContext.strict_std_compliance(AVCodecContext.FF_COMPLIANCE_EXPERIMENTAL); + } + } + + av_dump_format(outFormatContext, 0, filename, 1); + + /* now that all the parameters are set, we can open the audio and + video codecs and allocate the necessary encode buffers */ + if (videoStream != null) { + AVDictionary options = new AVDictionary(null); + if (videoQuality >= 0) { + av_dict_set(options, "crf", "" + videoQuality, 0); + } + for (Entry<String, String> e : videoOptions.entrySet()) { + av_dict_set(options, e.getKey(), e.getValue(), 0); + } + /* open the codec */ + if ((ret = avcodec_open2(avCodecContext, video_codec, options)) < 0) { + release(); + throw new Exception("avcodec_open2() error " + ret + ": Could not open video codec."); + } + av_dict_free(options); + + videoOutBuf = null; + if ((outFormat.flags() & AVFMT_RAWPICTURE) == 0) { + /* allocate output buffer */ + /* XXX: API change will be done */ + /* buffers passed into lav* can be allocated any way you prefer, + as long as they're aligned enough for the architecture, and + they're freed appropriately (such as using av_free for buffers + allocated with av_malloc) */ + video_outbuf_size = Math.max(256 * 1024, 8 * avCodecContext.width() * avCodecContext.height()); // a la ffmpeg.c + videoOutBuf = new BytePointer(av_malloc(video_outbuf_size)); + } + + /* allocate the encoded raw picture */ + if ((picture = av_frame_alloc()) == null) { + release(); + throw new Exception("av_frame_alloc() error: Could not allocate picture."); + } + picture.pts(0); // magic required by libx264 + + int size = avpicture_get_size(avCodecContext.pix_fmt(), avCodecContext.width(), avCodecContext.height()); + if ((picture_buf = new BytePointer(av_malloc(size))).isNull()) { + release(); + throw new Exception("av_malloc() error: Could not allocate picture buffer."); + } + + /* if the output format is not equal to the image format, then a temporary + picture is needed too. It is then converted to the required output format */ + if ((tmp_picture = av_frame_alloc()) == null) { + release(); + throw new Exception("av_frame_alloc() error: Could not allocate temporary picture."); + } + } + + /* open the output file, if needed */ + if ((outFormat.flags() & AVFMT_NOFILE) == 0) { + AVIOContext pb = new AVIOContext(null); + if ((ret = avio_open(pb, filename, AVIO_FLAG_WRITE)) < 0) { + release(); + throw new Exception("avio_open error() error " + ret + ": Could not open '" + filename + "'"); + } + outFormatContext.pb(pb); + } + + /* write the stream header, if any */ + avformat_write_header(outFormatContext, (PointerPointer)null); + } + + public void stop() throws Exception { + if (outFormatContext != null) { + try { + /* ralam TODO: flush all the buffers */ + // while (video_st != null && record((IplImage)null, AV_PIX_FMT_NONE)); + av_write_frame(outFormatContext, null); + + /* write the trailer, if any */ + av_write_trailer(outFormatContext); + } finally { + release(); + } + } + } + + public boolean record(BytePointer data, int width, int height, int pixelFormat) throws Exception { + if (videoStream == null) { + throw new Exception("No video output stream (Is imageWidth > 0 && imageHeight > 0 and has start() been called?)"); + } + int ret; + + if (data == null) { + /* no more frame to compress. The codec has a latency of a few + frames if using B frames, so we get the last frames by + passing the same picture again */ + } else { + // Should get the step programatically. 3 comes from RGB bytes x width (ralam jan 22, 2014) + int step = 3 * width; + + // System.out.println("PIXEL FORMAT=[" + pixelFormat + "] step=[" + step + "]"); + + if (avCodecContext.pix_fmt() != pixelFormat || avCodecContext.width() != width || avCodecContext.height() != height) { + // System.out.println("Converting picture: vcfmt=[" + avCodecContext.pix_fmt() + " pxfmt=[" + pixelFormat + "]" + + // "vcw=[" + avCodecContext.width() + "] width=[" + width + "] vch=[" + avCodecContext.height() + "] height=[" + height + "]"); + + /* convert to the codec pixel format if needed */ + img_convert_ctx = sws_getCachedContext(img_convert_ctx, width, height, pixelFormat, + avCodecContext.width(), avCodecContext.height(), avCodecContext.pix_fmt(), SWS_BILINEAR, + null, null, (DoublePointer)null); + if (img_convert_ctx == null) { + throw new Exception("sws_getCachedContext() error: Cannot initialize the conversion context."); + } + avpicture_fill(new AVPicture(tmp_picture), data, pixelFormat, width, height); + avpicture_fill(new AVPicture(picture), picture_buf, avCodecContext.pix_fmt(), avCodecContext.width(), avCodecContext.height()); + tmp_picture.linesize(0, step); + sws_scale(img_convert_ctx, new PointerPointer(tmp_picture), tmp_picture.linesize(), + 0, height, new PointerPointer(picture), picture.linesize()); + } else { + avpicture_fill(new AVPicture(picture), data, pixelFormat, width, height); + picture.linesize(0, step); + } + } + + if ((outFormat.flags() & AVFMT_RAWPICTURE) != 0) { + if (data == null) { + return false; + } + /* raw video case. The API may change slightly in the future for that? */ + av_init_packet(videoPacket); + videoPacket.flags(videoPacket.flags() | AV_PKT_FLAG_KEY); + videoPacket.stream_index(videoStream.index()); + videoPacket.data(new BytePointer(picture)); + videoPacket.size(Loader.sizeof(AVPicture.class)); + } else { + /* encode the image */ + av_init_packet(videoPacket); + videoPacket.data(videoOutBuf); + videoPacket.size(video_outbuf_size); + picture.quality(avCodecContext.global_quality()); + if ((ret = avcodec_encode_video2(avCodecContext, videoPacket, data == null ? null : picture, got_video_packet)) < 0) { + throw new Exception("avcodec_encode_video2() error " + ret + ": Could not encode video packet."); + } + picture.pts(picture.pts() + 1); // magic required by libx264 + + /* if zero size, it means the image was buffered */ + if (got_video_packet[0] != 0) { + if (videoPacket.pts() != AV_NOPTS_VALUE) { + videoPacket.pts(av_rescale_q(videoPacket.pts(), avCodecContext.time_base(), videoStream.time_base())); + } + if (videoPacket.dts() != AV_NOPTS_VALUE) { + videoPacket.dts(av_rescale_q(videoPacket.dts(), avCodecContext.time_base(), videoStream.time_base())); + } + videoPacket.stream_index(videoStream.index()); + } else { + return false; + } + } + + synchronized (outFormatContext) { + /* write the compressed frame in the media file */ + if ((ret = av_write_frame(outFormatContext, videoPacket)) < 0) { + throw new Exception("av_write_frame() error " + ret + " while writing video frame."); + } + } + return (videoPacket.flags() & AV_PKT_FLAG_KEY) == 1; + } + + + public void copyFrom(BufferedImage image, double gamma, boolean flipChannels) { + Rectangle r = new Rectangle(0, 0, image.getWidth(), image.getHeight()); + copyFrom(image, gamma, flipChannels, r); + } + + public void copyFrom(BufferedImage image, double gamma, boolean flipChannels, Rectangle roi) { + + // ByteBuffer out = getByteBuffer(roi == null ? 0 : roi.y*arrayStep() + roi.x); + SampleModel sm = image.getSampleModel(); + Raster r = image.getRaster(); + DataBuffer in = r.getDataBuffer(); + int x = -r.getSampleModelTranslateX(); + int y = -r.getSampleModelTranslateY(); + int step = sm.getWidth()*sm.getNumBands(); + int channels = sm.getNumBands(); + + if (sm instanceof ComponentSampleModel) { + step = ((ComponentSampleModel)sm).getScanlineStride(); + channels = ((ComponentSampleModel)sm).getPixelStride(); + } else if (sm instanceof SinglePixelPackedSampleModel) { + step = ((SinglePixelPackedSampleModel)sm).getScanlineStride(); + channels = 1; + } else if (sm instanceof MultiPixelPackedSampleModel) { + step = ((MultiPixelPackedSampleModel)sm).getScanlineStride(); + channels = ((MultiPixelPackedSampleModel)sm).getPixelBitStride()/8; // ?? + } + int start = y*step + x*channels; + + if (in instanceof DataBufferByte) { + System.out.println("DataBufferByte"); + // byte[] a = ((DataBufferByte)in).getData(); + // flipCopyWithGamma(ByteBuffer.wrap(a, start, a.length - start), step, out, arrayStep(), false, gamma, false, flipChannels ? channels : 0); + } else if (in instanceof DataBufferDouble) { + System.out.println("DataBufferDouble"); + // double[] a = ((DataBufferDouble)in).getData(); + // flipCopyWithGamma(DoubleBuffer.wrap(a, start, a.length - start), step, out.asDoubleBuffer(), arrayStep()/8, gamma, false, flipChannels ? channels : 0); + } else if (in instanceof DataBufferFloat) { + System.out.println("DataBufferFloat"); + // float[] a = ((DataBufferFloat)in).getData(); + // flipCopyWithGamma(FloatBuffer.wrap(a, start, a.length - start), step, out.asFloatBuffer(), arrayStep()/4, gamma, false, flipChannels ? channels : 0); + } else if (in instanceof DataBufferInt) { + System.out.println("DataBufferInt"); + // int[] a = ((DataBufferInt)in).getData(); + // flipCopyWithGamma(IntBuffer.wrap(a, start, a.length - start), step, out.asIntBuffer(), arrayStep()/4, gamma, false, flipChannels ? channels : 0); + } else if (in instanceof DataBufferShort) { + System.out.println("DataBufferShort"); + // short[] a = ((DataBufferShort)in).getData(); + // flipCopyWithGamma(ShortBuffer.wrap(a, start, a.length - start), step, out.asShortBuffer(), arrayStep()/2, true, gamma, false, flipChannels ? channels : 0); + } else if (in instanceof DataBufferUShort) { + System.out.println("DataBufferUShort"); + // short[] a = ((DataBufferUShort)in).getData(); + // flipCopyWithGamma(ShortBuffer.wrap(a, start, a.length - start), step, out.asShortBuffer(), arrayStep()/2, false, gamma, false, flipChannels ? channels : 0); + } else { + assert false; + } + + // if (bufferedImage == null && roi == null && + // image.getWidth() == arrayWidth() && image.getHeight() == arrayHeight()) { + // bufferedImage = image; + // } + + } + + /** + public static void flipCopyWithGamma(ByteBuffer srcBuf, int srcStep, + ByteBuffer dstBuf, int dstStep, boolean signed, double gamma, boolean flip, int channels) { + assert srcBuf != dstBuf; + int w = Math.min(srcStep, dstStep); + int srcLine = srcBuf.position(), dstLine = dstBuf.position(); + byte[] buffer = new byte[channels]; + while (srcLine < srcBuf.capacity() && dstLine < dstBuf.capacity()) { + if (flip) { + srcBuf.position(srcBuf.capacity() - srcLine - srcStep); + } else { + srcBuf.position(srcLine); + } + dstBuf.position(dstLine); + w = Math.min(Math.min(w, srcBuf.remaining()), dstBuf.remaining()); + if (signed) { + if (channels > 1) { + for (int x = 0; x < w; x+=channels) { + for (int z = 0; z < channels; z++) { + int in = srcBuf.get(); + byte out; + if (gamma == 1.0) { + out = (byte)in; + } else { + out = (byte)Math.round(Math.pow((double)in/Byte.MAX_VALUE, gamma)*Byte.MAX_VALUE); + } + buffer[z] = out; + } + for (int z = channels-1; z >= 0; z--) { + dstBuf.put(buffer[z]); + } + } + } else { + for (int x = 0; x < w; x++) { + int in = srcBuf.get(); + byte out; + if (gamma == 1.0) { + out = (byte)in; + } else { + out = (byte)Math.round(Math.pow((double)in/Byte.MAX_VALUE, gamma)*Byte.MAX_VALUE); + } + dstBuf.put(out); + } + } + } else { + if (channels > 1) { + for (int x = 0; x < w; x+=channels) { + for (int z = 0; z < channels; z++) { + byte out; + int in = srcBuf.get() & 0xFF; + if (gamma == 1.0) { + out = (byte)in; + } else if (gamma == 2.2) { + out = gamma22[in]; + } else if (gamma == 1/2.2) { + out = gamma22inv[in]; + } else { + out = (byte)Math.round(Math.pow((double)in/0xFF, gamma)*0xFF); + } + buffer[z] = out; + } + for (int z = channels-1; z >= 0; z--) { + dstBuf.put(buffer[z]); + } + } + } else { + for (int x = 0; x < w; x++) { + byte out; + int in = srcBuf.get() & 0xFF; + if (gamma == 1.0) { + out = (byte)in; + } else if (gamma == 2.2) { + out = gamma22[in]; + } else if (gamma == 1/2.2) { + out = gamma22inv[in]; + } else { + out = (byte)Math.round(Math.pow((double)in/0xFF, gamma)*0xFF); + } + dstBuf.put(out); + } + } + } + srcLine += srcStep; + dstLine += dstStep; + } +} + + **/ + + + +} \ No newline at end of file diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFrameRecorder.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFrameRecorder.java new file mode 100755 index 0000000000000000000000000000000000000000..09612e31d460f8efa2870a4a73d8b602c737e25f --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/BBBFrameRecorder.java @@ -0,0 +1,193 @@ +package org.bigbluebutton.screenshare.client.javacv; + +import java.io.File; +import java.lang.reflect.InvocationTargetException; +import java.util.Arrays; +import java.util.HashMap; +import java.util.LinkedList; +import java.util.List; + +public abstract class BBBFrameRecorder { + + public static final List<String> list = new LinkedList<String>(Arrays.asList(new String[] { "FFmpeg", "OpenCV" })); + public static void init() { + for (String name : list) { + try { + Class<? extends BBBFrameRecorder> c = get(name); + c.getMethod("tryLoad").invoke(null); + } catch (Throwable t) { } + } + } + public static Class<? extends BBBFrameRecorder> getDefault() { + // select first frame recorder that can load.. + for (String name : list) { + try { + Class<? extends BBBFrameRecorder> c = get(name); + c.getMethod("tryLoad").invoke(null); + return c; + } catch (Throwable t) { } + } + return null; + } + public static Class<? extends BBBFrameRecorder> get(String className) throws Exception { + className = BBBFrameRecorder.class.getPackage().getName() + "." + className; + try { + return Class.forName(className).asSubclass(BBBFrameRecorder.class); + } catch (ClassNotFoundException e) { + String className2 = className + "FrameRecorder"; + try { + return Class.forName(className2).asSubclass(BBBFrameRecorder.class); + } catch (ClassNotFoundException ex) { + throw new Exception("Could not get FrameRecorder class for " + className + " or " + className2, e); + } + } + } + + public static BBBFrameRecorder create(Class<? extends BBBFrameRecorder> c, Class p, Object o, int w, int h) throws Exception { + Throwable cause = null; + try { + return c.getConstructor(p, int.class, int.class).newInstance(o, w, h); + } catch (InstantiationException ex) { + cause = ex; + } catch (IllegalAccessException ex) { + cause = ex; + } catch (IllegalArgumentException ex) { + cause = ex; + } catch (NoSuchMethodException ex) { + cause = ex; + } catch (InvocationTargetException ex) { + cause = ex.getCause(); + } + throw new Exception("Could not create new " + c.getSimpleName() + "(" + o + ", " + w + ", " + h + ")", cause); + } + + public static BBBFrameRecorder createDefault(File file, int width, int height) throws Exception { + return create(getDefault(), File.class, file, width, height); + } + public static BBBFrameRecorder createDefault(String filename, int width, int height) throws Exception { + return create(getDefault(), String.class, filename, width, height); + } + + public static BBBFrameRecorder create(String className, File file, int width, int height) throws Exception { + return create(get(className), File.class, file, width, height); + } + public static BBBFrameRecorder create(String className, String filename, int width, int height) throws Exception { + return create(get(className), String.class, filename, width, height); + } + + protected String format, videoCodecName; + protected int imageWidth, imageHeight, audioChannels; + protected int pixelFormat, videoCodec, videoBitrate, gopSize = -1; + protected double frameRate, videoQuality = -1; + protected HashMap<String, String> videoOptions = new HashMap<String, String>(); + protected int frameNumber = 0; + protected long timestamp = 0; + + public String getFormat() { + return format; + } + public void setFormat(String format) { + this.format = format; + } + + public String getVideoCodecName() { + return videoCodecName; + } + public void setVideoCodecName(String videoCodecName) { + this.videoCodecName = videoCodecName; + } + + public int getImageWidth() { + return imageWidth; + } + public void setImageWidth(int imageWidth) { + this.imageWidth = imageWidth; + } + + public int getImageHeight() { + return imageHeight; + } + public void setImageHeight(int imageHeight) { + this.imageHeight = imageHeight; + } + + public int getAudioChannels() { + return audioChannels; + } + public void setAudioChannels(int audioChannels) { + this.audioChannels = audioChannels; + } + + public int getPixelFormat() { + return pixelFormat; + } + public void setPixelFormat(int pixelFormat) { + this.pixelFormat = pixelFormat; + } + + public int getVideoCodec() { + return videoCodec; + } + public void setVideoCodec(int videoCodec) { + this.videoCodec = videoCodec; + } + + public int getVideoBitrate() { + return videoBitrate; + } + public void setVideoBitrate(int videoBitrate) { + this.videoBitrate = videoBitrate; + } + + public int getGopSize() { + return gopSize; + } + public void setGopSize(int gopSize) { + this.gopSize = gopSize; + } + + public double getFrameRate() { + return frameRate; + } + public void setFrameRate(double frameRate) { + this.frameRate = frameRate; + } + + public double getVideoQuality() { + return videoQuality; + } + public void setVideoQuality(double videoQuality) { + this.videoQuality = videoQuality; + } + + public String getVideoOption(String key) { + return videoOptions.get(key); + } + public void setVideoOption(String key, String value) { + videoOptions.put(key, value); + } + + public int getFrameNumber() { + return frameNumber; + } + public void setFrameNumber(int frameNumber) { + this.frameNumber = frameNumber; + } + + public long getTimestamp() { + return timestamp; + } + public void setTimestamp(long timestamp) { + this.timestamp = timestamp; + } + + public static class Exception extends java.lang.Exception { + public Exception(String message) { super(message); } + public Exception(String message, Throwable cause) { super(message, cause); } + } + + public abstract void start() throws Exception; + public abstract void stop() throws Exception; + + public abstract void release() throws Exception; +} \ No newline at end of file diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/JavaCVScreenshare.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/JavaCVScreenshare.java new file mode 100755 index 0000000000000000000000000000000000000000..3938102ff6d4c368a6b4d4b120dffb7df8dbaa20 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/javacv/JavaCVScreenshare.java @@ -0,0 +1,204 @@ +package org.bigbluebutton.screenshare.client.javacv; + +import java.awt.AWTException; +import java.awt.image.BufferedImage; +import java.awt.image.DataBuffer; +import java.awt.image.DataBufferByte; +import java.io.IOException; +import java.nio.ByteBuffer; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import org.bigbluebutton.screenshare.client.ScreenCaptureTaker; +import org.bigbluebutton.screenshare.client.ScreenShareInfo; +import org.bytedeco.javacpp.BytePointer; +import static org.bytedeco.javacpp.avcodec.*; +import static org.bytedeco.javacpp.avutil.*; + +public class JavaCVScreenshare { + + private volatile boolean startBroadcast = false; + private final Executor startBroadcastExec = Executors.newSingleThreadExecutor(); + private Runnable startBroadcastRunner; + private BBBFFmpegFrameRecorder recorder = null; + private Double defaultFrameRate = 12.0; + private Double frameRate = 12.0; + private int defaultKeyFrameInterval = 6; + + private long startTime; + private int frameNumber = 1; + + private ScreenShareInfo ssi; + private ScreenCaptureTaker captureTaker; + + + private final String FRAMERATE_KEY = "frameRate"; + private final String KEYFRAMEINTERVAL_KEY = "keyFrameInterval"; + + public JavaCVScreenshare(ScreenShareInfo ssi) { + this.ssi = ssi; + captureTaker = new ScreenCaptureTaker(ssi.x, ssi.y, ssi.captureWidth, ssi.captureHeight, ssi.scaleWidth, ssi.scaleHeight); + } + + public void setCaptureCoordinates(int x, int y){ + captureTaker.setCaptureCoordinates(x, y); + } + + private Map<String, String> splitToMap(String source, String entriesSeparator, String keyValueSeparator) { + System.out.println("CODEC_OPTS=" + source); + Map<String, String> map = new HashMap<String, String>(); + String[] entries = source.split(entriesSeparator); + for (String entry : entries) { + if (entry != "" && entry.contains(keyValueSeparator)) { + String[] keyValue = entry.split(keyValueSeparator); + System.out.println("OPTION: " + keyValue[0] + "=" + keyValue[1]); + map.put(keyValue[0], keyValue[1]); + } + } + return map; + } + + public void go(String URL, int x, int y, int width, int height) throws IOException, BBBFrameRecorder.Exception, + AWTException, InterruptedException { + + captureTaker = new ScreenCaptureTaker(x, y, width, height, width, height); + + System.out.println("Capturing w=[" + width + "] h=[" + height + "] at x=[" + x + "] y=[" + y + "]"); + + recorder = new BBBFFmpegFrameRecorder(URL, width, height); + recorder.setFormat("flv"); + + /// + // Flash SVC2 + //recorder.setVideoCodec(AV_CODEC_ID_FLASHSV2); + //recorder.setPixelFormat(AV_PIX_FMT_BGR24); + + // H264 + recorder.setVideoCodec(AV_CODEC_ID_H264); + recorder.setPixelFormat(AV_PIX_FMT_YUV420P); + + Map<String, String> codecOptions = splitToMap(ssi.codecOptions, "&", "="); + + frameRate = parseFrameRate(codecOptions.get(FRAMERATE_KEY)); + recorder.setFrameRate(frameRate); + + int keyFrameInterval = parseKeyFrameInterval(codecOptions.get(KEYFRAMEINTERVAL_KEY)); + int gopSize = frameRate.intValue() * keyFrameInterval; + recorder.setGopSize(gopSize); + + System.out.println("==== CODEC OPTIONS ====="); + for (Map.Entry<String, String> entry : codecOptions.entrySet()) { + System.out.println("Key = " + entry.getKey() + ", Value = " + entry.getValue()); + if (entry.getKey().equals(FRAMERATE_KEY) || entry.getKey().equals(KEYFRAMEINTERVAL_KEY)) { + // ignore as we have handled this above + } else { + recorder.setVideoOption(entry.getKey(), entry.getValue()); + } + + } + System.out.println("==== END CODEC OPTIONS ====="); + + startTime = System.currentTimeMillis(); + + try { + recorder.start(); + } catch (Exception e1) { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + } + + private Double parseFrameRate(String value) { + Double fr = defaultFrameRate; + + try { + fr = Double.parseDouble(value); + } catch (NumberFormatException e) { + fr = defaultFrameRate; + } + + return fr; + } + + private int parseKeyFrameInterval(String value) { + int fr = defaultKeyFrameInterval; + + try { + fr = Integer.parseInt(value); + } catch (NumberFormatException e) { + fr = defaultKeyFrameInterval; + } + + return fr; + } + + private void captureScreen() { + long now = System.currentTimeMillis(); + + BufferedImage currentScreenshot = captureTaker.captureScreen(); + DataBuffer in = currentScreenshot.getData().getDataBuffer(); + + byte[] a = ((DataBufferByte)in).getData();; + + ByteBuffer bbuffer = ByteBuffer.wrap(a); + + BytePointer bpointer = new BytePointer(bbuffer); + try { + recorder.record(bpointer, currentScreenshot.getWidth(), currentScreenshot.getHeight(), AV_PIX_FMT_BGR24); + } catch (Exception e1) { + // TODO Auto-generated catch block + e1.printStackTrace(); + } + + long sleepFramerate = (long) (1000 / frameRate); + long timestamp = now - startTime; + recorder.setTimestamp(timestamp * 1000); + + // System.out.println("i=[" + i + "] timestamp=[" + timestamp + "]"); + recorder.setFrameNumber(frameNumber); + +// System.out.println("[ENCODER] encoded image " + frameNumber + " in " + (System.currentTimeMillis() - now)); + frameNumber++; + + long execDuration = (System.currentTimeMillis() - now); + long sleepDuration = Math.max(sleepFramerate - execDuration, 0); + pause(sleepDuration); + + } + + private void pause(long dur) { + try{ + Thread.sleep(dur); + } catch (Exception e){ + System.out.println("Exception sleeping."); + } + } + + public void start() { + startBroadcast = true; + startBroadcastRunner = new Runnable() { + public void run() { + while (startBroadcast){ + captureScreen(); + } + System.out.println("Stopping screen capture."); + } + }; + startBroadcastExec.execute(startBroadcastRunner); + } + + public void stop() { + startBroadcast = false; + if (recorder != null) { + + try { + recorder.stop(); + recorder.release(); + } catch (Exception e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamProtocolEncoder.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamProtocolEncoder.java new file mode 100755 index 0000000000000000000000000000000000000000..58a171e0602628d0d3f297f4bd24971797589805 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamProtocolEncoder.java @@ -0,0 +1,119 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +import java.awt.Point; +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.util.zip.Adler32; + +public class BlockStreamProtocolEncoder { + private static final byte[] END_FRAME = new byte[] {'D', 'S', '-', 'E', 'N', 'D'}; + private static final byte[] HEADER = new byte[] {'B', 'B', 'B', '-', 'D', 'S'}; + private static final byte CAPTURE_START_EVENT = 0; + private static final byte CAPTURE_UPDATE_EVENT = 1; + private static final byte CAPTURE_END_EVENT = 2; + private static final byte MOUSE_LOCATION_EVENT = 3; + private static final byte CORRUPT_PACKET_EVENT = 7; + + public static void encodeStartStreamMessage(String meetingId, int width, int height, + ByteArrayOutputStream data, int seqNum) throws IOException { + + data.write(CAPTURE_START_EVENT); + encodeMeetingId(data, meetingId); + encodeSequenceNumber(data, seqNum); + + data.write(intToBytes(width)); + data.write(intToBytes(height)); + + } + + + public static void encodeRoomAndSequenceNumber(String meetingId, int seqNum, ByteArrayOutputStream data) throws IOException{ + data.write(CAPTURE_UPDATE_EVENT); + encodeMeetingId(data, meetingId); + encodeSequenceNumber(data, seqNum); + } + + public static byte[] encodeHeaderAndLength(ByteArrayOutputStream data) throws IOException { + ByteArrayOutputStream header = new ByteArrayOutputStream(); + header.write(HEADER); + header.write(intToBytes(data.size())); + + return header.toByteArray(); + } + + public static void encodeMouseLocation(Point mouseLoc, String meetingId, ByteArrayOutputStream data, int seqNum) throws IOException { + data.write(MOUSE_LOCATION_EVENT); + encodeMeetingId(data, meetingId); + encodeSequenceNumber(data, seqNum); + data.write(intToBytes(mouseLoc.x)); + data.write(intToBytes(mouseLoc.y)); + } + + public static void encodeEndStreamMessage(String meetingId, ByteArrayOutputStream data, int seqNum) throws IOException { + data.write(CAPTURE_END_EVENT); + encodeMeetingId(data, meetingId); + encodeSequenceNumber(data, seqNum); + } + + public static void encodeDelimiter(ByteArrayOutputStream data) throws IOException { + data.write(END_FRAME); + } + + public static byte[] encodeChecksum(byte[] data) { + Adler32 checksum = new Adler32(); + checksum.reset(); + checksum.update(data); + return longToBytes(checksum.getValue()); + } + + private static byte[] longToBytes(long i) { + byte[] data = new byte[8]; + + data[0] = (byte)((i >> 56) & 0xff); + data[1] = (byte)((i >> 48) & 0xff); + data[2] = (byte)((i >> 40) & 0xff); + data[3] = (byte)((i >> 32) & 0xff); + data[4] = (byte)((i >> 24) & 0xff); + data[5] = (byte)((i >> 16) & 0xff); + data[6] = (byte)((i >> 8) & 0xff); + data[7] = (byte)(i & 0xff); + + return data; + } + + private static byte[] intToBytes(int i) { + byte[] data = new byte[4]; + data[0] = (byte)((i >> 24) & 0xff); + data[1] = (byte)((i >> 16) & 0xff); + data[2] = (byte)((i >> 8) & 0xff); + data[3] = (byte)(i & 0xff); + return data; + } + + private static void encodeMeetingId(ByteArrayOutputStream data, String meetingId) throws IOException { + data.write(meetingId.length()); + data.write(meetingId.getBytes()); + } + + private static void encodeSequenceNumber(ByteArrayOutputStream data, int seqNum) throws IOException { + data.write(intToBytes(seqNum)); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamSender.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamSender.java new file mode 100755 index 0000000000000000000000000000000000000000..5b720681d80588cf5a8498d047ec17e5313ac5bc --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/BlockStreamSender.java @@ -0,0 +1,160 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +import java.io.ByteArrayOutputStream; +import java.io.DataOutputStream; +import java.io.IOException; +import java.net.Socket; +import java.net.UnknownHostException; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; + +public class BlockStreamSender implements ScreenCaptureSender { + private static final int PORT = 9123; + + private Socket socket = null; + + private DataOutputStream outStream = null; + private String meetingId; + + private BlockingQueue<String> screenQ = new LinkedBlockingQueue<String>(500); + private final Executor exec = Executors.newSingleThreadExecutor(); + private Runnable capturedScreenSender; + private volatile boolean sendCapturedScreen = false; + + private ByteArrayOutputStream dataToSend; + + private static final byte[] HEADER = new byte[] {'B', 'B', 'B', '-', 'D', 'S'}; + private static final byte CAPTURE_START_EVENT = 0; + private static final byte CAPTURE_UPDATE_EVENT = 1; + private static final byte CAPTURE_END_EVENT = 2; + + public BlockStreamSender() { + dataToSend = new ByteArrayOutputStream(); + } + + public void connect(String host, String meetingId, int width, int height) throws ConnectionException { + this.meetingId = meetingId; + + System.out.println("Starting capturedScreenSender "); + try { + socket = new Socket(host, PORT); + outStream = new DataOutputStream(socket.getOutputStream()); + sendStartStreamMessage(meetingId, width, height); + outStream.flush(); + } catch (UnknownHostException e) { + e.printStackTrace(); + throw new ConnectionException("UnknownHostException: " + host); + } catch (IOException e) { + e.printStackTrace(); + throw new ConnectionException("IOException: " + host + ":" + PORT); + } + + sendCapturedScreen = true; + capturedScreenSender = new Runnable() { + public void run() { + while (sendCapturedScreen) { + try { + String block = screenQ.take(); + +// long now = System.currentTimeMillis(); +// if ((now - block.getTimestamp()) < 500) { + sendBlock(block); +// if (screenQ.size() == 500) screenQ.clear(); +// } else { +// System.out.println("Discarding stale block."); +// } + } catch (InterruptedException e) { + System.out.println("InterruptedExeption while taking event."); + } + } + } + }; + exec.execute(capturedScreenSender); + } + + private void sendStartStreamMessage(String meetingId, int width, int height) { + dataToSend.reset(); + + try { + dataToSend.write(CAPTURE_START_EVENT); + dataToSend.write(meetingId.length()); + dataToSend.write(meetingId.getBytes()); + dataToSend.write(intToByte(width)); + dataToSend.write(intToByte(height)); + sendToStream(dataToSend); + } catch (IOException e) { + e.printStackTrace(); + } + } + + private void sendBlock(String block) { + long start = System.currentTimeMillis(); + dataToSend.reset(); + try { + dataToSend.write(CAPTURE_UPDATE_EVENT); + dataToSend.write(meetingId.length()); + dataToSend.write(meetingId.getBytes()); + + sendToStream(dataToSend); + } catch (IOException e) { + e.printStackTrace(); + } + long end = System.currentTimeMillis(); + if ((end - start) > 200) { + System.out.println("Sending " + dataToSend.size() + " bytes took " + (end-start) + " ms."); + } + } + + private byte[] intToByte(int i) { + byte[] data = new byte[4]; + data[0] = (byte)((i >> 24) & 0xff); + data[1] = (byte)((i >> 16) & 0xff); + data[2] = (byte)((i >> 8) & 0xff); + data[3] = (byte)(i & 0xff); + return data; + } + + private void sendToStream(ByteArrayOutputStream data) throws IOException { +// System.out.println("Sending length " + data.size()); + outStream.write(HEADER); + outStream.writeInt(data.size()); + //outStream.write(data.toByteArray()); + data.writeTo(outStream); + } + + + public void disconnect() throws ConnectionException { + System.out.println("Closing connection."); + sendCapturedScreen = false; + + dataToSend.reset(); + try { + dataToSend.write(CAPTURE_END_EVENT); + dataToSend.write(meetingId.length()); + dataToSend.write(meetingId.getBytes()); + sendToStream(dataToSend); + } catch (IOException e) { + e.printStackTrace(); + } + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/CaptureEvents.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/CaptureEvents.java new file mode 100755 index 0000000000000000000000000000000000000000..0f123b8e9abb3ff829a72cd361d4e302094b6900 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/CaptureEvents.java @@ -0,0 +1,52 @@ +/** +* +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2010 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 2.1 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +**/ +package org.bigbluebutton.screenshare.client.net; + +public enum CaptureEvents { + /** + * WARNING: Must match corresponding values with deskshare-app on the server. + * org.bigbluebutton.deskshare.CaptureEvents + */ + CAPTURE_START(0), CAPTURE_UPDATE(1), CAPTURE_END(2), MOUSE_LOCATION_EVENT(3); + + private final int event; + + CaptureEvents(int event) { + this.event = event; + } + + public int getEvent() { + return event; + } + + @Override + public String toString() { + switch (event) { + case 0: + return "Capture Start Event"; + case 1: + return "Capture Update Event"; + case 2: + return "Capture End Event"; + } + + return "Unknown Capture Event"; + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ConnectionException.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ConnectionException.java new file mode 100755 index 0000000000000000000000000000000000000000..3b4bac17c0eb59c5e209bbc0cd35aacbe5692303 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ConnectionException.java @@ -0,0 +1,28 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +public class ConnectionException extends Exception { + + private static final long serialVersionUID = -8836714569259091334L; + + public ConnectionException(String message) { + super(message); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/Message.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/Message.java new file mode 100755 index 0000000000000000000000000000000000000000..621e670efb01338029189ac2f015d848f40834d5 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/Message.java @@ -0,0 +1,26 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +public interface Message { + + public enum MessageType {UPDATE, STARTED, STOPPED, CANCELLED, POISON}; + + public MessageType getMessageType(); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkConnectionListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkConnectionListener.java new file mode 100755 index 0000000000000000000000000000000000000000..54d67ab3ca79b25303b4d2b598fb73a80e6395e0 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkConnectionListener.java @@ -0,0 +1,26 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +import org.bigbluebutton.screenshare.client.ExitCode; + +public interface NetworkConnectionListener { + + public void networkConnectionException(ExitCode reason); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkHttpStreamSender.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkHttpStreamSender.java new file mode 100755 index 0000000000000000000000000000000000000000..f47baa56586741777a8b762c830e7eb694cf7a5a --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkHttpStreamSender.java @@ -0,0 +1,235 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.net; + +import java.io.IOException; +import java.net.HttpURLConnection; +import java.net.MalformedURLException; +import java.net.URL; +import java.net.URLConnection; +import org.bigbluebutton.screenshare.client.ExitCode; +import com.myjavatools.web.ClientHttpRequest; +import java.util.Date; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.LinkedBlockingQueue; +import java.text.SimpleDateFormat; + +public class NetworkHttpStreamSender { + private static final String SEQ_NUM = "sequenceNumber"; + private static final String MEETING_ID = "meetingId"; + private static final String STREAM_ID = "streamId"; + + private static final String EVENT = "event"; + private static final String SCREEN = "screenInfo"; + + private String host = "localhost"; + private static final String SCREEN_CAPTURE__URL = "/tunnel/screenCapture"; + private URL url; + private URLConnection conn; + private String meetingId; + private String streamId; + private NetworkStreamListener listener; + private final SequenceNumberGenerator seqNumGenerator; + + private ExecutorService executor; + private final BlockingQueue<Message> messages = new LinkedBlockingQueue<Message>(); + private volatile boolean sendMessages = false; + + public NetworkHttpStreamSender(String meetingId, String streamId, SequenceNumberGenerator seqNumGenerator) { + this.meetingId = meetingId; + this.streamId = streamId; + this.seqNumGenerator = seqNumGenerator; + + executor = Executors.newFixedThreadPool(1); + } + + public void addListener(NetworkStreamListener listener) { + this.listener = listener; + } + + private void notifyNetworkStreamListener(ExitCode reason) { + if (listener != null) listener.networkException(reason); + } + + public void connect(String host) throws ConnectionException { + this.host = host; + System.out.println("Starting NetworkHttpStreamSender to " + host); + openConnection(); + } + + public void send(Message message) { + messages.offer(message); + } + + public void start() { + sendMessages = true; + Runnable sender = new Runnable() { + public void run() { + while (sendMessages) { + Message message; + try { + message = messages.take(); + sendMessageToServer(message); + } catch (InterruptedException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + + } + } + }; + executor.execute(sender); + } + + private void sendMessageToServer(Message message) { + if (message.getMessageType() == Message.MessageType.UPDATE) { + sendUpdateMessage((ShareUpdateMessage) message); + } else if (message.getMessageType() == Message.MessageType.STARTED) { + sendStartStreamMessage((ShareStartedMessage)message); + } else if (message.getMessageType() == Message.MessageType.STOPPED) { + sendCaptureEndEvent(); + } + } + + public void stop() { + sendMessages = false; +} + + private void openConnection() throws ConnectionException { + /** + * Need to re-establish connection each time, otherwise, + * we get java.net.ProtocolException: Cannot write output after reading input. + * + * http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4382944 + * + */ + long start = System.currentTimeMillis(); + try { + url = new URL(host + SCREEN_CAPTURE__URL); + conn = url.openConnection(); + } catch (MalformedURLException e) { + e.printStackTrace(); + throw new ConnectionException("MalformedURLException " + url.toString()); + } catch (IOException e) { + e.printStackTrace(); + throw new ConnectionException("IOException while connecting to " + url.toString()); + } + long end = System.currentTimeMillis(); + System.out.println("Http Open connection took [" + (end-start) + " ms]"); + } + + private void sendStartStreamMessage(ShareStartedMessage message) { + try { + System.out.println("Http Open connection. In sendStartStreamMessage"); + openConnection(); + sendCaptureStartEvent(message.width, message.height); + } catch (ConnectionException e) { + e.printStackTrace(); + notifyNetworkStreamListener(ExitCode.DESKSHARE_SERVICE_UNAVAILABLE); + } + } + + private void sendCaptureStartEvent(int width, int height) throws ConnectionException { + ClientHttpRequest chr; + try { + System.out.println(getTimeStamp() + " - Sending Start Sharing Event."); + chr = new ClientHttpRequest(conn); + chr.setParameter(MEETING_ID, meetingId); + chr.setParameter(STREAM_ID, streamId); + chr.setParameter(SEQ_NUM, seqNumGenerator.getNext()); + String screenInfo = Integer.toString(width) + "x" + Integer.toString(height); + chr.setParameter(SCREEN, screenInfo); + chr.setParameter(EVENT, CaptureEvents.CAPTURE_START.getEvent()); + chr.post(); + } catch (IOException e) { + e.printStackTrace(); + throw new ConnectionException("IOException while sending capture start event."); + } + + } + + public void disconnect() throws ConnectionException { + try { + System.out.println("Http Open connection. In disconnect"); + openConnection(); + sendCaptureEndEvent(); + } catch (ConnectionException e) { + e.printStackTrace(); + notifyNetworkStreamListener(ExitCode.DESKSHARE_SERVICE_UNAVAILABLE); + throw e; + } finally { + + } + } + + private void sendCaptureEndEvent() { + ClientHttpRequest chr; + try { + System.out.println(getTimeStamp() + " - Sending End Sharing Event."); + chr = new ClientHttpRequest(conn); + chr.setParameter(MEETING_ID, meetingId); + chr.setParameter(STREAM_ID, streamId); + chr.setParameter(SEQ_NUM, seqNumGenerator.getNext()); + chr.setParameter(EVENT, CaptureEvents.CAPTURE_END.getEvent()); + chr.post(); + } catch (IOException e) { + e.printStackTrace(); + } + } + + private void sendUpdateMessage(ShareUpdateMessage message) { + ClientHttpRequest chr; + + try { + + // Open a connection to the web server and create a request that has + // the room and event type. + System.out.println(getTimeStamp() + " - Sending Update Sharing Event."); + openConnection(); + chr = new ClientHttpRequest(conn); + chr.setParameter(MEETING_ID, meetingId); + chr.setParameter(STREAM_ID, streamId); + chr.setParameter(EVENT, CaptureEvents.CAPTURE_UPDATE.getEvent()); + + // Post the multi-part form to the server + chr.post(); + HttpURLConnection httpConnection = (HttpURLConnection) chr.connection; + int status = httpConnection.getResponseCode(); + + System.out.println("******* POST status = [" + status + "] ***************"); + + } catch (IOException e) { + notifyNetworkStreamListener(ExitCode.NORMAL); + } catch (ConnectionException e) { + System.out.println("ERROR: Failed to send block data."); + } + } + + + private String getTimeStamp() + { + SimpleDateFormat sdfDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSSS");//dd/MM/yyyy + Date now = new Date(); + String strDate = sdfDate.format(now); + return strDate; + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamListener.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamListener.java new file mode 100755 index 0000000000000000000000000000000000000000..de81405e2afb306e47bfb27f67d1eefd486149b7 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamListener.java @@ -0,0 +1,26 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +import org.bigbluebutton.screenshare.client.ExitCode; + +public interface NetworkStreamListener { + + public void networkException(ExitCode reason); +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamSender.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamSender.java new file mode 100755 index 0000000000000000000000000000000000000000..b705fa1f52016e8dc50be36d006471c408f3ed7f --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NetworkStreamSender.java @@ -0,0 +1,117 @@ +/** + * BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ + * + * Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). + * + * This program is free software; you can redistribute it and/or modify it under the + * terms of the GNU Lesser General Public License as published by the Free Software + * Foundation; either version 3.0 of the License, or (at your option) any later + * version. + * + * BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY + * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A + * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License along + * with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. + * + */ +package org.bigbluebutton.screenshare.client.net; + +import java.util.Timer; +import java.util.TimerTask; +import org.bigbluebutton.screenshare.client.ExitCode; + +public class NetworkStreamSender implements NetworkStreamListener { + public static final String NAME = "NETWORKSTREAMSENDER: "; + + private final String meetingId; + private final String streamId; + private NetworkHttpStreamSender httpSenders; + private NetworkConnectionListener listener; + private final SequenceNumberGenerator seqNumGenerator = new SequenceNumberGenerator(); + private String host = "192.168.23.22"; + + private TimerTask timerTask = new UpdateTimerTask(); + private Timer timer = new Timer(); + + public NetworkStreamSender(String host, String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + this.host = host; + connect(); + } + + public void addNetworkConnectionListener(NetworkConnectionListener listener) { + this.listener = listener; + } + + private void notifyNetworkConnectionListener(ExitCode reason) { + if (listener != null) listener.networkConnectionException(reason); + } + + private boolean connect() { + httpSenders = new NetworkHttpStreamSender(meetingId, streamId, seqNumGenerator); + httpSenders.addListener(this); + try { + httpSenders.connect(host); + } catch (ConnectionException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + return true; + } + + + public void stopSharing() { + System.out.println("Queueing ShareStoppedMessage"); + send(new ShareStoppedMessage(meetingId, streamId)); + } + + public void startSharing(int width, int height) { + System.out.println("Queueing ShareStartedMessage"); + send(new ShareStartedMessage(meetingId, streamId, width, height)); + } + + private void send(Message message) { + httpSenders.send(message); + } + + public void start() { + System.out.println(NAME + "Starting network sender."); + httpSenders.start(); + timer.scheduleAtFixedRate(timerTask, 0, 2 * 1000); + } + + public void stop() throws ConnectionException { + timer.cancel(); + + if (httpSenders != null) { + httpSenders.disconnect(); + httpSenders.stop(); + } + + } + + + @Override + public void networkException(ExitCode reason) { + try { + System.out.println(NAME + "Failed to use http tunneling. Stopping."); + stop(); + notifyNetworkConnectionListener(reason); + } catch (ConnectionException e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + } + + private class UpdateTimerTask extends TimerTask { + @Override + public void run() { + System.out.println("Queueing ShareUpdateMessage"); + send(new ShareUpdateMessage(meetingId, streamId)); + } + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NextBlockRetriever.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NextBlockRetriever.java new file mode 100755 index 0000000000000000000000000000000000000000..cc1d3be3524f880b6449da9896a33a4f45ae77a0 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/NextBlockRetriever.java @@ -0,0 +1,24 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +public interface NextBlockRetriever { + public void blockSent(int position); + public Message getNextMessageToSend() throws InterruptedException; +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/PoisonMessage.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/PoisonMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..c9fcf88fbb20eabc1dc40ea46cf69c2983f1735d --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/PoisonMessage.java @@ -0,0 +1,27 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +public class PoisonMessage implements Message { + + @Override + public MessageType getMessageType() { + return MessageType.POISON; + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ScreenCaptureSender.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ScreenCaptureSender.java new file mode 100755 index 0000000000000000000000000000000000000000..5c4dc082d8fa07bf6acb834b69f123413c4cbfc2 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ScreenCaptureSender.java @@ -0,0 +1,26 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +public interface ScreenCaptureSender { + + public void connect(String host, String meetingId, int width, int height) throws ConnectionException; + public void disconnect() throws ConnectionException; + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/SequenceNumberGenerator.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/SequenceNumberGenerator.java new file mode 100755 index 0000000000000000000000000000000000000000..895bd5efa5e146f0785663d89ea8b02f7cfdf49e --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/SequenceNumberGenerator.java @@ -0,0 +1,35 @@ +/** +* BigBlueButton open source conferencing system - http://www.bigbluebutton.org/ +* +* Copyright (c) 2012 BigBlueButton Inc. and by respective authors (see below). +* +* This program is free software; you can redistribute it and/or modify it under the +* terms of the GNU Lesser General Public License as published by the Free Software +* Foundation; either version 3.0 of the License, or (at your option) any later +* version. +* +* BigBlueButton is distributed in the hope that it will be useful, but WITHOUT ANY +* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A +* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. +* +* You should have received a copy of the GNU Lesser General Public License along +* with BigBlueButton; if not, see <http://www.gnu.org/licenses/>. +* +*/ +package org.bigbluebutton.screenshare.client.net; + +import java.util.concurrent.atomic.AtomicInteger; + + +public class SequenceNumberGenerator { + + private final AtomicInteger sequenceNum; + + public SequenceNumberGenerator() { + sequenceNum = new AtomicInteger(0); + } + + public int getNext() { + return sequenceNum.incrementAndGet(); + } +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStartedMessage.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStartedMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..3385403177813ad773cfa413de5861a91e43ba11 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStartedMessage.java @@ -0,0 +1,22 @@ +package org.bigbluebutton.screenshare.client.net; + +public class ShareStartedMessage implements Message { + + public final String meetingId; + public final String streamId; + public final int width; + public final int height; + + public ShareStartedMessage(String meetingId, String streamId, int width, int height) { + this.meetingId = meetingId; + this.streamId = streamId; + this.width = width; + this.height = height; + } + + @Override + public MessageType getMessageType() { + return Message.MessageType.STARTED; + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStoppedMessage.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStoppedMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..19418e83edbb890f767143c86082a35c67eb195b --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareStoppedMessage.java @@ -0,0 +1,18 @@ +package org.bigbluebutton.screenshare.client.net; + +public class ShareStoppedMessage implements Message { + + public final String meetingId; + public final String streamId; + + public ShareStoppedMessage(String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + } + + @Override + public MessageType getMessageType() { + return Message.MessageType.STOPPED; + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareUpdateMessage.java b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareUpdateMessage.java new file mode 100755 index 0000000000000000000000000000000000000000..cb390a67ebb347de4d804fe89d67ab6e3c38ec69 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/main/java/org/bigbluebutton/screenshare/client/net/ShareUpdateMessage.java @@ -0,0 +1,18 @@ +package org.bigbluebutton.screenshare.client.net; + +public class ShareUpdateMessage implements Message { + + public final String meetingId; + public final String streamId; + + public ShareUpdateMessage(String meetingId, String streamId) { + this.meetingId = meetingId; + this.streamId = streamId; + } + + @Override + public MessageType getMessageType() { + return Message.MessageType.UPDATE; + } + +} diff --git a/bbb-screenshare/jws/webstart/src/main/resources/images/Cursor.png b/bbb-screenshare/jws/webstart/src/main/resources/images/Cursor.png new file mode 100755 index 0000000000000000000000000000000000000000..6e29c43729b8711191d7120348e15bbcfa236dd0 Binary files /dev/null and b/bbb-screenshare/jws/webstart/src/main/resources/images/Cursor.png differ diff --git a/bbb-screenshare/jws/webstart/src/main/resources/images/bbb.gif b/bbb-screenshare/jws/webstart/src/main/resources/images/bbb.gif new file mode 100755 index 0000000000000000000000000000000000000000..b01a452375a5213065dc237f34f22d97c3f69473 Binary files /dev/null and b/bbb-screenshare/jws/webstart/src/main/resources/images/bbb.gif differ diff --git a/bbb-screenshare/jws/webstart/src/main/resources/images/move-cursor.png b/bbb-screenshare/jws/webstart/src/main/resources/images/move-cursor.png new file mode 100755 index 0000000000000000000000000000000000000000..4e64fc6e4ae7f6653495dd9bf22f5d09435343d0 Binary files /dev/null and b/bbb-screenshare/jws/webstart/src/main/resources/images/move-cursor.png differ diff --git a/bbb-screenshare/jws/webstart/src/main/resources/images/resize-cursor.png b/bbb-screenshare/jws/webstart/src/main/resources/images/resize-cursor.png new file mode 100755 index 0000000000000000000000000000000000000000..50d5f933c6eaf6a6470a16935d5ece83e74182c2 Binary files /dev/null and b/bbb-screenshare/jws/webstart/src/main/resources/images/resize-cursor.png differ diff --git a/bbb-screenshare/jws/webstart/src/test/resources/testng.xml b/bbb-screenshare/jws/webstart/src/test/resources/testng.xml new file mode 100755 index 0000000000000000000000000000000000000000..8a186d44ccc73481ead0fc211f8b3f641434cdf5 --- /dev/null +++ b/bbb-screenshare/jws/webstart/src/test/resources/testng.xml @@ -0,0 +1,18 @@ +<!DOCTYPE suite SYSTEM "http://beust.com/testng/testng-1.0.dtd" > +<suite name="BigBlueButton Test Suite"> + <test name="Conference tests"> + <groups> + <run> + <exclude name="broken"/> + <include name="unit"/> + </run> + </groups> + + <packages> + <package name="org.bigbluebutton.deskshare.client.encode"/> + <package name="org.bigbluebutton.deskshare.client.encode3"/> + <package name="org.bigbluebutton.deskshare.client.net"/> + <package name="org.bigbluebutton.deskshare.client"/> + </packages> + </test> +</suite> \ No newline at end of file