日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

SpringBoot + FFmpeg實現一個簡單的M3U8切片轉碼系統

瀏覽:195日期:2023-03-10 13:58:18
目錄想法實現工程pom配置文件TranscodeConfig,用于控制轉碼的一些參數MediaInfo,封裝視頻的一些基礎信息FFmpegUtils,工具類封裝FFmpeg的一些操作UploadController,執行轉碼操作index.html,客戶端使用想法

客戶端上傳視頻到服務器,服務器對視頻進行切片后,返回m3u8,封面等訪問路徑。可以在線的播放。 服務器可以對視頻做一些簡單的處理,例如裁剪,封面的截取時間。

視頻轉碼文件夾的定義

喜羊羊與灰太狼 // 文件夾名稱就是視頻標題 |-index.m3u8 // 主m3u8文件,里面可以配置多個碼率的播放地址 |-poster.jpg // 截取的封面圖片 |-ts // 切片目錄 |-index.m3u8 // 切片播放索引 |-key // 播放需要解密的AES KEY實現

需要先在本機安裝FFmpeg,并且添加到PATH環境變量,如果不會先通過搜索引擎找找資料

工程

SpringBoot + FFmpeg實現一個簡單的M3U8切片轉碼系統

pom

<project xmlns='http://maven.apache.org/POM/4.0.0'xmlns:xsi='http://www.w3.org/2001/XMLSchema-instance'xsi:schemaLocation='http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd'><modelVersion>4.0.0</modelVersion><groupId>com.demo</groupId><artifactId>demo</artifactId><version>0.0.1-SNAPSHOT</version><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.4.5</version><relativePath /> <!-- lookup parent from repository --></parent><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.junit.vintage</groupId><artifactId>junit-vintage-engine</artifactId><scope>test</scope></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-tomcat</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-undertow</artifactId></dependency><dependency><groupId>commons-codec</groupId><artifactId>commons-codec</artifactId></dependency><dependency><groupId>com.google.code.gson</groupId><artifactId>gson</artifactId></dependency></dependencies><build><finalName>${project.artifactId}</finalName><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><configuration><executable>true</executable></configuration></plugin></plugins></build></project>配置文件

server: port: 80app: # 存儲轉碼視頻的文件夾地址 video-folder: 'C:UsersAdministratorDesktoptmp'spring: servlet: multipart: enabled: true # 不限制文件大小 max-file-size: -1 # 不限制請求體大小 max-request-size: -1 # 臨時IO目錄 location: '${java.io.tmpdir}' # 不延遲解析 resolve-lazily: false # 超過1Mb,就IO到臨時目錄 file-size-threshold: 1MB web: resources: static-locations:- 'classpath:/static/'- 'file:${app.video-folder}' # 把視頻文件夾目錄,添加到靜態資源目錄列表TranscodeConfig,用于控制轉碼的一些參數

package com.demo.ffmpeg;public class TranscodeConfig {private String poster;// 截取封面的時間HH:mm:ss.[SSS]private String tsSeconds;// ts分片大小,單位是秒private String cutStart;// 視頻裁剪,開始時間HH:mm:ss.[SSS]private String cutEnd;// 視頻裁剪,結束時間HH:mm:ss.[SSS]public String getPoster() {return poster;}public void setPoster(String poster) {this.poster = poster;}public String getTsSeconds() {return tsSeconds;}public void setTsSeconds(String tsSeconds) {this.tsSeconds = tsSeconds;}public String getCutStart() {return cutStart;}public void setCutStart(String cutStart) {this.cutStart = cutStart;}public String getCutEnd() {return cutEnd;}public void setCutEnd(String cutEnd) {this.cutEnd = cutEnd;}@Overridepublic String toString() {return 'TranscodeConfig [poster=' + poster + ', tsSeconds=' + tsSeconds + ', cutStart=' + cutStart + ', cutEnd='+ cutEnd + ']';}}MediaInfo,封裝視頻的一些基礎信息

package com.demo.ffmpeg;import java.util.List;import com.google.gson.annotations.SerializedName;public class MediaInfo {public static class Format {@SerializedName('bit_rate')private String bitRate;public String getBitRate() {return bitRate;}public void setBitRate(String bitRate) {this.bitRate = bitRate;}}public static class Stream {@SerializedName('index')private int index;@SerializedName('codec_name')private String codecName;@SerializedName('codec_long_name')private String codecLongame;@SerializedName('profile')private String profile;}// ----------------------------------@SerializedName('streams')private List<Stream> streams;@SerializedName('format')private Format format;public List<Stream> getStreams() {return streams;}public void setStreams(List<Stream> streams) {this.streams = streams;}public Format getFormat() {return format;}public void setFormat(Format format) {this.format = format;}}FFmpegUtils,工具類封裝FFmpeg的一些操作

package com.demo.ffmpeg;import java.io.BufferedReader;import java.io.File;import java.io.IOException;import java.io.InputStreamReader;import java.nio.charset.StandardCharsets;import java.nio.file.Files;import java.nio.file.Path;import java.nio.file.Paths;import java.nio.file.StandardOpenOption;import java.security.NoSuchAlgorithmException;import java.util.ArrayList;import java.util.List;import javax.crypto.KeyGenerator;import org.apache.commons.codec.binary.Hex;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.util.StringUtils;import com.google.gson.Gson;public class FFmpegUtils {private static final Logger LOGGER = LoggerFactory.getLogger(FFmpegUtils.class);// 跨平臺換行符private static final String LINE_SEPARATOR = System.getProperty('line.separator');/** * 生成隨機16個字節的AESKEY * @return */private static byte[] genAesKey () {try {KeyGenerator keyGenerator = KeyGenerator.getInstance('AES');keyGenerator.init(128);return keyGenerator.generateKey().getEncoded();} catch (NoSuchAlgorithmException e) {return null;}}/** * 在指定的目錄下生成key_info, key文件,返回key_info文件 * @param folder * @throws IOException */private static Path genKeyInfo(String folder) throws IOException {// AES 密鑰byte[] aesKey = genAesKey();// AES 向量String iv = Hex.encodeHexString(genAesKey());// key 文件寫入Path keyFile = Paths.get(folder, 'key');Files.write(keyFile, aesKey, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);// key_info 文件寫入StringBuilder stringBuilder = new StringBuilder();stringBuilder.append('key').append(LINE_SEPARATOR);// m3u8加載key文件網絡路徑stringBuilder.append(keyFile.toString()).append(LINE_SEPARATOR);// FFmeg加載key_info文件路徑stringBuilder.append(iv);// ASE 向量Path keyInfo = Paths.get(folder, 'key_info');Files.write(keyInfo, stringBuilder.toString().getBytes(), StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);return keyInfo;}/** * 指定的目錄下生成 master index.m3u8 文件 * @param fileNamemaster m3u8文件地址 * @param indexPath訪問子index.m3u8的路徑 * @param bandWidth流碼率 * @throws IOException */private static void genIndex(String file, String indexPath, String bandWidth) throws IOException {StringBuilder stringBuilder = new StringBuilder();stringBuilder.append('#EXTM3U').append(LINE_SEPARATOR);stringBuilder.append('#EXT-X-STREAM-INF:BANDWIDTH=' + bandWidth).append(LINE_SEPARATOR); // 碼率stringBuilder.append(indexPath);Files.write(Paths.get(file), stringBuilder.toString().getBytes(StandardCharsets.UTF_8), StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING);}/** * 轉碼視頻為m3u8 * @param source源視頻 * @param destFolder目標文件夾 * @param config配置信息 * @throws IOException * @throws InterruptedException */public static void transcodeToM3u8(String source, String destFolder, TranscodeConfig config) throws IOException, InterruptedException {// 判斷源視頻是否存在if (!Files.exists(Paths.get(source))) {throw new IllegalArgumentException('文件不存在:' + source);}// 創建工作目錄Path workDir = Paths.get(destFolder, 'ts');Files.createDirectories(workDir);// 在工作目錄生成KeyInfo文件Path keyInfo = genKeyInfo(workDir.toString());// 構建命令List<String> commands = new ArrayList<>();commands.add('ffmpeg');commands.add('-i');commands.add(source);// 源文件commands.add('-c:v');commands.add('libx264');// 視頻編碼為H264commands.add('-c:a');commands.add('copy');// 音頻直接copycommands.add('-hls_key_info_file');commands.add(keyInfo.toString());// 指定密鑰文件路徑commands.add('-hls_time');commands.add(config.getTsSeconds());// ts切片大小commands.add('-hls_playlist_type');commands.add('vod');// 點播模式commands.add('-hls_segment_filename');commands.add('%06d.ts');// ts切片文件名稱if (StringUtils.hasText(config.getCutStart())) {commands.add('-ss');commands.add(config.getCutStart());// 開始時間}if (StringUtils.hasText(config.getCutEnd())) {commands.add('-to');commands.add(config.getCutEnd());// 結束時間}commands.add('index.m3u8');// 生成m3u8文件// 構建進程Process process = new ProcessBuilder().command(commands).directory(workDir.toFile()).start();// 讀取進程標準輸出new Thread(() -> {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {String line = null;while ((line = bufferedReader.readLine()) != null) {LOGGER.info(line);}} catch (IOException e) {}}).start();// 讀取進程異常輸出new Thread(() -> {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {String line = null;while ((line = bufferedReader.readLine()) != null) {LOGGER.info(line);}} catch (IOException e) {}}).start();// 阻塞直到任務結束if (process.waitFor() != 0) {throw new RuntimeException('視頻切片異常');}// 切出封面if (!screenShots(source, String.join(File.separator, destFolder, 'poster.jpg'), config.getPoster())) {throw new RuntimeException('封面截取異常');}// 獲取視頻信息MediaInfo mediaInfo = getMediaInfo(source);if (mediaInfo == null) {throw new RuntimeException('獲取媒體信息異常');}// 生成index.m3u8文件genIndex(String.join(File.separator, destFolder, 'index.m3u8'), 'ts/index.m3u8', mediaInfo.getFormat().getBitRate());// 刪除keyInfo文件Files.delete(keyInfo);}/** * 獲取視頻文件的媒體信息 * @param source * @return * @throws IOException * @throws InterruptedException */public static MediaInfo getMediaInfo(String source) throws IOException, InterruptedException {List<String> commands = new ArrayList<>();commands.add('ffprobe');commands.add('-i');commands.add(source);commands.add('-show_format');commands.add('-show_streams');commands.add('-print_format');commands.add('json');Process process = new ProcessBuilder(commands).start(); MediaInfo mediaInfo = null;try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {mediaInfo = new Gson().fromJson(bufferedReader, MediaInfo.class);} catch (IOException e) {e.printStackTrace();}if (process.waitFor() != 0) {return null;}return mediaInfo;}/** * 截取視頻的指定時間幀,生成圖片文件 * @param source源文件 * @param file圖片文件 * @param time截圖時間 HH:mm:ss.[SSS] * @throws IOException * @throws InterruptedException */public static boolean screenShots(String source, String file, String time) throws IOException, InterruptedException {List<String> commands = new ArrayList<>();commands.add('ffmpeg');commands.add('-i');commands.add(source);commands.add('-ss');commands.add(time);commands.add('-y');commands.add('-q:v');commands.add('1');commands.add('-frames:v');commands.add('1');commands.add('-f');;commands.add('image2');commands.add(file);Process process = new ProcessBuilder(commands).start();// 讀取進程標準輸出new Thread(() -> {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {String line = null;while ((line = bufferedReader.readLine()) != null) {LOGGER.info(line);}} catch (IOException e) {}}).start();// 讀取進程異常輸出new Thread(() -> {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {String line = null;while ((line = bufferedReader.readLine()) != null) {LOGGER.error(line);}} catch (IOException e) {}}).start();return process.waitFor() == 0;}}UploadController,執行轉碼操作

package com.demo.web.controller;import java.io.IOException;import java.nio.file.Files;import java.nio.file.Path;import java.nio.file.Paths;import java.time.LocalDate;import java.time.format.DateTimeFormatter;import java.util.HashMap;import java.util.Map;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.beans.factory.annotation.Value;import org.springframework.http.HttpStatus;import org.springframework.http.ResponseEntity;import org.springframework.web.bind.annotation.PostMapping;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestPart;import org.springframework.web.bind.annotation.RestController;import org.springframework.web.multipart.MultipartFile;import com.demo.ffmpeg.FFmpegUtils;import com.demo.ffmpeg.TranscodeConfig;@RestController@RequestMapping('/upload')public class UploadController {private static final Logger LOGGER = LoggerFactory.getLogger(UploadController.class);@Value('${app.video-folder}')private String videoFolder;private Path tempDir = Paths.get(System.getProperty('java.io.tmpdir'));/** * 上傳視頻進行切片處理,返回訪問路徑 * @param video * @param transcodeConfig * @return * @throws IOException */@PostMappingpublic Object upload (@RequestPart(name = 'file', required = true) MultipartFile video,@RequestPart(name = 'config', required = true) TranscodeConfig transcodeConfig) throws IOException {LOGGER.info('文件信息:title={}, size={}', video.getOriginalFilename(), video.getSize());LOGGER.info('轉碼配置:{}', transcodeConfig);// 原始文件名稱,也就是視頻的標題String title = video.getOriginalFilename();// io到臨時文件Path tempFile = tempDir.resolve(title);LOGGER.info('io到臨時文件:{}', tempFile.toString());try {video.transferTo(tempFile);// 刪除后綴title = title.substring(0, title.lastIndexOf('.'));// 按照日期生成子目錄String today = DateTimeFormatter.ofPattern('yyyyMMdd').format(LocalDate.now());// 嘗試創建視頻目錄Path targetFolder = Files.createDirectories(Paths.get(videoFolder, today, title));LOGGER.info('創建文件夾目錄:{}', targetFolder);Files.createDirectories(targetFolder);// 執行轉碼操作LOGGER.info('開始轉碼');try {FFmpegUtils.transcodeToM3u8(tempFile.toString(), targetFolder.toString(), transcodeConfig);} catch (Exception e) {LOGGER.error('轉碼異常:{}', e.getMessage());Map<String, Object> result = new HashMap<>();result.put('success', false);result.put('message', e.getMessage());return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(result);}// 封裝結果Map<String, Object> videoInfo = new HashMap<>();videoInfo.put('title', title);videoInfo.put('m3u8', String.join('/', '', today, title, 'index.m3u8'));videoInfo.put('poster', String.join('/', '', today, title, 'poster.jpg'));Map<String, Object> result = new HashMap<>();result.put('success', true);result.put('data', videoInfo);return result;} finally {// 始終刪除臨時文件Files.delete(tempFile);}}}index.html,客戶端

<html lang='en'> <head><meta charset='UTF-8'><title>Title</title><script src='https://cdn.jsdelivr.net/hls.js/latest/hls.min.js'></script> </head> <body>選擇轉碼文件: <input name='file' type='file' accept='video/*' onchange='upload(event)'><hr/><videocontrols='controls'></video> </body> <script> const video = document.getElementById(’video’); function upload (e){ let files = e.target.files if (!files) {return }// TODO 轉碼配置這里固定死了 var transCodeConfig = { poster: '00:00:00.001', // 截取第1毫秒作為封面 tsSeconds: 15, cutStart: '', cutEnd: '' }// 執行上傳 let formData = new FormData(); formData.append('file', files[0]) formData.append('config', new Blob([JSON.stringify(transCodeConfig)], {type: 'application/json; charset=utf-8'})) fetch(’/upload’, {method: ’POST’,body: formData }) .then(resp => resp.json()) .then(message => { if (message.success){ // 設置封面 video.poster = message.data.poster;// 渲染到播放器 var hls = new Hls(); hls.loadSource(message.data.m3u8); hls.attachMedia(video); } else { alert('轉碼異常,詳情查看控制臺'); console.log(message.message); } }) .catch(err => { alert('轉碼異常,詳情查看控制臺');throw err })} </script></html>使用 在配置文件中,配置到本地視頻目錄后啟動 打開頁面 localhost 點擊【選擇文件】,選擇一個視頻文件進行上傳,等待執行完畢(沒有做加載動畫) 后端轉碼完成后,會自動把視頻信息加載到播放器,此時可以手動點擊播放按鈕進行播放

可以打開控制臺,查看上傳進度,以及播放時的網絡加載信息

以上就是SpringBoot + FFmpeg實現一個簡單的M3U8切片轉碼系統的詳細內容,更多關于SpringBoot 實現M3U8切片轉碼的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
91久久国产| 久久激情av| 久久99国产精品视频| 欧美午夜精品一区二区三区电影| 亚洲18在线| 日韩中文在线电影| 国产福利资源一区| 午夜亚洲一区| 日韩久久精品| 欧美精品二区| 欧美国产另类| 精品国产一区二区三区2021| 日韩区欧美区| 国产剧情一区二区在线观看| 国产色噜噜噜91在线精品| 欧美黑人巨大videos精品| 欧美精品1区| 香蕉视频亚洲一级| 日韩亚洲国产欧美| 日本不卡高清| 成人在线视频免费| 亚洲欧洲一区二区天堂久久| 久久亚洲影院| 国产精品jk白丝蜜臀av小说| 成人一区而且| 亚洲专区欧美专区| 69精品国产久热在线观看| 欧美日韩1区| 久久精品123| 亚洲精品乱码久久久久久蜜桃麻豆| 日韩成人精品一区二区三区| 国内精品伊人| 国产精品av一区二区| 国产图片一区| 午夜欧美精品久久久久久久| 欧美在线首页| 欧美特黄一级大片| 久久xxx视频| 中文字幕日本一区二区| 久久久国产精品入口麻豆| 欧美在线亚洲综合一区| 国产亚洲精品美女久久 | 丝袜美腿诱惑一区二区三区| 亚洲综合日韩| 亚洲一区资源| 青草久久视频| 日本不卡视频一二三区| 日韩av网站在线免费观看| 狠狠色狠狠色综合日日tαg| 日韩精品麻豆| 久久精品亚洲人成影院| 丝袜av一区| 亚洲高清av| 亚洲性视频在线| 亚州精品视频| 视频一区中文字幕精品| 蜜桃视频在线观看一区| 蜜臀av国产精品久久久久| 日韩中文字幕不卡| 亚洲色图网站| 日本成人一区二区| 国产精品高清一区二区| 91视频久久| 欧美+日本+国产+在线a∨观看| 韩日一区二区三区| 天堂成人国产精品一区| 日韩一区二区三免费高清在线观看 | 国产精品丝袜在线播放| 久久国产欧美日韩精品| 九九99久久精品在免费线bt| 国产成年精品| 成人羞羞在线观看网站| 最新亚洲一区| 国产精品毛片aⅴ一区二区三区| 日韩一区二区三区免费| 午夜精品网站| 欧美日韩在线精品一区二区三区激情综合| 国产精品99精品一区二区三区∴| 999久久久91| 日韩精品三级| 97精品国产| 亚洲青青久久| 成人日韩在线观看| 欧美久久一区二区三区| sm捆绑调教国产免费网站在线观看| 激情欧美一区| 六月丁香综合在线视频| 视频在线在亚洲| 国产66精品| 四虎成人精品一区二区免费网站 | 国产精品99久久精品| 首页国产欧美日韩丝袜| 久久毛片亚洲| 久久激情中文| 欧美日韩一区二区三区在线电影| 亚洲毛片视频| 久久青草久久| 日韩国产一区二区| 欧美日韩国产v| 精品一区av| 国产高清精品二区| 久久久久97| 国产一区二区三区亚洲综合| 国产乱子精品一区二区在线观看| 亚洲日本免费电影| 日韩不卡一二三区| 国产日韩欧美一区二区三区 | 四虎国产精品免费观看| 国产精品色在线网站| 国产精品网在线观看| 国产精品白丝av嫩草影院| 久久精品国产久精国产| 福利精品在线| 99视频精品全国免费| 国产高清一区| 亚洲一二三区视频| 久久99久久久精品欧美| 欧美激情另类| 99riav国产精品| 精品91久久久久| 中文字幕在线看片| 国产精品v日韩精品v欧美精品网站 | 欧美日韩在线播放视频| 婷婷综合社区| 日韩av午夜在线观看| 国产极品嫩模在线观看91精品| 久久亚洲人体| 欧美日韩中文一区二区| 免费看精品久久片| 国产午夜一区| 日韩欧美不卡| 亚洲精品综合| 国产一二在线播放| 欧美sm一区| 亚洲1区在线| av在线资源| 中文字幕日韩高清在线 | 久久国产视频网| 欧美日韩在线观看首页| 视频在线观看一区二区三区| 国产精品一区二区中文字幕| 亚洲综合在线电影| 日韩精品一页| 久久久久久久久久久妇女| 日韩一区精品| 国产91一区| 国产精品啊啊啊| 国产一在线精品一区在线观看| 一区二区精品| 精品亚洲美女网站| 久久国内精品视频| 日韩网站在线| 伊人久久视频| 国产经典一区| 日本在线不卡视频一二三区| av资源亚洲| 麻豆精品蜜桃视频网站| 亚洲伊人精品酒店| 国产综合亚洲精品一区二| 久久精品国产网站| 欧美日韩 国产精品| 亚洲视频电影在线| 久久国产中文字幕| 成人国产精品久久| 久久精品国产精品亚洲毛片| 国产亚洲精品精品国产亚洲综合| 亚洲影视一区二区三区| 亚洲专区欧美专区| 国产综合亚洲精品一区二| 日韩a一区二区| 美女视频黄免费的久久| 麻豆中文一区二区| 卡一卡二国产精品| 久久不见久久见中文字幕免费| 国产欧美三级| 精品资源在线| 国产一区二区三区四区五区| 欧美一级一区| 国产精品任我爽爆在线播放| 免费在线亚洲欧美| 国产一区二区三区成人欧美日韩在线观看 | 国产精品片aa在线观看| 久久成人福利| 国产在线观看www| 国内亚洲精品| 日韩精品一二三区| 三级久久三级久久久| 亚洲18在线| 久久99久久久精品欧美| 国产精品调教视频| 美女国产精品久久久| 人在线成免费视频| 免费日韩av| 国产精品毛片久久久| 日韩av免费| 亚洲人www| 日韩高清欧美| 涩涩涩久久久成人精品| 国产一区二区三区探花| 日韩精品一卡二卡三卡四卡无卡|