def test_encoding_start_request_with_vp9_per_title_configuration(self): auto_representation = AutoRepresentation( adopt_configuration_threshold=1.5) vp9_configuration = VP9PerTitleConfiguration( auto_representations=auto_representation, min_bitrate_step_size=15000, max_bitrate_step_size=20000, min_bitrate=500000, max_bitrate=8000000, target_quality_crf=0.5) per_title = PerTitle(vp9_configuration=vp9_configuration) per_title_serialized = json.dumps(per_title, cls=BitmovinJSONEncoder) self.assertEqual( first=per_title_serialized, second='{"vp9Configuration": {"minBitrate": 500000, ' '"maxBitrate": 8000000, "minBitrateStepSize": 15000, ' '"maxBitrateStepSize": 20000, "targetQualityCrf": 0.5, ' '"autoRepresentations": {"adoptConfigurationThreshold": ' '1.5}}}') start_encoding_request = StartEncodingRequest(per_title=per_title) start_encoding_request_serialized = json.dumps( obj=start_encoding_request, cls=BitmovinJSONEncoder) self.assertEqual(first=start_encoding_request_serialized, second='{"perTitle": {"vp9Configuration": {' '"minBitrate": 500000, "maxBitrate": ' '8000000, "minBitrateStepSize": 15000, ' '"maxBitrateStepSize": 20000, ' '"targetQualityCrf": 0.5, ' '"autoRepresentations": {' '"adoptConfigurationThreshold": 1.5}}}}')
def start_encoding(encoding): """ The encoding will be started with the per title object and the auto representations set. If the auto representation is set, stream configurations will be automatically added to the Per-Title profile. In that case at least one PER_TITLE_TEMPLATE stream configuration must be available. All other configurations will be automatically chosen by the Per-Title algorithm. All relevant settings for streams and muxings will be taken from the closest PER_TITLE_TEMPLATE stream defined. The closest stream will be chosen based on the resolution specified in the codec configuration. :param encoding: The reference of the encoding :return: """ auto_representations = AutoRepresentation() h264_per_title_configuration = H264PerTitleConfiguration( auto_representations=auto_representations) per_title = PerTitle(h264_configuration=h264_per_title_configuration) start_encoding_request = StartEncodingRequest( per_title=per_title, encoding_mode=EncodingMode.THREE_PASS) bitmovin.encodings.Encoding.start( encoding_id=encoding.id, start_encoding_request=start_encoding_request) try: bitmovin.encodings.Encoding.wait_until_finished( encoding_id=encoding.id) except BitmovinError as bitmovin_error: print('Exception occurred while waiting for encoding to finish: {}'. format(bitmovin_error))
def test_encoding_start_request_serialization_with_tweaks(self): tweaks = Tweaks(audio_video_sync_mode=AudioVideoSyncMode.RESYNC_AT_START) tweaks_serialized = json.dumps(obj=tweaks, cls=BitmovinJSONEncoder) self.assertEqual(first=tweaks_serialized, second='{"audioVideoSyncMode": "RESYNC_AT_START"}') start_encoding_request = StartEncodingRequest(tweaks=tweaks) start_encoding_request_serialized = json.dumps(obj=start_encoding_request, cls=BitmovinJSONEncoder) self.assertEqual(first=start_encoding_request_serialized, second='{"tweaks": {"audioVideoSyncMode": "RESYNC_AT_START"}}')
def test_encoding_start_request_serialization_with_prewarmed_instance_ids(self): prewarmed_instance_pool_ids = ['4a67260e-2fd3-4e9e-9829-651280ea8f06', '4b67260e-2fd3-4e9e-9829-651280ea8f07'] scheduling = Scheduling(prewarmed_instance_pool_ids=prewarmed_instance_pool_ids) scheduling_serialized = json.dumps(obj=scheduling, cls=BitmovinJSONEncoder) self.assertEqual(first=scheduling_serialized, second='{"prewarmedInstancePoolIds": ["4a67260e-2fd3-4e9e-9829-651280ea8f06", "4b67260e-2fd3-4e9e-9829-651280ea8f07"]}') start_encoding_request = StartEncodingRequest(scheduling=scheduling) start_encoding_request_serialized = json.dumps(obj=start_encoding_request, cls=BitmovinJSONEncoder) self.assertEqual(first=start_encoding_request_serialized, second='{"scheduling": {"prewarmedInstancePoolIds": ["4a67260e-2fd3-4e9e-9829-651280ea8f06", "4b67260e-2fd3-4e9e-9829-651280ea8f07"]}}')
def test_encoding_start_request_with_h264_per_title_configuration(self): auto_representation = AutoRepresentation( adopt_configuration_threshold=1.5) h264_configuration = H264PerTitleConfiguration( auto_representations=auto_representation, min_bitrate_step_size=15000, max_bitrate_step_size=20000, min_bitrate=500000, max_bitrate=8000000, target_quality_crf=0.5, codec_min_bitrate_factor=1, codec_max_bitrate_factor=1, codec_bufsize_factor=2, complexity_factor=1.7) per_title = PerTitle(h264_configuration=h264_configuration) per_title_serialized = json.dumps(per_title, cls=BitmovinJSONEncoder) self.assertEqual( first=per_title_serialized, second='{"h264Configuration": {"minBitrate": 500000, ' '"maxBitrate": 8000000, "minBitrateStepSize": 15000, ' '"maxBitrateStepSize": 20000, "complexityFactor": 1.7, ' '"targetQualityCrf": 0.5, "codecMinBitrateFactor": 1, ' '"codecMaxBitrateFactor": 1, "codecBufsizeFactor": 2, ' '"autoRepresentations": ' '{"adoptConfigurationThreshold": 1.5}}}') start_encoding_request = StartEncodingRequest(per_title=per_title) start_encoding_request_serialized = json.dumps( obj=start_encoding_request, cls=BitmovinJSONEncoder) self.assertEqual( first=start_encoding_request_serialized, second='{"perTitle": {"h264Configuration": {"minBitrate": 500000, ' '"maxBitrate": 8000000, "minBitrateStepSize": 15000, ' '"maxBitrateStepSize": 20000, "complexityFactor": 1.7, ' '"targetQualityCrf": 0.5, "codecMinBitrateFactor": 1, ' '"codecMaxBitrateFactor": 1, "codecBufsizeFactor": 2, ' '"autoRepresentations": {"adoptConfigurationThreshold": 1.5}}}}')
def test_encoding_start_request_with_hls_vod_manifest(self): uuid_1 = 'b252e1d0-a252-4c7b-a99b-340c14d15bbb' uuid_2 = '90aba55c-b449-4a7e-b691-5607c7ac971b' vod_hls_manifest_1 = VodHlsStartManifest(manifest_id=uuid_1) vod_hls_manifest_2 = VodHlsStartManifest(manifest_id=uuid_2) vod_hls_manifests = [vod_hls_manifest_1, vod_hls_manifest_2] start_encoding_request = StartEncodingRequest( vod_hls_manifests=vod_hls_manifests) start_encoding_request_serialized = json.dumps( obj=start_encoding_request, cls=BitmovinJSONEncoder) expected_start_manifest_payload = '{"vodHlsManifests": [' + \ '{"manifestId": "b252e1d0-a252-4c7b-a99b-340c14d15bbb"}' + \ ', ' + \ '{"manifestId": "90aba55c-b449-4a7e-b691-5607c7ac971b"}' + \ ']}' self.assertEqual(first=start_encoding_request_serialized, second=expected_start_manifest_payload)
def test_encoding_start_request_with_dash_vod_manifest(self): uuid_1 = '731ec108-c5fb-4f31-ac42-823428e295f8' uuid_2 = 'a38ead99-e24a-4130-80fd-95502eb388f9' vod_dash_manifest_1 = VodDashStartManifest(manifest_id=uuid_1) vod_dash_manifest_2 = VodDashStartManifest(manifest_id=uuid_2) vod_dash_manifests = [vod_dash_manifest_1, vod_dash_manifest_2] start_encoding_request = StartEncodingRequest( vod_dash_manifests=vod_dash_manifests) start_encoding_request_serialized = json.dumps( obj=start_encoding_request, cls=BitmovinJSONEncoder) expected_start_manifest_payload = '{"vodDashManifests": [' + \ '{"manifestId": "731ec108-c5fb-4f31-ac42-823428e295f8"}' + \ ', ' + \ '{"manifestId": "a38ead99-e24a-4130-80fd-95502eb388f9"}' + \ ']}' self.assertEqual(first=start_encoding_request_serialized, second=expected_start_manifest_payload)
def test_encoding_start_request_with_dash_and_hls_vod_manifest(self): uuid_1 = '731ec108-c5fb-4f31-ac42-823428e295f8' uuid_2 = 'a38ead99-e24a-4130-80fd-95502eb388f9' uuid_3 = 'b252e1d0-a252-4c7b-a99b-340c14d15bbb' uuid_4 = '90aba55c-b449-4a7e-b691-5607c7ac971b' vod_dash_manifest_1 = VodDashStartManifest(manifest_id=uuid_1) vod_dash_manifest_2 = VodDashStartManifest(manifest_id=uuid_2) vod_dash_manifests = [vod_dash_manifest_1, vod_dash_manifest_2] vod_hls_manifest_1 = VodHlsStartManifest(manifest_id=uuid_3) vod_hls_manifest_2 = VodHlsStartManifest(manifest_id=uuid_4) vod_hls_manifests = [vod_hls_manifest_1, vod_hls_manifest_2] start_encoding_request = StartEncodingRequest( vod_dash_manifests=vod_dash_manifests, vod_hls_manifests=vod_hls_manifests) start_encoding_request_serialized = json.dumps( obj=start_encoding_request, cls=BitmovinJSONEncoder) expected_start_manifest_payload = '{"vodDashManifests": [' + \ '{"manifestId": "731ec108-c5fb-4f31-ac42-823428e295f8"}' + \ ', ' + \ '{"manifestId": "a38ead99-e24a-4130-80fd-95502eb388f9"}' + \ '], ' + \ '"vodHlsManifests": [' + \ '{"manifestId": "b252e1d0-a252-4c7b-a99b-340c14d15bbb"}' + \ ', ' + \ '{"manifestId": "90aba55c-b449-4a7e-b691-5607c7ac971b"}' + \ ']}' self.assertEqual(first=start_encoding_request_serialized, second=expected_start_manifest_payload)
def main(): s3_input = S3Input(access_key=S3_INPUT_ACCESS_KEY, secret_key=S3_INPUT_SECRET_KEY, bucket_name=S3_INPUT_BUCKET_NAME, name='Sample S3 Input') s3_input = bitmovin.inputs.S3.create(s3_input).resource s3_output = S3Output(access_key=S3_OUTPUT_ACCESSKEY, secret_key=S3_OUTPUT_SECRETKEY, bucket_name=S3_OUTPUT_BUCKETNAME, name='Sample S3 Output') s3_output = bitmovin.outputs.S3.create(s3_output).resource encoding = Encoding(name='Python Example - Per Title', cloud_region=CloudRegion.GOOGLE_EUROPE_WEST_1, encoder_version=EncoderVersion.BETA) encoding = bitmovin.encodings.Encoding.create(encoding).resource video_input_stream = StreamInput(input_id=s3_input.id, input_path=INPUT_PATH, selection_mode=SelectionMode.AUTO) audio_input_stream = StreamInput(input_id=s3_input.id, input_path=INPUT_PATH, selection_mode=SelectionMode.AUTO) audio_codec_configuration = AACCodecConfiguration( name='example_audio_codec_configuration_english', bitrate=128000, rate=48000) audio_codec_configuration = bitmovin.codecConfigurations.AAC.create( object_=audio_codec_configuration).resource audio_stream = Stream(codec_configuration_id=audio_codec_configuration.id, input_streams=[audio_input_stream], name='Sample Stream Audio') audio_stream = bitmovin.encodings.Stream.create( object_=audio_stream, encoding_id=encoding.id).resource video_codec_configuration = H264CodecConfiguration( name='Sample video codec configuration', profile=H264Profile.HIGH) video_codec_configuration = bitmovin.codecConfigurations.H264.create( video_codec_configuration).resource video_stream = Stream(codec_configuration_id=video_codec_configuration.id, input_streams=[video_input_stream], name='Sample Stream Video', mode=StreamMode.PER_TITLE_TEMPLATE) video_stream = bitmovin.encodings.Stream.create( object_=video_stream, encoding_id=encoding.id).resource audio_output_fmp4 = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'audio/fmp4') audio_muxing_stream = MuxingStream(audio_stream.id) audio_muxing_fmp4 = FMP4Muxing(streams=[audio_muxing_stream], segment_length=4.0, outputs=[audio_output_fmp4]) audio_muxing_fmp4 = bitmovin.encodings.Muxing.FMP4.create( object_=audio_muxing_fmp4, encoding_id=encoding.id).resource audio_output_ts = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'audio/ts') audio_muxing_ts = TSMuxing(streams=[audio_muxing_stream], segment_length=4.0, outputs=[audio_output_ts]) audio_muxing_ts = bitmovin.encodings.Muxing.TS.create( object_=audio_muxing_ts, encoding_id=encoding.id).resource audio_representation_info = dict( fmp4_muxing=audio_muxing_fmp4, ts_muxing=audio_muxing_ts, stream=audio_stream, ) video_output_fmp4 = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'video/fmp4/{uuid}') video_output_ts = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'video/ts/{uuid}') video_muxing_stream = MuxingStream(video_stream.id) video_muxing_fmp4 = FMP4Muxing(streams=[video_muxing_stream], segment_length=4.0, outputs=[video_output_fmp4]) video_muxing_fmp4 = bitmovin.encodings.Muxing.FMP4.create( object_=video_muxing_fmp4, encoding_id=encoding.id).resource video_muxing_ts = TSMuxing(streams=[video_muxing_stream], segment_length=4.0, outputs=[video_output_ts]) video_muxing_ts = bitmovin.encodings.Muxing.TS.create( object_=video_muxing_ts, encoding_id=encoding.id).resource auto_representations = AutoRepresentation() h264_per_title_configuration = H264PerTitleConfiguration( auto_representations=auto_representations) per_title = PerTitle(h264_configuration=h264_per_title_configuration) start_encoding_request = StartEncodingRequest( per_title=per_title, encoding_mode=EncodingMode.THREE_PASS) bitmovin.encodings.Encoding.start( encoding_id=encoding.id, start_encoding_request=start_encoding_request) try: bitmovin.encodings.Encoding.wait_until_finished( encoding_id=encoding.id) except BitmovinError as bitmovin_error: print('Exception occurred while waiting for encoding to finish: {}'. format(bitmovin_error)) create_dash_manifest(output_id=s3_output.id, encoding_id=encoding.id, audio_representation_info=audio_representation_info) create_hls_manifest(output_id=s3_output.id, encoding_id=encoding.id, audio_representation_info=audio_representation_info)
def main(): # Create an S3 input. This resource is then used as base to acquire input files. s3_input = S3Input(access_key=S3_INPUT_ACCESS_KEY, secret_key=S3_INPUT_SECRET_KEY, bucket_name=S3_INPUT_BUCKET_NAME, name='Test S3 Input') s3_input = bitmovin.inputs.S3.create(s3_input).resource # Create an S3 Output. This will be used as target bucket for the muxings, sprites and manifests s3_output = S3Output(access_key=S3_OUTPUT_ACCESSKEY, secret_key=S3_OUTPUT_SECRETKEY, bucket_name=S3_OUTPUT_BUCKETNAME, name='Test S3 Output') s3_output = bitmovin.outputs.S3.create(s3_output).resource acl_entry = ACLEntry(permission=ACLPermission.PUBLIC_READ) # Create DRM resources widevine_drm = CENCWidevineEntry(pssh=CENC_WIDEVINE_PSSH) playready_drm = CENCPlayReadyEntry(la_url=CENC_PLAYREADY_LA_URL) # Create an Encoding. This is the base entity used to configure the encoding. encoding = Encoding(name='Constrained Per-title encoding test', cloud_region=CloudRegion.AUTO, encoder_version=EncoderVersion.BETA) encoding = bitmovin.encodings.Encoding.create(encoding).resource encoding_configs = [] # Iterate over all encoding profiles and create the H264 configuration. # As we are using per-title, we do not define bitrates, instead just providing the target height as indicator for idx, _ in enumerate(encoding_profiles_h264): profile_h264 = encoding_profiles_h264[idx] encoding_config = dict(profile_h264=profile_h264) h264_codec = H264CodecConfiguration( name='Sample video codec configuration', profile=H264Profile.HIGH, height=profile_h264.get("height") ) encoding_config['h264_codec'] = bitmovin.codecConfigurations.H264.create(h264_codec).resource encoding_configs.append(encoding_config) # Also the AAC configuration has to be created, which will be later on used to create the streams. audio_codec_configuration = AACCodecConfiguration(name='AAC Codec Configuration', bitrate=128000, rate=48000) audio_codec_configuration = bitmovin.codecConfigurations.AAC.create(audio_codec_configuration).resource # create the input stream resources video_input_stream = StreamInput(input_id=s3_input.id, input_path=INPUT_PATH, selection_mode=SelectionMode.AUTO) audio_input_stream = StreamInput(input_id=s3_input.id, input_path=INPUT_PATH, selection_mode=SelectionMode.AUTO) # With the configurations and the input file, streams are now created that will be muxed later on. # As we use per-title, the streams are used as templates for encoding_config in encoding_configs: encoding_profile = encoding_config.get("profile_h264") video_stream_condition = Condition(attribute="HEIGHT", operator=">=", value=str(encoding_profile.get('height'))) video_stream = Stream(codec_configuration_id=encoding_config.get("h264_codec").id, input_streams=[video_input_stream], conditions=video_stream_condition, name='Stream H264 {}p'.format(encoding_profile.get('height')), mode=StreamMode.PER_TITLE_TEMPLATE) encoding_config['h264_stream'] = bitmovin.encodings.Stream.create(object_=video_stream, encoding_id=encoding.id).resource # create the stream for the audio audio_stream = Stream(codec_configuration_id=audio_codec_configuration.id, input_streams=[audio_input_stream], name='Audio Stream') audio_stream = bitmovin.encodings.Stream.create(object_=audio_stream, encoding_id=encoding.id).resource # === FMP4 Muxings === # Create FMP4 muxings which are later used for the DASH manifest. The current settings will set a segment length # of 4 seconds. for encoding_config in encoding_configs: encoding_profile = encoding_config.get("profile_h264") video_muxing_stream = MuxingStream(encoding_config['h264_stream'].id) video_muxing_output = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + "video/dash/{height}p_{bitrate}_{uuid}/", acl=[acl_entry]) video_muxing = FMP4Muxing(segment_length=4, segment_naming='seg_%number%.m4s', init_segment_name='init.mp4', streams=[video_muxing_stream], name="Video FMP4 Muxing {}p".format(encoding_profile.get('height'))) encoding_config['fmp4_muxing'] = bitmovin.encodings.Muxing.FMP4.create(object_=video_muxing, encoding_id=encoding.id).resource video_cenc = CENCDRM(key=CENC_KEY, kid=CENC_KID, widevine=widevine_drm, playReady=playready_drm, outputs=[video_muxing_output], name="Video FMP4 CENC") encoding_config['fmp4_cenc'] = bitmovin.encodings.Muxing.FMP4.DRM.CENC.create( object_=video_cenc, encoding_id=encoding.id, muxing_id=encoding_config['fmp4_muxing'].id).resource audio_muxing_stream = MuxingStream(audio_stream.id) audio_muxing_output = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'audio/dash/', acl=[acl_entry]) audio_fmp4_muxing = FMP4Muxing(segment_length=4, segment_naming='seg_%number%.m4s', init_segment_name='init.mp4', streams=[audio_muxing_stream], name='Audio FMP4 Muxing') audio_fmp4_muxing = bitmovin.encodings.Muxing.FMP4.create(object_=audio_fmp4_muxing, encoding_id=encoding.id).resource audio_cenc = CENCDRM(key=CENC_KEY, kid=CENC_KID, widevine=widevine_drm, playReady=playready_drm, outputs=[audio_muxing_output], name='Audio FMP4 CENC') audio_cenc = bitmovin.encodings.Muxing.FMP4.DRM.CENC.create(object_=audio_cenc, encoding_id=encoding.id, muxing_id=audio_fmp4_muxing.id).resource # === TS Muxings === # Create TS muxings which are later used for the HLS manifest. The current settings will set a segment length # of 4 seconds. for encoding_config in encoding_configs: encoding_profile = encoding_config.get('profile_h264') video_muxing_stream = MuxingStream(encoding_config['h264_stream'].id) video_muxing_output = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'video/hls/{height}p_{bitrate}_{uuid}/', acl=[acl_entry]) video_muxing = TSMuxing(segment_length=4, segment_naming='seg_%number%.ts', streams=[video_muxing_stream], name='Video TS Muxing {}p'.format(encoding_profile.get('height'))) encoding_config['ts_muxing'] = bitmovin.encodings.Muxing.TS.create(object_=video_muxing, encoding_id=encoding.id).resource video_fairplay = FairPlayDRM(key=FAIRPLAY_KEY, iv=FAIRPLAY_IV, uri=FAIRPLAY_URI, outputs=[video_muxing_output], name='Video TS FairPlay') encoding_config['ts_fairplay'] = bitmovin.encodings.Muxing.TS.DRM.FairPlay.create( object_=video_fairplay, encoding_id=encoding.id, muxing_id=encoding_config['ts_muxing'].id).resource audio_muxing_stream = MuxingStream(audio_stream.id) audio_muxing_output = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH + 'audio/hls/', acl=[acl_entry]) audio_ts_muxing = TSMuxing(segment_length=4, segment_naming='seg_%number%.ts', streams=[audio_muxing_stream], name='Audio TS Muxing') audio_ts_muxing = bitmovin.encodings.Muxing.TS.create(object_=audio_ts_muxing, encoding_id=encoding.id).resource audio_fairplay = FairPlayDRM(key=FAIRPLAY_KEY, iv=FAIRPLAY_IV, uri=FAIRPLAY_URI, outputs=[audio_muxing_output], name='Audio FairPlay') audio_fairplay = bitmovin.encodings.Muxing.TS.DRM.FairPlay.create(object_=audio_fairplay, encoding_id=encoding.id, muxing_id=audio_ts_muxing.id).resource # Keep the audio info together audio_representation_info = dict( fmp4_muxing=audio_fmp4_muxing, ts_muxing=audio_ts_muxing, stream=audio_stream, ts_fairplay=audio_fairplay, fmp4_cenc=audio_cenc ) # Finally create the per-title configuration to pass to the encoding auto_representations = AutoRepresentation(adopt_configuration_threshold=0.5) h264_per_title_configuration = H264PerTitleConfiguration(auto_representations=auto_representations) per_title = PerTitle(h264_configuration=h264_per_title_configuration) # And start the encoding start_encoding_request = StartEncodingRequest(per_title=per_title, encoding_mode=EncodingMode.THREE_PASS) bitmovin.encodings.Encoding.start(encoding_id=encoding.id, start_encoding_request=start_encoding_request) try: bitmovin.encodings.Encoding.wait_until_finished(encoding_id=encoding.id) except BitmovinError as bitmovin_error: print("Exception occurred while waiting for encoding to finish: {}".format(bitmovin_error)) # Specify the output for manifest which will be in the OUTPUT_BASE_PATH. manifest_output = EncodingOutput(output_id=s3_output.id, output_path=OUTPUT_BASE_PATH, acl=[acl_entry]) # === DASH MANIFEST === muxing_for_contentprotection = None drm_for_contentprotection = None # Create a DASH manifest and add one period with an adapation set for audio and video dash_manifest = DashManifest(manifest_name='stream.mpd', outputs=[manifest_output], name='DASH Manifest') dash_manifest = bitmovin.manifests.DASH.create(dash_manifest).resource period = Period() period = bitmovin.manifests.DASH.add_period(object_=period, manifest_id=dash_manifest.id).resource video_adaptation_set = VideoAdaptationSet() video_adaptation_set = bitmovin.manifests.DASH.add_video_adaptation_set(object_=video_adaptation_set, manifest_id=dash_manifest.id, period_id=period.id).resource audio_adaptation_set = AudioAdaptationSet(lang='en') audio_adaptation_set = bitmovin.manifests.DASH.add_audio_adaptation_set(object_=audio_adaptation_set, manifest_id=dash_manifest.id, period_id=period.id).resource # Add the audio representation segment_path = audio_representation_info.get('fmp4_cenc').outputs[0].outputPath segment_path = remove_output_base_path(segment_path) fmp4_representation_audio = DRMFMP4Representation(FMP4RepresentationType.TEMPLATE, encoding_id=encoding.id, muxing_id=audio_representation_info.get('fmp4_muxing').id, drm_id=audio_representation_info.get('fmp4_cenc').id, segment_path=segment_path) bitmovin.manifests.DASH.add_drm_fmp4_representation(object_=fmp4_representation_audio, manifest_id=dash_manifest.id, period_id=period.id, adaptationset_id=audio_adaptation_set.id) # Add all video representations to the video adaption set muxings = bitmovin.encodings.Muxing.FMP4.list(encoding_id=encoding.id).resource for muxing in muxings: drm = bitmovin.encodings.Muxing.FMP4.DRM.CENC.list(encoding.id, muxing.id).resource segment_path = drm[0].outputs[0].outputPath if 'audio' in segment_path: # we ignore the audio muxing continue if '{uuid}' in segment_path: # we ignore any muxing with placeholders in the path - they are the template muxings, not the result muxings continue segment_path = remove_output_base_path(segment_path) muxing_for_contentprotection = muxing drm_for_contentprotection = drm[0] fmp4_representation = DRMFMP4Representation( type=FMP4RepresentationType.TEMPLATE, encoding_id=encoding.id, muxing_id=muxing.id, segment_path=segment_path, drm_id=drm[0].id ) bitmovin.manifests.DASH.add_drm_fmp4_representation( object_=fmp4_representation, manifest_id=dash_manifest.id, period_id=period.id, adaptationset_id=video_adaptation_set.id) # add content protection to the adaptation set video_content_protection = ContentProtection(encoding_id=encoding.id, muxing_id=muxing_for_contentprotection.id, drm_id=drm_for_contentprotection.id) bitmovin.manifests.DASH.add_content_protection_to_adaptionset(object_=video_content_protection, manifest_id=dash_manifest.id, period_id=period.id, adaptationset_id=video_adaptation_set.id) bitmovin.manifests.DASH.start(manifest_id=dash_manifest.id) # === HLS MANIFEST === # Create a HLS manifest and add one period with an adapation set for audio and video hls_manifest = HlsManifest(manifest_name='stream.m3u8', outputs=[manifest_output], name='HLS Manifest') hls_manifest = bitmovin.manifests.HLS.create(hls_manifest).resource segment_path = audio_representation_info.get('ts_fairplay').outputs[0].outputPath segment_path = remove_output_base_path(segment_path) audio_media = AudioMedia(name='HLS Audio Media', group_id='audio', segment_path=segment_path, encoding_id=encoding.id, stream_id=audio_representation_info.get('stream').id, muxing_id=audio_representation_info.get('ts_muxing').id, drm_id=audio_representation_info.get('ts_fairplay').id, language='en', uri='audio.m3u8') audio_media = bitmovin.manifests.HLS.AudioMedia.create(manifest_id=hls_manifest.id, object_=audio_media).resource # Add all video representations to the video adaption set muxings = bitmovin.encodings.Muxing.TS.list(encoding_id=encoding.id).resource for muxing in muxings: drm = bitmovin.encodings.Muxing.TS.DRM.FairPlay.list(encoding.id, muxing.id).resource segment_path = drm[0].outputs[0].outputPath if 'audio' in segment_path: # we ignore the audio muxing continue if '{uuid}' in segment_path: # we ignore any muxing with placeholders in the path - they are the template muxings, not the result muxings continue segment_path = remove_output_base_path(segment_path) variant_stream = VariantStream(audio=audio_media.groupId, closed_captions='NONE', segment_path=segment_path, uri='video_{}.m3u8'.format(muxing.avgBitrate), encoding_id=encoding.id, stream_id=muxing.streams[0].streamId, muxing_id=muxing.id, drm_id=drm[0].id) bitmovin.manifests.HLS.VariantStream.create(manifest_id=hls_manifest.id, object_=variant_stream) bitmovin.manifests.HLS.start(manifest_id=hls_manifest.id) try: bitmovin.manifests.DASH.wait_until_finished(manifest_id=dash_manifest.id, check_interval=1) except BitmovinError as bitmovin_error: print('Exception occurred while waiting for manifest creation to finish: {}'.format(bitmovin_error)) try: bitmovin.manifests.HLS.wait_until_finished(manifest_id=hls_manifest.id, check_interval=1) except BitmovinError as bitmovin_error: print('Exception occurred while waiting for manifest creation to finish: {}'.format(bitmovin_error)) exit(-1)