Skip to content
This repository was archived by the owner on Nov 11, 2024. It is now read-only.

Commit 20cc38f

Browse files
Pengfei QuPengfei Qu
Pengfei Qu
authored and
Pengfei Qu
committedJun 10, 2021
add scenario encode to measure the quality with vmaf
1 parent 3d7a199 commit 20cc38f

File tree

6 files changed

+190
-6
lines changed

6 files changed

+190
-6
lines changed
 

‎CMakeLists.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ if (NOT DEFINED SCENARIO)
2222
set(SCENARIO "transcode")
2323
endif()
2424

25-
if (SCENARIO STREQUAL "transcode")
25+
if (NOT (SCENARIO STREQUAL "cdn"))
2626
set(NLIVES "0")
2727
endif()
2828

‎common/ffmpegcmd.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -190,13 +190,14 @@ def _dash(self):
190190
def _tc(self):
191191
cmd_1 = []
192192
params = self._tc_params
193+
stream_name = self._target.split("/")[-1].split(".")[0]
193194
for item in self._renditions:
194195
width = item[0]
195196
height = item[1]
196197
v_bitrate = self._to_kps(item[2])
197198
a_bitrate = self._to_kps(item[3])
198199
maxrate = self._to_kps(item[2] * self._max_bitrate_ratio)
199-
name= self._target+"/"+self._codec_type+"_"+str(height)+"p."+self._streaming_type if self._streaming_type == "mp4" else self._target+"_"+self._codec_type+str(height)+"p"
200+
name= self._target+"/"+stream_name+self._codec_type+"_"+str(height)+"p."+self._streaming_type if self._streaming_type == "mp4" else self._target+"_"+self._codec_type+str(height)+"p"
200201

201202
if self._acc_type == "vaapi":
202203
cmd_1 += ["-vf", "scale_vaapi=w="+str(width)+":"+"h="+str(height)+":format=nv12", "-c:v", self._codec]

‎content-provider/build.sh

+7
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ clips=(
2121
case "$(cat /proc/1/sched | head -n 1)" in
2222
*build.sh*)
2323
cd /mnt
24+
mkdir -p /mnt/raw
2425
for clip in "${clips[@]}"; do
2526
clip_name="$(echo $clip | cut -f5 -d/).mp4"
2627
if test ! -f "archive/$clip_name"; then
@@ -40,6 +41,12 @@ case "$(cat /proc/1/sched | head -n 1)" in
4041
ffmpeg -i "archive/$clip_name" -vf "thumbnail,scale=640:360" -frames:v 1 -y "archive/$clip_name".png
4142
fi
4243
done
44+
for clip in `find archive -name "*.mp4" -print`; do
45+
clip_name="${clip/*\//}"
46+
if test ! -f "raw/$clip_name".yuv; then
47+
ffmpeg -i "archive/$clip_name" -vcodec rawvideo -an -frames:v 600 -y "raw/$clip_name".yuv
48+
fi
49+
done
4350
wait
4451
;;
4552
*)

‎xcode-server/main.py

+54-4
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,12 @@
2525
DASH_ROOT = "/var/www/video/dash"
2626
HLS_ROOT = "/var/www/video/hls"
2727
MP4_ROOT = "/var/www/video/mp4"
28+
CACHE_RAW_YUV_ROOT = "/var/www/video/rawyuv"
29+
CACHE_DECODED_YUV_ROOT = "/var/www/video/decodedyuv"
2830

2931
HW_ACC_TYPE=os.getenv("HW_ACC_TYPE","sw")
3032
HW_DEVICE=os.getenv("HW_DEVICE",None)
33+
SCENARIO=os.getenv("SCENARIO","transcode")
3134

3235
fps_regex = re.compile(
3336
r"\s*frame=\s*(?P<frame_count>\d+)\s*fps=\s*(?P<fps>\d+\.?\d*).*"
@@ -54,7 +57,7 @@ def get_fps(next_line,start_time):
5457
return {"fps":round(fps,1), "speed":round(speed,3), "frames":frame_count, "start":round(start_time,3), "duration":round(now-start_time,3), "end":round(now,3), "status": "active"}
5558
return {}
5659

57-
def execute(idx, name, cmd):
60+
def execute(idx, name, cmd,kafka=True):
5861
p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, bufsize=1, universal_newlines=True)
5962
p.poll()
6063
start_time=time.time()
@@ -84,6 +87,46 @@ def execute(idx, name, cmd):
8487
print("Exception: {}".format(e))
8588
return p.returncode
8689

90+
def decode_yuv(yuv_path,in_stream_name,nframes=None):
91+
try:
92+
yuv_name=yuv_path+"/"+in_stream_name.split("/")[-1].replace(".mp4",".yuv")
93+
if not os.path.exists(path): makedirs(yuv_path)
94+
if os.path.exists(yuv_name): return
95+
if nframes:
96+
cmd = ["ffmpeg", "-hide_banner", "-i",in_stream_name, "-vcodec rawvideo", "-an","-frames:v", str(nframes),"-y",yuv_name]
97+
else:
98+
cmd = ["ffmpeg", "-hide_banner", "-i",in_stream_name, "-vcodec rawvideo", "-an","-y",yuv_name]
99+
print(cmd, flush=True)
100+
execute(5001,"decoded",cmd,kafka=False)
101+
except Exception as e:
102+
print("Exception: {}".format(e))
103+
pass
104+
return yuv_name
105+
106+
def measure_quality_vamf(idx,name, raw_mp4_path, target_mp4_path,width,height, nframes=100):
107+
vmaf_score=None
108+
model_path="/home/models/"
109+
try:
110+
if width >=1920 and height >=1080:
111+
model_name=model_path+"vmaf_4k_v0.6.1.json"
112+
else:
113+
model_name=model_path+"vmaf_v0.6.1.json"
114+
log_path=target_mp4_path+".json"
115+
framerate=24
116+
cmd = ["ffmpeg", "-r", str(framerate), "-i",raw_mp4_path, "-r", str(framerate), "-i", target_mp4_path, "-lavfi", "[0:v]trim=end_frame={},scale={}:{}:flags=bicubic,setpts=PTS-STARTPTS[reference];[1:v]trim=end_frame={},setpts=PTS-STARTPTS[distorted];[distorted][reference]libvmaf=log_fmt=json:log_path={}:model_path={}".format(nframes,width,height,nframes,log_path,model_name),"-f", "null", "-"]
117+
print(cmd,flush=True)
118+
execute(str(idx+1000),name,cmd,kafka=False)
119+
with open(log_path) as f:
120+
obj = json.load(f)
121+
sinfo={"id": str(idx+1000), "stream":name}
122+
vmaf_score=float(obj["pooled_metrics"]["vmaf"]["mean"])
123+
sinfo.update({"vmaf":vmaf_score})
124+
producer.send(KAFKA_WORKLOAD_TOPIC, json.dumps(sinfo))
125+
126+
except Exception as e:
127+
print("Exception: {}".format(e))
128+
return vmaf_score
129+
87130
def process_stream_vods(msg):
88131
stream_name=msg["name"]
89132
stream_type=msg["output"]["type"]
@@ -110,7 +153,8 @@ def process_stream_vods(msg):
110153

111154
if zk.process_start():
112155
try:
113-
cmd = FFMpegCmd(ARCHIVE_ROOT+"/"+stream_name, target_root+"/"+stream_name, stream_type, params=stream_parameters, acc_type=HW_ACC_TYPE, loop=loop, device=HW_DEVICE).cmd()
156+
input_stream=ARCHIVE_ROOT+"/"+stream_name
157+
cmd = FFMpegCmd(input_stream, target_root+"/"+stream_name, stream_type, params=stream_parameters, acc_type=HW_ACC_TYPE, loop=loop, device=HW_DEVICE).cmd()
114158
if cmd:
115159
print(cmd, flush=True)
116160
r = execute(idx, stream_name, cmd)
@@ -130,7 +174,7 @@ def process_stream_lives(msg):
130174
stream_type=msg["output"]["type"]
131175
target=msg["output"]["target"]
132176
loop= msg["loop"]
133-
idx=msg["idx"] if "idx" in msg.keys() else int(random.random()*10000)
177+
idx=int(msg["idx"]) if "idx" in msg.keys() else int(random.random()*10000)
134178
stream=stream_type+"/"+stream_name
135179

136180
if not isfile(ARCHIVE_ROOT+"/"+stream_name):
@@ -156,13 +200,19 @@ def process_stream_lives(msg):
156200

157201
if zk.process_start():
158202
try:
159-
cmd = FFMpegCmd(ARCHIVE_ROOT+"/"+stream_name, target_name, stream_type, params=stream_parameters, acc_type=HW_ACC_TYPE, loop=loop, device=HW_DEVICE).cmd()
203+
input_stream = ARCHIVE_ROOT+"/"+stream_name
204+
cmd = FFMpegCmd(input_stream, target_name, stream_type, params=stream_parameters, acc_type=HW_ACC_TYPE, loop=loop, device=HW_DEVICE).cmd()
160205

161206
if cmd:
162207
print(cmd, flush=True)
163208
r = execute(idx, stream_name, cmd)
164209
if r:
165210
raise Exception("status code: "+str(r))
211+
if SCENARIO == "encode":
212+
width=stream_parameters["renditions"][0][0]
213+
height=stream_parameters["renditions"][0][1]
214+
mp4_file=cmd[-1]
215+
measure_quality_vamf(idx,stream_name,raw_mp4_path=input_stream,target_mp4_path=mp4_file,width=width,height=height)
166216
zk.process_end()
167217
except:
168218
print(traceback.format_exc(), flush=True)

‎xcode-server/models/vmaf_4k_v0.6.1.json

+57
Large diffs are not rendered by default.

0 commit comments

Comments
 (0)