一. 设计需求

使用阿里云视频点播服务对客户端上传的视频进行转码处理, 并存储到本地. 支持海外用户的大视频文件上传. 解决海外上传的有无问题.

二. 实现思路

客户端向服务端请求凭证, 获取上传地址, 使用阿里云sdk上传视频.

缺点:

目前美国,  澳大利亚没有节点, 这2地视频上传目前使用的是德国法兰克福的节点. 后台请求凭证接口缺少选择上传节点的模块.

后台处理需要时间, 刚刚完成上传的视频无法在中台页面播放.

三. 代码结构

1. 基础jar包. 导入点播Java SDK, 封装阿里云点播各个接口和上传所需的基础参数, 事件监听, 打成jar包, 供服务调用

2. 业务代码分层:  config, controller, service, dao, 事件消息处理, 工具类, 定时扫描作业.

四. 基础配置

1. 开通vod服务, mns服务, 并创建mns消息队列

2. 配置vod服务的转码模版组, 设置默认模板, 消息回调设置(这里使用mns)

3. 测试环境与生产环境

    默认环境配置可通过控制台或者OpenApi配置, 

    非默认环境需要调用阿里云的接口, 传appId进行配置. 部分通过OpenApi配置无效.

五. 上传相关代码

MNSListener: 监听器, 使用阿里提供的实现方案.

ClientConfiguration clientConf = new ClientConfiguration();
        clientConf.setMaxConnections(maxConnection);
        clientConf.setIoReactorThreadCount(ioCount);
        CloudAccount account = new CloudAccount(accessKeyId, accessKeySecret, endPoint,         clientConf);
        //this client need only initialize once
        mnsClient = account.getMNSClient();
        // 开启监听线程
        try {
            Thread thread = new Thread(() -> {
                CloudQueue cloudQueue = mnsClient.getQueueRef(queue);
                while (true) {
                    Message message = null;
                    try {
                        message = cloudQueue.popMessage(15);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                    try {
                        if (message == null) {
                            //PollingWaitSeconds expire
                            //And you could do some work here or do nothing according to your business
                        } else {
                            // 使用自定义的msgHandler处理消息
                            boolean res = mnsMessageHandler.handle(message);
                            if (res) {
                                if (delMsg) {
                                    cloudQueue.deleteMessage(message.getReceiptHandle());
                                }
                            }
                        }
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            });
            thread.start();
        } catch (ClientException ce) {

UploadService: 实现阿里的上传接口. 例如文件流上传接口:

public UploadStreamResponse uploadFile(VodUploadModule file,
                                           String extend,
                                           InputStream inputStream,
                                           String regionId,
                                           String securityToken) {
        UploadStreamRequest request = new UploadStreamRequest(accessKeyId, accessKeySecret, file.getTitle(),
                file.getFileName(), inputStream);
        request.setSecurityToken(securityToken);
        request.setApiRegionId(regionId);
        request.setCoverURL(file.getCover());
        //视频分类ID(可选)
        request.setCateId(file.getCateId());
        //视频标签,多个用逗号分隔(可选)
        request.setTitle(file.getTitle());
        if (request.getTemplateGroupId() != null) {
            request.setTemplateGroupId(file.getTemplateGroupId());
        }
        request.setTags(file.getTags());
        request.setDescription(file.getDescription());
        request.setAppId(file.getAppId());
        request.setUserData(VodRequestUserDataUtil.extendWrapper(extend));
        return videoUploadExecute(request);
    }

    private UploadStreamResponse videoUploadExecute(UploadStreamRequest request) {
        UploadVideoImpl uploader = new UploadVideoImpl();
        UploadStreamResponse response = uploader.uploadStream(request);
        if (response != null && !response.isSuccess()) {
            log.info("UploadVideoResponse ErrorCode: {} ErrorMessage: {} VideoId: {}", response.getCode(),
                    response.getMessage(), response.getVideoId());
        }
        return response;
    }

VodUploadModule: 图片/视频上传实体类: 

private String title;
    private String tags;
    private String url;
    private String description;
    private String appId;
    private Long cateId;
    // 视频必须带后缀
    private String fileName;
    private String cover;
    /**
     * default/cover
     */
    private String imageType;
    private String businessType;
    private String fileSize;
    private String templateGroupId;
     // imageExt materialExt
    private String fileExt;

MediaExtend: 业务所需实体类. 用户上传记录.

public class MediaExtend {
    private Long userId;
    private String userType;
    private String plt;
    private String videoId;
    private String imageType;
    private String region;
    // 当需要下载到本地时, 由上传者预生成视频地址作为转码完成后的访问地址
    private String localVideoPath;
    private String localCoverPath;
    private Date createTime;
    private String extend;
}

用户请求凭证controller: 

public ServiceResponse getMaterialUploadAddress(@RequestBody FileUploadRequest info) {
        String message = null;
        try {
            MediaExtendUtil.getCurrentUserInfo(info);
            MediaExtend extend = info.getMediaExtend();
            if (!extend.validateMediaExtend()) {
                message = "Extend 不能为空";
            }
            CreateUploadAttachedMediaResponse response = uploadService.createUploadAttachedMedia(info, extend);
            return apiResponse(ResponseCode.SC_OK, message, response);
        } catch (Exception e) {
            log.error("ErrorMessage = " + e.getMessage());
            return apiResponse(ResponseCode.SC_INTERNAL_SERVER_ERROR, e.getMessage());
        }
    }

凭证service:

public CreateUploadAttachedMediaResponse createUploadAttachedMedia(FileUploadRequest info, MediaExtend extend) throws Exception {
        CreateUploadAttachedMediaRequest request = new CreateUploadAttachedMediaRequest();
        request.setAppId(appId);
        BeanUtils.copyProperties(info, request);
        request.setMediaExt(info.getFileExt());
        DefaultAcsClient client = aliConfigService.getAcsClient(regionId);
        // extend包含了视频上传信息, 可在视频信息和转码事件中获取
request.setUserData(VodRequestUserDataUtil.extendWrapper(JSONObject.toJSONString(extend)));
        return (CreateUploadAttachedMediaResponse)client.getAcsResponse(request)        
    }

中台或App在获取到凭证后, 使用阿里云SDK上传视频. 

分享生成阿里云配置链接的代码(已删除敏感信息):

public class AliSignatureUtil {

    /*对所有参数名称和参数值做URL编码*/
    public static List<String> getAllParams(Map<String, String> publicParams, Map<String, String> privateParams) {
        List<String> encodeParams = new ArrayList<String>();
        if (publicParams != null) {
            for (Map.Entry<String, String> entry : publicParams.entrySet()) {
                //将参数和值都urlEncode一下。
                String encodeKey = percentEncode(entry.getKey());
                String encodeVal = percentEncode(entry.getValue());
                encodeParams.add(encodeKey + "=" + encodeVal);
            }
        }
        if (privateParams != null) {
            for (String key : privateParams.keySet()) {
                String value = privateParams.get(key);
                //将参数和值都urlEncode一下。
                String encodeKey = percentEncode(key);
                String encodeVal = percentEncode(value);
                encodeParams.add(encodeKey + "=" + encodeVal);
            }
        }
        return encodeParams;
    }

    /*获取 CanonicalizedQueryString*/
    public static String getCQS(List<String> allParams) {
        ParamsComparator paramsComparator = new ParamsComparator();
        Collections.sort(allParams, paramsComparator);
        StringBuffer cqString = new StringBuffer("");
        for (int i = 0; i < allParams.size(); i++) {
            cqString.append(allParams.get(i));
            if (i != allParams.size() - 1) {
                cqString.append("&");
            }
        }
        return cqString.toString();
    }

    /*字符串参数比较器,按字母序升序*/
    public static class ParamsComparator implements Comparator<String> {
        @Override
        public int compare(String lhs, String rhs) {
            return lhs.compareTo(rhs);
        }
    }

    /*构造待签名的字符串*/
    //String StringToSign = httpMethod + "&" + percentEncode("/") + "&" + percentEncode(CanonicalizedQueryString);
    /*特殊字符替换为转义字符*/
    public static String percentEncode(String value) {
        try {
            String urlEncodeOrignStr = URLEncoder.encode(value, "UTF-8");
            String plusReplaced = urlEncodeOrignStr.replace("+", "%20");
            String starReplaced = plusReplaced.replace("*", "%2A");
            String waveReplaced = starReplaced.replace("%7E", "~");
            return waveReplaced;
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        return value;
    }

    public static byte[] hmacSHA1Signature(String accessKeySecret, String stringToSign) {
        try {
            String key = accessKeySecret + "&";
            try {
                SecretKeySpec signKey = new SecretKeySpec(key.getBytes(), "HmacSHA1");
                Mac mac = Mac.getInstance("HmacSHA1");
                mac.init(signKey);
                return mac.doFinal(stringToSign.getBytes());
            } catch (Exception e) {
                throw new SignatureException("Failed to generate HMAC : " + e.getMessage());
            }
        } catch (SignatureException e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String newStringByBase64(byte[] bytes)
            throws UnsupportedEncodingException {
        if (bytes == null || bytes.length == 0) {
            return null;
        }
        return new BASE64Encoder().encode(bytes);
    }

    /*生成当前UTC时间戳Time*/
    public static String generateTimestamp() {
        Date date = new Date(System.currentTimeMillis());
        SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'");
        df.setTimeZone(new SimpleTimeZone(0, "GMT"));
        return df.format(date);
    }

    public static String generateRandom() {
        String signatureNonce = UUID.randomUUID().toString();
        return signatureNonce;
    }

   /* public static void main(String[] args) {
        Map<String, String> publicParams = new HashMap<>();
        Map<String, String> privateParams = new HashMap<>();

        publicParams.put("Format", "json");
        publicParams.put("Version", "2017-03-21");
        publicParams.put("AccessKeyId", "");
        publicParams.put("SignatureMethod", "HMAC-SHA1");
        publicParams.put("Timestamp", generateTimestamp());
        publicParams.put("SignatureVersion", "1.0");
        publicParams.put("SignatureNonce", generateRandom());

        privateParams.put("Action", "SetMessageCallback");
        privateParams.put("CallbackType", "MNS");
        privateParams.put("MnsEndpoint", "https://111111.mns.cn-shanghai.aliyuncs.com/");
        privateParams.put("MnsQueueName", "name");
        //privateParams.put("EventTypeList", "ALL");
        privateParams.put("EventTypeList", "FileUploadComplete,ImageUploadComplete,TranscodeComplete,SnapshotComplete,UploadByURLComplete,CreateAuditComplete");
        privateParams.put("AppId", "app-1000002");

        List<String> allParams = getAllParams(publicParams, privateParams);
        String cqs = getCQS(allParams);
        System.out.println(cqs);
        String httpMethod = "GET";
        String stringToSign = httpMethod + "&" + percentEncode("/") + "&" + percentEncode(cqs);

        String signature = null;
        String domain = "http://vod.eu-central-1.aliyuncs.com/";
        try {
            signature = newStringByBase64(hmacSHA1Signature("", stringToSign));
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        String url = domain + "?" + cqs + "&Signature=" + signature;
        System.out.println(url);
    }*/


    /*public static void main(String[] args) {
        Map<String, String> publicParams = new HashMap<>();
        Map<String, String> privateParams = new HashMap<>();

        publicParams.put("Format", "json");
        publicParams.put("Version", "2017-03-21");
        publicParams.put("AccessKeyId", "");
        publicParams.put("SignatureMethod", "HMAC-SHA1");
        publicParams.put("Timestamp", generateTimestamp());
        publicParams.put("SignatureVersion", "1.0");
        publicParams.put("SignatureNonce", generateRandom());

        privateParams.put("Action", "SetMessageCallback");
        privateParams.put("CallbackType", "MNS");
        privateParams.put("MnsEndpoint", "https://111111.mns.cn-shanghai.aliyuncs.com/");
        privateParams.put("MnsQueueName", "name");
        //privateParams.put("EventTypeList", "ALL");
        privateParams.put("EventTypeList", "FileUploadComplete,ImageUploadComplete,TranscodeComplete,SnapshotComplete,UploadByURLComplete,CreateAuditComplete");



        privateParams.put("AppId", "app-1000002");

        List<String> allParams = getAllParams(publicParams, privateParams);
        String cqs = getCQS(allParams);
        System.out.println(cqs);
        String httpMethod = "GET";
        String stringToSign = httpMethod + "&" + percentEncode("/") + "&" + percentEncode(cqs);

        String signature = null;
        String domain = "http://vod.eu-central-1.aliyuncs.com/";
        try {
            signature = newStringByBase64(hmacSHA1Signature("", stringToSign));
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        String url = domain + "?" + cqs + "&Signature=" + signature;
        System.out.println(url);
    }*/


    /**
     * 添加转码模板组配置
     */
    public static AddTranscodeTemplateGroupResponse addTranscodeTemplateGroup(DefaultAcsClient client) throws Exception {
        AddTranscodeTemplateGroupRequest request = new AddTranscodeTemplateGroupRequest();
        //转码模板ID
        request.setName("video-transcode-LD-1");
        request.setAppId("app-1000002");
        System.out.println(buildTranscodeTemplateList().toJSONString());
        request.setTranscodeTemplateList(buildTranscodeTemplateList().toJSONString());
        return client.getAcsResponse(request);
    }

    /**
     * 构建需要添加的转码模板配置数据
     *
     * @return
     */
    public static JSONArray buildTranscodeTemplateList() {
        JSONObject transcodeTemplate = new JSONObject();
        //模板名称
        transcodeTemplate.put("TemplateName", "video-transcode-LD-1");
        //清晰度
        transcodeTemplate.put("Definition", "SD");
        //视频流转码配置
        JSONObject video = new JSONObject();
        // 文档地址https://help.aliyun.com/document_detail/52839.html?spm=a2c4g.11186623.2.16.6bb96d22lCXeg4#h2--transcodetemplategroup-div-id-transcodetemplategroup-div-36
        // 转码视频流配置Video
        //分辨率(宽x高)
        video.put("Width", 960);
        //码率(Kbps)
        video.put("Bitrate", 900);
        //帧率(fps)
        video.put("Fps", 25);
        video.put("Remove", false);
        //编码格式
        video.put("Codec", "H.264");
        // 关键帧最大间隔(帧)
        video.put("Gop", "250");
        transcodeTemplate.put("Video", video);
        //音频流转码配置
        JSONObject audio = new JSONObject();
        // 编码格式
        audio.put("Codec", "AAC");
        // 码率(Kbps)
        audio.put("Bitrate", "96");
        // 声道数
        audio.put("Channels", "2");
        // 采样率
        audio.put("Samplerate", "44100");
        transcodeTemplate.put("Audio", audio);
        // 封装容器
        JSONObject container = new JSONObject();
        // 封装格式
        container.put("Format", "mp4");
        transcodeTemplate.put("Container", container);
        //条件转码配置
        JSONObject transconfig = new JSONObject();
        transconfig.put("IsCheckReso", false);
        transconfig.put("IsCheckResoFail", false);
        transconfig.put("IsCheckVideoBitrate", false);
        transconfig.put("IsCheckVideoBitrateFail", false);
        transconfig.put("IsCheckAudioBitrate", false);
        transconfig.put("IsCheckAudioBitrateFail", false);
        transcodeTemplate.put("TransConfig", transconfig);
        //加密配置(只支持m3u8)
        //JSONObject encryptSetting = new JSONObject();
        //encryptSetting.put("EncryptType", "Private");
        //transcodeTemplate.put("EncryptSetting", encryptSetting);
        //水印ID(多水印关联)
       /* JSONArray watermarkIdList = new JSONArray();
        watermarkIdList.add("263261bdc1ff65782f8995c6dd22a16a");
        //USER_DEFAULT_WATERMARK 代表默认水印ID
        watermarkIdList.add("USER_DEFAULT_WATERMARK");
        transcodeTemplate.put("WatermarkIds", watermarkIdList);*/
        JSONArray transcodeTemplateList = new JSONArray();
        transcodeTemplateList.add(transcodeTemplate);
        return transcodeTemplateList;
    }
    /*  */

    /**
     * 以下为调用示例
     */
  /* public static void main(String[] args) throws ClientException {
        DefaultAcsClient client = initVodClient("", "");
        AddTranscodeTemplateGroupResponse response = new AddTranscodeTemplateGroupResponse();
        try {
            response = addTranscodeTemplateGroup(client);
            System.out.println("TranscodeTemplateGroupId = " + response.getTranscodeTemplateGroupId());
        } catch (Exception e) {
            System.out.println("ErrorMessage = " + e.getLocalizedMessage());
        }
        System.out.println("RequestId = " + response.getRequestId());
    }*/

    public static DefaultAcsClient initVodClient(String accessKeyId, String accessKeySecret) throws ClientException {
       /* String regionId = "cn-shanghai";  // 点播服务接入区域
        
        String regionId2 = "ap-southeast-1";
      
        String regionId3 = "ap-southeast-5";
        
        String regionId4 = "ap-south-1";*/
        
        String regionId5 = "eu-central-1";
        
        //String regionId6 = "ap-northeast-1";
        
        DefaultProfile profile = DefaultProfile.getProfile(regionId5, accessKeyId, accessKeySecret);
        DefaultAcsClient client = new DefaultAcsClient(profile);
        return client;
    }

    public static ListTranscodeTemplateGroupResponse listTranscodeTemplateGroup(DefaultAcsClient client) throws Exception {
        ListTranscodeTemplateGroupRequest request = new ListTranscodeTemplateGroupRequest();
        request.setAppId("app-1000002");
        return client.getAcsResponse(request);
    }
    /**
     * 以下为调用示例
     */
    /*public static void main(String[] args) throws ClientException {
        DefaultAcsClient client = initVodClient("", "");
        ListTranscodeTemplateGroupResponse response = new ListTranscodeTemplateGroupResponse();
        try {
            response = listTranscodeTemplateGroup(client);
            System.out.println(JSONObject.toJSONString(response.getTranscodeTemplateGroupList()));
        } catch (Exception e) {
            System.out.println("ErrorMessage = " + e.getLocalizedMessage());
        }
        System.out.println("RequestId = " + response.getRequestId());
    }*/

    /**
     * 设置默认转码模板组
     */
    public static SetDefaultTranscodeTemplateGroupResponse setDefaultTranscodeTemplateGroup(DefaultAcsClient client) throws Exception {
        SetDefaultTranscodeTemplateGroupRequest request = new SetDefaultTranscodeTemplateGroupRequest();
        request.setTranscodeTemplateGroupId("111111111111111111");
        return client.getAcsResponse(request);
    }
    /**
     * 以下为调用示例
     */
    /*public static void main(String[] args) throws ClientException {
        DefaultAcsClient client = initVodClient("", "");
        SetDefaultTranscodeTemplateGroupResponse response = new SetDefaultTranscodeTemplateGroupResponse();
        try {
            response = setDefaultTranscodeTemplateGroup(client);
        } catch (Exception e) {
            System.out.println("ErrorMessage = " + e.getLocalizedMessage());
        }
        System.out.println("RequestId = " + response.getRequestId());
    }*/
}