腾讯云视频接入初探

今日来公司的项目要求接入视频,如是根据需求对网易云视频,anychat,腾讯云视频进行了技术调研,因为基于响应时间,降噪等诸多效果的测试,考虑,最终选择了腾讯云视频。因为腾讯云视频刚刚推出不久,demo,文档写的不是太集中,故在集成时出现多个坑。现对集成步骤做一下简单归纳:

1.集成播放列表:

直播播放列表需要开通查询统计信息(Beta)

该接口的相关文档地址:https://www.qcloud.com/document/product/267/6110
这个接口需要提交工单才会开通,默认是不开通的。
开通后会产生appid和推流鉴权key (这个东西会在生成sign时使用,sign生成的规则是推流鉴权key直接拼接过期时间)

 String playUrl ="http://statcgi.video.qcloud.com/common_access?"+
            "cmd=1251175924&interface=Get_LiveStat"+
                    "&Param.n.page_no=1"+"&Param.n.page_size=20"+
                    "&t="+time+"&sign="+signValue;

cmd后面需要根据你自己的需要更换成自己的appid,我的核心代码如下:
sign是推流鉴权key直接拼接过期时间进行md5生成:

md5工具类:

public class MD5Utils {
      static String MD5(String sourceStr) {
            String result = "";
            try {
                MessageDigest md = MessageDigest.getInstance("MD5");
                md.update(sourceStr.getBytes());
                byte b[] = md.digest();
                int i;
                StringBuffer buf = new StringBuffer("");
                for (int offset = 0; offset < b.length; offset++) {
                    i = b[offset];
                    if (i < 0)
                        i += 256;
                    if (i < 16)
                        buf.append("0");
                    buf.append(Integer.toHexString(i));
                }
                result = buf.toString();
                System.out.println("MD5(" + sourceStr + ",32) = " + result);
                System.out.println("MD5(" + sourceStr + ",16) = " + buf.toString().substring(8, 24));
            } catch (NoSuchAlgorithmException e) {
                System.out.println(e);
            }
            return result;
        }

}

过期时间戳转化工具类:

public class TimeUtils {
//日期转化为时间戳
     public static String dateToStamp(String s) throws Exception{
            String res;
            SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            Date date = simpleDateFormat.parse(s);
            long ts = (date.getTime()+300000)/1000;
            res = String.valueOf(ts);
            return res;
        }
     public static String stampToDate(String s){
            String res;
            SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            long lt = new Long(s);
            Date date = new Date(lt);
            res = simpleDateFormat.format(date);
            return res;
        }
}

查询时间列表的核心逻辑:

    public static void main(String[] args) throws Exception {
        OkHttpClient client = new OkHttpClient();
       SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
          String time="";
        try {
            //获取当前日期的时间戳
            time = TimeUtils.dateToStamp(sdf.format(new Date()));
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
           System.out.println(time);
             //推流鉴权key
            String key="7407d5829e270dd92661ffcf06169bcf";
            String signValue=MD5Utils.MD5(key+time);
            String playUrl ="http://statcgi.video.qcloud.com/common_access?"+
            "cmd=1251175924&interface=Get_LiveStat"+
                    "&Param.n.page_no=1"+"&Param.n.page_size=20"+
                    "&t="+time+"&sign="+signValue;
            System.out.println(playUrl);
        Request request = new Request.Builder().url(playUrl).build();
        Response response = client.newCall(request).execute();
        
        if (response.isSuccessful()) {
            System.out.println(response.body().string());
            
        } else {
            throw new IOException("Unexpected code " + response);
        }

获取成功会如下显示:

{"errmsg":"has no stream is living","message":"has no stream is living","output":null,"ret":0,"retcode":0}

如果有推流的话是:

{"errmsg":"","message":"","output":{"stream_count":1,"stream_info":[{"bandwidth":0.0,"client_ip":"125.46.216.125","flr":1.0,"fps":0,"online":0,"server_ip":"123.138.162.79","speed":0,"stream_name":"8818_b262bb2709","time":"2017-04-13 11:04:13"}],"total_bandwidth":0.0,"total_online":0},"ret":0,"retcode":0}

返回的是json这个由你自己的业务需求定

2.登录用户体系:

腾讯提供了两种账户体系,一种是托管的,一种是独立的,托管的是需要注册登录腾讯tls体系,账户管理交由腾讯,独立的就是不对现有用户体系有影响,而是在需要调用腾讯的时候,去到腾讯上验证一下,就像融云的token一样。

2.1托管体系注册

android:


accountHelper.TLSStrAccReg("xtfgq", "12345678", new TLSStrAccRegListener() {
            @Override
            public void OnStrAccRegSuccess(TLSUserInfo tlsUserInfo) {
//                tlsUserInfo.

                Log.e("vv","111");
            }

            @Override
            public void OnStrAccRegFail(TLSErrInfo tlsErrInfo) {
                Log.e("vv","222");
            }

            @Override
            public void OnStrAccRegTimeout(TLSErrInfo tlsErrInfo) {
                Log.e("vv","333");
            }
        });

需要在oncreate中完成初始化,这里面的appid与前面的不同,(特别注意),这个是开通腾讯云云通信里面的那个千万不要搞错。

 accountHelper=TLSAccountHelper.getInstance().init(SettingActivity.this,1251175924,12001,"1.0.1");

托管登录:

  loginHelper = TLSLoginHelper.getInstance()
                .init(getApplicationContext(), 11270, 12001, "1.0.1");
        loginHelper.TLSPwdLogin("用户帐号", "用户密码".getBytes(), new TLSPwdLoginListener(){
            @Override
            public void OnPwdLoginSuccess(TLSUserInfo tlsUserInfo) {
  final TIMUser user = new TIMUser();
                user.setIdentifier("用户帐号");
                user.setAccountType("12001");
                user.setAppIdAt3rd(String.valueOf(Constants.sdkAppId));
                TIMManager.getInstance().login(
                        Constants.sdkAppId,                   //sdkAppId,由腾讯分配
                        user,
                        loginHelper.getUserSig("用户帐号"),                    //用户帐号签名,由私钥加密获得,具体请参考文档
                        new TIMCallBack() {//回调接口

                            @Override
                            public void onSuccess() {//登录成功
                                Log.e("vvv","succuss");
                            //根据逻辑跳到不同界面
                            }

                            @Override
                            public void onError(int code, String desc) {//登录失败

                                //错误码code和错误描述desc,可用于定位请求失败原因
                                //错误码code含义请参见错误码表
                                Log.e("eer","eeror");

                            }
                        });

            }

            @Override
            public void OnPwdLoginReaskImgcodeSuccess(byte[] bytes) {

            }

            @Override
            public void OnPwdLoginNeedImgcode(byte[] bytes, TLSErrInfo tlsErrInfo) {

            }

            @Override
            public void OnPwdLoginFail(TLSErrInfo tlsErrInfo) {

            }

            @Override
            public void OnPwdLoginTimeout(TLSErrInfo tlsErrInfo) {

            }
        });

2.2 独立模式,即usersinger需从自己的服务器获得,可参考usersinger生成文档:https://www.qcloud.com/document/product/269/1510

客户端不需要像刚才那样先登陆tls,然后在成功回调里调用 TIMManager.getInstance().login,而是直接从自己的服务取得usersinger,
代码如下:

 final TIMUser user = new TIMUser();
                        user.setIdentifier(userid);
                        user.setAccountType("12001");
                        user.setAppIdAt3rd(String.valueOf(Constants.sdkAppId));
                        TIMManager.getInstance().login(
                                Constants.sdkAppId,                   //sdkAppId,由腾讯分配
                                user,
                               sig,                    //用户帐号签名,由私钥加密获得,具体请参考文档
                                new TIMCallBack() {//回调接口

                                    @Override
                                    public void onSuccess() {//登录成功
                                        Log.e("vvv","succuss");
                                    
                                    

                                    }

                                    @Override
                                    public void onError(int code, String desc) {//登录失败

                                        //错误码code和错误描述desc,可用于定位请求失败原因
                                        //错误码code含义请参见错误码表
                                        Log.e("eer","eeror");

                                    }
                                });

最后,不管哪中模式,都需要调用修改昵称的腾讯接口,否则会出现发弹幕上用户名变成数字方面的问题。

 public void setMyNickName(String nickName){
        TIMFriendshipManager.getInstance().setNickName(nickName, new TIMCallBack() {
            @Override
            public void onError(int i, String s) {

            }

            @Override
            public void onSuccess() {

            }
        });
    }

3.加入聊天室:

 private void joinIMChatRoom(final String chatRoomId) {

        mGroupConversation = TIMManager.getInstance().getConversation(TIMConversationType.Group, chatRoomId);

        TIMGroupManager.getInstance().applyJoinGroup(chatRoomId, Constants.APPLY_CHATROOM + chatRoomId, new TIMCallBack() {
            @Override
            public void onError(int i, String s) {
                //已经在是成员了
                if (i == Constants.IS_ALREADY_MEMBER) {

                    initTIMListener(chatRoomId);

                }
            }

            @Override
            public void onSuccess() {

                initTIMListener(chatRoomId);
            }
        });

    }

消息监听:

  public void initTIMListener(String chatRoomId) {
        mGroupConversation = TIMManager.getInstance().getConversation(TIMConversationType.Group, chatRoomId);
        TIMManager.getInstance().addMessageListener(msgListener);

    }
 /**
     * 群消息回调
     */
    private TIMMessageListener msgListener = new TIMMessageListener() {
        @Override
        public boolean onNewMessages(List<TIMMessage> list) {
            //SxbLog.d(TAG, "onNewMessages readMessage " + list.size());
            //解析TIM推送消息
            parseIMMessage(list);
            return false;
        }
    };
 /**
     * 解析消息回调
     *
     * @param list 消息列表
     */
    private void parseIMMessage(List<TIMMessage> list) {
        List<TIMMessage> tlist = list;

        if (tlist.size() > 0) {
            if (mGroupConversation != null)
                mGroupConversation.setReadMessage(tlist.get(0));
        }


        for (int i = tlist.size() - 1; i >= 0; i--) {
            TIMMessage currMsg = tlist.get(i);
            for (int j = 0; j < currMsg.getElementCount(); j++) {
                if (currMsg.getElement(j) == null)
                    continue;
                TIMElem elem = currMsg.getElement(j);
                TIMElemType type = elem.getType();

                //系统消息
                if (type == TIMElemType.GroupSystem) {
                    if (TIMGroupSystemElemType.TIM_GROUP_SYSTEM_DELETE_GROUP_TYPE == ((TIMGroupSystemElem) elem).getSubtype()) {

                    }


                }

                //最后处理文本消息
                if (type == TIMElemType.Text) {
                    if (currMsg.isSelf()) {
                        handleTextMessage(elem, name);
                    } else {
                        TIMUserProfile sendUser = currMsg.getSenderProfile();
                       String nick="";
                        if(sendUser!=null){
                            nick=sendUser.getNickName();
                        }else{
                            nick=currMsg.getSender();
                        }
                        handleTextMessage(elem,nick);
                    }
                }
            }
        }
    }

发送消息:

 mGroupConversation = TIMManager.getInstance().getConversation(TIMConversationType.Group, groupId);

        mGroupConversation.sendMessage(msg, new TIMValueCallBack<TIMMessage>() {//发送消息回调
            @Override
            public void onError(int code, String desc) {//发送消息失败
                //错误码code和错误描述desc,可用于定位请求失败原因
                //错误码code含义请参见错误码表
                if (code == 85) { //消息体太长
                    if (null != getActivity()) {
                        Toast.makeText(getActivity(), "输入内容太长", Toast.LENGTH_SHORT).show();
                    }
                } else if (code == 6011) {//群主不存在
                    if (null != getActivity()) {
                        Toast.makeText(getActivity(), "群主不存在", Toast.LENGTH_SHORT).show();
                    }
                }

            }

            @Override
            public void onSuccess(TIMMessage timMessage) {//发送消息成功
                mBoolRefreshLock=false;
                for (int j = 0; j < timMessage.getElementCount(); j++) {
                    TIMElem elem = (TIMElem) timMessage.getElement(0);
                    if (timMessage.isSelf()) {
                        handleTextMessage(elem, name);
                    } else {
                        TIMUserProfile sendUser = timMessage.getSenderProfile();
                        String name;
                        if (sendUser != null) {
                            name = sendUser.getNickName();
                        }
                        else {
                            name = timMessage.getSender();
                        }
                        //String sendId = timMessage.getSender();

                        handleTextMessage(elem,name);
                    }
                }
            }
        });
    }

处理消息文本:

  /**
     * 处理文本消息解析
     *
     * @param elem
     * @param name
     */
    private void handleTextMessage(TIMElem elem, String name) {
        TIMTextElem textElem = (TIMTextElem) elem;
        if (textElem.getText() != null) {
            refreshTextListView(name, textElem.getText(), Constants.TEXT_TYPE);
        }

    }
 /**
     * 消息刷新显示
     *
     * @param name    发送者
     * @param context 内容
     * @param type    类型 (上线线消息和 聊天消息)
     */
    public void refreshTextListView(String name, String context, int type) {
        ChatEntity entity = new ChatEntity();
        entity.setSenderName(name);
        entity.setContext(context);
        entity.setType(type);
        notifyRefreshListView(entity);
        mListViewMsgItems.setVisibility(View.VISIBLE);
        if (mListViewMsgItems.getCount() > 1) {
            if (true)
                mListViewMsgItems.setSelection(0);
            else
                mListViewMsgItems.setSelection(mListViewMsgItems.getCount() - 1);
        }
    }

最后实现效果如图:


2.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,006评论 25 707
  • 点击查看原文 Web SDK 开发手册 SDK 概述 网易云信 SDK 为 Web 应用提供一个完善的 IM 系统...
    layjoy阅读 13,745评论 0 15
  • 社交红利阅读笔记 书名:社交红利(修订升级版) 作者:徐志斌 出版社:中信出版社 正文前笔记: 推荐序1摘要 社交...
    凫水阅读 8,933评论 4 26
  • 这个学期起初的时候对一些东西有所了解过,就跳过很多东西吧,把一些自己认为重要的做出笔记,看了昨天的那东西,截图搞得...
    六六的建斌阅读 1,061评论 0 1
  • 大家特别是一些经常减肥的人看到这个标题的第一反应可能是不相信。请你耐心的往下看,看完你就会相信了。 药方:运动并通...
    轩昂妈阅读 339评论 0 2