跟 Red5 类似,Wowza 支持服务扩展,用户可以进行自定义应用程序开发,然后将其作为一个模块部署在 Wowza 服务器。Red5 提供了一个 Eclipse 插件进行应用扩展开发(参见《eclipse 的 Red5 插件安装简介》),Wowza 则提供了一个 IDE。本文简要介绍如何使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用。《安装并使用 Wowza 发布你的 RTMP 直播流》一文介绍了如何安装 Wowza 服务器并提供直播服务,本文将继续以此为例,介绍如何使用 Wowza IDE 开发应用对每个流频道进行监控。
I. 下载 Wowza IDE
       官方下载地址 http://wowza.cn/mediaserver/developers,选择适合你自己的平台的版本进行下载。
       作者上传了一个 Windows 版本的到 CSDN 资源以做备份,如果看官嫌从官网下载速度太慢,可以点击下载:
WowzaIDE-2.0.0.exe
II. 安装
       Windows 下直接运行步骤 I 下载的 WowzaIDE-2.0.0.exe。
       安装好以后,开始 -> 程序 -> Wowza IDE 2 -> Wowza IDE 2 启动 IDE,选择一个目录作为你的工作台,进入后的界面跟 Eclipse 一般无二:
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第1张图片
III. 新建项目

       File -> New -> Wowza Media Server Project,打开新建项目向导,输入项目名 defonds-live-module:

使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第2张图片

       其中,新项目名 defonds-live-module 也会作为 .jar 的文件名,之后作为一个模块被 Wowza IDE 自动部署在 Wowza 服务器 wowza/lib 目录下;Wowza Media Server /Location 应该指向你的 Wowza 服务器的安装目录。

       点击 Next > 按钮,进入新建 WMS 模块类对话框:

使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第3张图片

       包名栏输入:com.defonds.wms.module;
       类名栏输入:DefondsLiveModule;
       自定义方法名输入:doSomething,这个方法可以被客户端直接调用(NetConnection.call(“doSomething”, null);)。类 DefondsLiveModule 创建以后,你可以使用 doSomething 同样的标签来创建更多自定义方法;
       Event Methods 部分是留给你捕捉一系列事件的接口,在这些事件发生时,这些方法将被调用。本文例子里保持默认选择,点击 Finish 按钮。
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第4张图片
       IDE 会创建 WMS 模块项目,创建模块类,创建一个运行命令并编译和绑定模块类到一个 jar 文件里,这个 jar 文件会被自动部署到 WMS 安装目录下:
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第5张图片
       最后编辑 DefondsLiveModule 类内容如下:

[java] view plain copy print ?
  1. package com.defonds.wms.module;  

  2. import java.util.HashMap;  

  3. import java.util.Map;  

  4. import com.wowza.wms.livestreamrecord.model.ILiveStreamRecord;  

  5. import com.wowza.wms.livestreamrecord.model.LiveStreamRecorderMP4;  

  6. import com.wowza.wms.media.model.MediaCodecInfoAudio;  

  7. import com.wowza.wms.media.model.MediaCodecInfoVideo;  

  8. import com.wowza.wms.module.*;  

  9. import com.wowza.wms.stream.*;  

  10. import com.wowza.wms.amf.AMFPacket;  

  11. import com.wowza.wms.application.IApplicationInstance;  

  12. import com.wowza.wms.application.WMSProperties;  

  13. publicclass DefondsLiveModule extends ModuleBase implements IModuleOnStream  

  14. {  

  15. private Map recorders = new HashMap();  

  16. private IApplicationInstance appInstance;  

  17. publicvoid onAppStart(IApplicationInstance appInstance)  

  18.    {  

  19. this.appInstance = appInstance;  

  20.    }  

  21. class StreamListener implements IMediaStreamActionNotify3  

  22.    {  

  23. publicvoid onMetaData(IMediaStream stream, AMFPacket metaDataPacket)  

  24.        {  

  25.            System.out.println("onMetaData[" + stream.getContextStr() + "]: " + metaDataPacket.toString());  

  26.        }  

  27. publicvoid onPauseRaw(IMediaStream stream, boolean isPause, double location)  

  28.        {  

  29.            System.out.println("onPauseRaw[" + stream.getContextStr() + "]: isPause:" + isPause + " location:" + location);  

  30.        }  

  31. publicvoid onPause(IMediaStream stream, boolean isPause, double location)  

  32.        {  

  33.            System.out.println("onPause[" + stream.getContextStr() + "]: isPause:" + isPause + " location:" + location);  

  34.        }  

  35. publicvoid onPlay(IMediaStream stream, String streamName, double playStart, double playLen, int playReset)  

  36.        {  

  37.            System.out.println("onPlay[" + stream.getContextStr() + "]: playStart:" + playStart + " playLen:" + playLen + " playReset:" + playReset);  

  38.        }  

  39. publicvoid onPublish(IMediaStream stream, String streamName, boolean isRecord, boolean isAppend)  

  40.        {  

  41.            System.out.println("onPublish[" + stream.getContextStr() + "]: streamName:" + streamName + " isRecord:" + isRecord + " isAppend:" + isAppend);  

  42. //create a livestreamrecorder instance to create .mp4 files

  43.            ILiveStreamRecord recorder = new LiveStreamRecorderMP4();  

  44.            recorder.init(appInstance);  

  45.            recorder.setRecordData(true);  

  46.            recorder.setStartOnKeyFrame(true);  

  47.            recorder.setVersionFile(true);  

  48. // add it to the recorders list

  49. synchronized (recorders)  

  50.            {  

  51.                ILiveStreamRecord prevRecorder = recorders.get(streamName);  

  52. if (prevRecorder != null)  

  53.                    prevRecorder.stopRecording();  

  54.                recorders.put(streamName, recorder);  

  55.            }  

  56. // start recording, create 1 minute segments using default content path

  57.            System.out.println("--- startRecordingSegmentByDuration for 60 minutes");  

  58.            recorder.startRecordingSegmentByDuration(stream, null, null, 60*60*1000);  

  59. // start recording, create 1MB segments using default content path

  60. //System.out.println("--- startRecordingSegmentBySize for 1MB");

  61. //recorder.startRecordingSegmentBySize(stream, null, null, 1024*1024);

  62. // start recording, create new segment at 1:00am each day.

  63. //System.out.println("--- startRecordingSegmentBySchedule every "0 1 * * * *");

  64. //recorder.startRecordingSegmentBySchedule(stream, null, null, "0 1 * * * *");

  65. // start recording, using the default content path, do not append (i.e. overwrite if file exists)

  66. //System.out.println("--- startRecording");

  67. //recorder.startRecording(stream, false);

  68. // log where the recording is being written

  69.            System.out.println("onPublish[" + stream.getContextStr() + "]: new Recording started:" + recorder.getFilePath());  

  70.        }  

  71. publicvoid onUnPublish(IMediaStream stream, String streamName, boolean isRecord, boolean isAppend)  

  72.        {  

  73.            System.out.println("onUnPublish[" + stream.getContextStr() + "]: streamName:" + streamName + " isRecord:" + isRecord + " isAppend:" + isAppend);  

  74.            ILiveStreamRecord recorder = null;  

  75. synchronized (recorders)  

  76.            {  

  77.                recorder = recorders.remove(streamName);  

  78.            }  

  79. if (recorder != null)  

  80.            {  

  81. // grab the current path to the recorded file

  82.                String filepath = recorder.getFilePath();  

  83. // stop recording

  84.                recorder.stopRecording();  

  85.                System.out.println("onUnPublish[" + stream.getContextStr() + "]: File Closed:" + filepath);  

  86.            }  

  87. else

  88.            {  

  89.                System.out.println("onUnPublish[" + stream.getContextStr() + "]: streamName:" + streamName + " stream recorder not found");  

  90.            }  

  91.        }  

  92. publicvoid onSeek(IMediaStream stream, double location)  

  93.        {  

  94.            System.out.println("onSeek[" + stream.getContextStr() + "]: location:" + location);  

  95.        }  

  96. publicvoid onStop(IMediaStream stream)  

  97.        {  

  98.            System.out.println("onStop[" + stream.getContextStr() + "]: ");  

  99.        }  

  100. publicvoid onCodecInfoAudio(IMediaStream stream,MediaCodecInfoAudio codecInfoAudio) {  

  101.            System.out.println("onCodecInfoAudio[" + stream.getContextStr() + " Audio Codec" + codecInfoAudio.toCodecsStr() + "]: ");  

  102.        }  

  103. publicvoid onCodecInfoVideo(IMediaStream stream,MediaCodecInfoVideo codecInfoVideo) {  

  104.            System.out.println("onCodecInfoVideo[" + stream.getContextStr() + " Video Codec" + codecInfoVideo.toCodecsStr() + "]: ");  

  105.        }  

  106.    }  

  107. publicvoid onStreamCreate(IMediaStream stream)  

  108.    {  

  109.        getLogger().info("onStreamCreate["+stream+"]: clientId:" + stream.getClientId());  

  110.        IMediaStreamActionNotify3 actionNotify = new StreamListener();  

  111.        WMSProperties props = stream.getProperties();  

  112. synchronized (props)  

  113.        {  

  114.            props.put("streamActionNotifier", actionNotify);  

  115.        }  

  116.        stream.addClientListener(actionNotify);  

  117.    }  

  118. publicvoid onStreamDestroy(IMediaStream stream)  

  119.    {  

  120.        getLogger().info("onStreamDestroy["+stream+"]: clientId:" + stream.getClientId());  

  121.        IMediaStreamActionNotify3 actionNotify = null;  

  122.        WMSProperties props = stream.getProperties();  

  123. synchronized (props)  

  124.        {  

  125.            actionNotify = (IMediaStreamActionNotify3) stream.getProperties().get("streamActionNotifier");  

  126.        }  

  127. if (actionNotify != null)  

  128.        {  

  129.            stream.removeClientListener(actionNotify);  

  130.            getLogger().info("removeClientListener: " + stream.getSrc());  

  131.        }  

  132.    }  

  133. }  


IV. 导入例子模块的类
       Package Explorer 视图下,右击 defonds-live-module 项目中 src 下的 com.defonds.wms.module -> 选择 Import… -> 在导入对话框里,展开 General 文件夹 -> 选中 File System 后点击 Next > 按钮 -> 点击 Browse… 按钮 -> 浏览至 %Wowza%/examples/ServerSideModules/server 文件夹(这个是 Wowza 服务器安装的一部分) 后确定 -> 勾选 ModuleServerSide.java 后点 Finish 按钮,如下图所示。
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第6张图片
       这样子我们就把 ModuleServerSide.java 给导入进来了,Package Explorer 视图中双击导入的类名,发现有编译错误:
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第7张图片
       将 package com.mycompany.wms.module; 换成我们自定义的包名 package com.defonds.wms.module; 即可。
V. 配置 Application.xml
       现在我们已经使用 Wowza IDE 构建好了我们自己定义的 defonds-live-module.jar,我们还需要指示 Wowza 服务器加载这个新模块。
       编辑 %Wowza%/conf/live/Application.xml 文件,将以下模块定义添加进 部分的结尾:

[html] view plain copy print ?
  1. <Module>

  2. <Name>DefondsLiveModuleName>

  3. <Description>DefondsLiveModuleDescription>

  4. <Class>com.defonds.wms.module.DefondsLiveModuleClass>

  5. Module>

  6. <Module>

  7. <Name>ModuleServerSideName>

  8. <Description>Defonds ModuleServerSideDescription>

  9. <Class>com.defonds.wms.module.ModuleServerSideClass>

  10. Module>


       这时我们的新模块才正式生效了。如果这一步不明白可以去看《安装并使用 Wowza 发布你的 RTMP 直播流》。
VI. 启动服务

       这时我们可以在 Wowza IDE 内部启动 Wowza 服务器了。如果你已经启动了一个 service 模式或者 standalone 模式的 Wowza 服务器,那么你要先将其关闭。点击 Wowza IDE 的工具栏里的 Run 菜单里的 WowzaMediaServer_defonds-live-module 启动 Wowza 服务器。当然你也可以点击 Debug 菜单里的 WowzaMediaServer_defonds-live-module 以 debug 模式启动 Wowza。

使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第8张图片

       这时 Wowza 服务器启动起来了,在 Wowza IDE 的下部的控制台标签里可以看到所有的控制台 log 输出。如同 Eclipse 中的 Tomcat,你可以在控制台窗口中点击关闭图标来停止服务器运行。

使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第9张图片
VII. 模块调试
       现在我们来测试一下新模块的运行情况。使用你的 RTMP Client 发送 RTMP 流到 Wowza,比如 Server URL 为 rtmp://172.21.30.104/live,流名为 xxxxS_2059a0734ccfqvga,成功连接 Wowza 服务器。
使用 Wowza IDE 开发第一个 Wowza 服务器扩展应用 -- 监控直播频道_第10张图片
       Wowza IDE 控制台有 onPublish[live/_definst_/xxxxS_2059a0734ccfqvga]: streamName:xxxxS_2059a0734ccfqvga isRecord:false isAppend:false 输出,这个正是我们 DefondsLiveModule 类里的 StreamListener.onPublish 里定义的,测试成功。
参考资料

  • Wowza IDE User’s Guide

  • http://www.wowza.com/forums/content.php?472-How-to-start-and-stop-live-stream-recordings-programmatically-(IMediaStreamActionNotify3)