| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466 | #include <stdio.h>#include <shlwapi.h>#include <strsafe.h>#include <fstream>#include <string>#include "M3ULoader.h"#include "../nu/ns_wc.h"#include "../WAT/WAT.h"M3ULoader::M3ULoader() : _utf8( false ){	wideTitle[ 0 ] = wideFilename[ 0 ] = 0;}M3ULoader::~M3ULoader( void ){	//Close();}struct cmpWchar_t {	bool operator()(const wchar_t* a, const wchar_t* b) const {		return wcscmp(a, b) < 0;	}};class M3UInfo : public ifc_plentryinfo{public:	M3UInfo()                                                         {}	M3UInfo( wchar_t *_mediahash, wchar_t *_metahash, wchar_t *_cloud_id, wchar_t *_cloud_status, wchar_t *_cloud_devices )	{		_extended_infos.emplace( _wcsdup( _INFO_NAME_MEDIA_HASH ),    _wcsdup( _mediahash) );		_extended_infos.emplace( _wcsdup( _INFO_NAME_META_HASH ),     _wcsdup( _metahash ) );		_extended_infos.emplace( _wcsdup( _INFO_NAME_CLOUD_ID ),      _wcsdup( _cloud_id ) );		_extended_infos.emplace( _wcsdup( _INFO_NAME_CLOUD_STATUS ),  _wcsdup( _cloud_status ) );		_extended_infos.emplace( _wcsdup( _INFO_NAME_CLOUD_DEVICES ), _wcsdup( _cloud_devices ) );	}	~M3UInfo()	{		for ( auto l_extended_infos_iterator = _extended_infos.begin(); l_extended_infos_iterator != _extended_infos.end(); ++l_extended_infos_iterator )		{			free( ( *l_extended_infos_iterator ).first );			free( ( *l_extended_infos_iterator ).second );		}		_extended_infos.clear();	}	void SetExtendedInfo( const wchar_t *p_parameter_name, const wchar_t *p_parameter_value )	{		_extended_infos.emplace( _wcsdup( p_parameter_name ), _wcsdup( p_parameter_value ) );	}	const wchar_t *GetExtendedInfo( wchar_t *parameter )	{		//for ( auto l_extended_infos_iterator = _extended_infos.begin(); l_extended_infos_iterator != _extended_infos.end(); ++l_extended_infos_iterator )		//{		//	wchar_t *l_key = _wcsdup( ( *l_extended_infos_iterator ).first );		//	if ( wcscmp( l_key, parameter ) == 0 )		//		return _wcsdup( ( *l_extended_infos_iterator ).second );		//}				// OLD		//std::map<wchar_t *, wchar_t *>::iterator l_extended_infos_iterator = _extended_infos.find( parameter );		//if ( l_extended_infos_iterator != _extended_infos.end() )		//	return _wcsdup( ( *l_extended_infos_iterator ).second );		auto it = _extended_infos.find(parameter);		if (_extended_infos.end() != it)		{			return it->second;		}		return 0;	}private:	RECVS_DISPATCH;	std::map<wchar_t *, wchar_t *, cmpWchar_t> _extended_infos;};#define CBCLASS M3UInfoSTART_DISPATCH;CB( IFC_PLENTRYINFO_GETEXTENDEDINFO, GetExtendedInfo )END_DISPATCH;#undef CBCLASSint M3ULoader::OnFileHelper( ifc_playlistloadercallback *playlist, const wchar_t *trackName, const wchar_t *title, int length, const wchar_t *rootPath, ifc_plentryinfo *extraInfo ){	if ( length == -1000 )		length = -1;	wcsncpy( wideFilename, trackName, FILENAME_SIZE );		int ret;	if ( wcsstr( wideFilename, L"://" ) || PathIsRootW( wideFilename ) )	{		ret = playlist->OnFile( wideFilename, title, length, extraInfo );	}	else	{		wchar_t fullPath[ MAX_PATH ] = { 0 };		if ( PathCombineW( fullPath, rootPath, wideFilename ) )		{			wchar_t canonicalizedPath[ MAX_PATH ] = { 0 };			PathCanonicalizeW( canonicalizedPath, fullPath );			ret = playlist->OnFile( canonicalizedPath, title, length, extraInfo );		}		else		{			ret = ifc_playlistloadercallback::LOAD_CONTINUE;		}	}	return ret;}static bool StringEnds( const wchar_t *a, const wchar_t *b ){	size_t aLen = wcslen( a );	size_t bLen = wcslen( b );	if ( aLen < bLen )		return false;  // too short	if ( !_wcsicmp( a + aLen - bLen, b ) )		return true;	return false;}int M3ULoader::Load( const wchar_t *p_filename, ifc_playlistloadercallback *playlist ){	// TODO: download temp file if it's a URL	// TODO - WDP2-198	FILE *fp = _wfopen( p_filename, L"rt,ccs=UNICODE" );	if ( !fp )		return IFC_PLAYLISTLOADER_FAILED;	fseek( fp, 0, SEEK_END );	int size = ftell( fp );	fseek( fp, 0, SEEK_SET );	if ( size == -1 )	{		fclose( fp );		fp = 0;		return IFC_PLAYLISTLOADER_FAILED;	}	if ( StringEnds( p_filename, L".m3u8" ) )		_utf8 = true;	int ext = 0;	wchar_t *p;	const int l_linebuf_size = 2048;	wchar_t linebuf[ l_linebuf_size ] = { 0 };	wchar_t ext_title[ MAX_PATH ]    = { 0 };	wchar_t ext_mediahash[ 128 ]     = { 0 };	wchar_t ext_metahash[ 128 ]      = { 0 };	wchar_t ext_cloud_id[ 128 ]      = { 0 };	wchar_t ext_cloud_status[ 16 ]   = { 0 };	wchar_t ext_cloud_devices[ 128 ] = { 0 };	int ext_len = -1;	wchar_t rootPath[ MAX_PATH ] = { 0 };	const wchar_t *callbackPath = playlist->GetBasePath();	if ( callbackPath )		StringCchCopyW( rootPath, MAX_PATH, callbackPath );	else	{		StringCchCopyW( rootPath, MAX_PATH, p_filename );		PathRemoveFileSpecW( rootPath );	}	unsigned char BOM[ 3 ] = { 0, 0, 0 };	if ( fread( BOM, 3, 1, fp ) == 1 && BOM[ 0 ] == 0xEF && BOM[ 1 ] == 0xBB && BOM[ 2 ] == 0xBF )		_utf8 = true;	else		fseek( fp, 0, SEEK_SET );	std::wstring l_separator     = L"\" ";	std::wstring l_key_separator = L"=";	const wchar_t _ASF[]                      = L"ASF ";	const wchar_t _DIRECTIVE_EXTINF[]         = L"#EXTINF:";	const wchar_t _DIRECTIVE_EXTM3U[]         = L"#EXTM3U";	const wchar_t _DIRECTIVE_EXT_X_NS_CLOUD[] = L"#EXT-X-NS-CLOUD:";	const wchar_t _DIRECTIVE_UTF8[]           = L"#UTF8";	const wchar_t _END_LINE[]                 = L"\r\n";	const int l_move_size = sizeof( wchar_t );	wa::strings::wa_string l_key_value_pair = "";	wa::strings::wa_string l_key            = "";	wa::strings::wa_string l_value          = "";	std::map<std::wstring, std::wstring> l_extended_infos;	while ( 1 )	{		if ( feof( fp ) )			break;		linebuf[ 0 ] = 0;		fgetws( linebuf, l_linebuf_size - 1, fp );		linebuf[ wcscspn( linebuf, _END_LINE ) ] = 0;		if ( wcslen( linebuf ) == 0 )			continue; 		if ( ext == 0 && wcsstr( linebuf, _DIRECTIVE_EXTM3U ) )		{			ext = 1;			continue;		}		if ( !wcsncmp( linebuf, _DIRECTIVE_UTF8, 5 ) )		{			_utf8 = true;			continue;		}		p = linebuf;		while ( p && *p == ' ' || *p == '\t' )			p = CharNextW( p );		if ( *p != '#' && *p != '\n' && *p != '\r' && *p )		{			wchar_t buf[ 4096 ] = { 0 };			wchar_t *p2 = CharPrevW( linebuf, linebuf + wcslen( linebuf ) ); //GetLastCharacter(linebuf);			if ( p2 && *p2 == '\n' )				*p2 = 0;			if ( !wcsncmp( p, _ASF, 4 ) && wcslen( p ) > 4 )				p += 4;			if ( wcsncmp( p, L"\\\\", 2 ) && wcsncmp( p + 1, L":\\", 2 ) && wcsncmp( p + 1, L":/", 2 ) && !wcsstr( p, L"://" ) )			{				if ( p[ 0 ] == '\\' )				{					buf[ 0 ] = rootPath[ 0 ];					buf[ 1 ] = rootPath[ 1 ];					StringCchCopyW( buf + 2, 4093, p );					//buf[ wcslen( buf ) - 1 ] = 0;					buf[ wcscspn( buf, _END_LINE ) ] = 0;					p = buf;				}			}			int ret;			// generate extra info from the cloud specific values (if present)			M3UInfo info( ext_mediahash, ext_metahash, ext_cloud_id, ext_cloud_status, ext_cloud_devices );			if ( !l_extended_infos.empty() )			{				for ( auto l_extended_infos_iterator = l_extended_infos.begin(); l_extended_infos_iterator != l_extended_infos.end(); ++l_extended_infos_iterator )				{					info.SetExtendedInfo( ( *l_extended_infos_iterator ).first.c_str(), ( *l_extended_infos_iterator ).second.c_str() );				}			}			l_extended_infos.clear();			if ( ext_title[ 0 ] )			{				wcsncpy( wideTitle, ext_title, FILETITLE_SIZE );				ret = OnFileHelper( playlist, p, wideTitle, ext_len * 1000, rootPath, &info );			}			else			{				ret = OnFileHelper( playlist, p, 0, -1, rootPath, &info );			}			if ( ret != ifc_playlistloadercallback::LOAD_CONTINUE )				break;			ext_len        = -1;			ext_title[ 0 ] = 0;		}		else		{			if ( ext && !wcsncmp( p, _DIRECTIVE_EXTINF, 8 ) )			{				p += 8;				ext_len = _wtoi( p );				int l_track_length = ext_len;				int l_digits = ( l_track_length < 0 ? 1 : 0 );				while ( l_track_length )				{					l_track_length /= 10;					++l_digits;				}				p += l_digits;				if ( p && *p )				{					wchar_t *p2 = CharPrevW( p, p + wcslen( p ) ); // GetLastCharacter(p);					if ( p2 && *p2 == '\n' )						*p2 = 0;					while ( p && *p == ' ' )						p = CharNextW( p );					std::wstring l_string( p );					int l_pos = l_string.find_first_of( L"," );					if ( l_pos > 0 )					{						int  l_key_separator_pos = 0;						wa::strings::wa_string l_line_trail( l_string.substr( 0, l_pos ) );						while ( !l_line_trail.empty() )						{							int l_separator_pos = l_line_trail.find( l_separator );							if ( l_separator_pos > 0 )								l_key_value_pair = l_line_trail.mid( 0, l_separator_pos + 1 );							else								l_key_value_pair = l_line_trail;							l_key_separator_pos = l_key_value_pair.find( l_key_separator );							l_key   = l_key_value_pair.mid( 0, l_key_separator_pos );							l_value = l_key_value_pair.mid( l_key_separator_pos + 1, l_key_value_pair.lengthS() - l_key_separator_pos + 1 );							l_value.replaceAll( "\"", "" );							l_extended_infos.emplace( l_key.GetW(), l_value.GetW() );							if ( l_separator_pos > 0 )								l_line_trail = l_line_trail.mid( l_separator_pos + l_move_size, l_line_trail.lengthS() - l_separator_pos + 1 );							else								l_line_trail.clear();						}						l_string = l_string.substr( l_pos + 1, l_string.size() - l_pos );						StringCchCopyW( ext_title, MAX_PATH, l_string.c_str() );					}					else						StringCchCopyW( ext_title, MAX_PATH, CharNextW( p ) );				}				else				{					ext_len        = -1;					ext_title[ 0 ] = 0;				}			}			// cloud specific playlist line for holding information about the entry			else if ( ext && !wcsncmp( p, _DIRECTIVE_EXT_X_NS_CLOUD, 16 ) )			{				p += 16;				wchar_t *pt = wcstok( p, L"," );				while ( pt != NULL )				{					int end = (int)wcscspn( pt, L"=" );					if ( !wcsncmp( pt, _INFO_NAME_MEDIA_HASH, end ) )					{						if ( ( lstrcpynW( ext_mediahash, pt + end + 1, 128 ) ) == NULL )							return IFC_PLAYLISTLOADER_FAILED;					}					else if ( !wcsncmp( pt, _INFO_NAME_META_HASH, end ) )					{						if ( ( lstrcpynW( ext_metahash, pt + end + 1, 128 ) ) == NULL )							return IFC_PLAYLISTLOADER_FAILED;					}					else if ( !wcsncmp( pt, _INFO_NAME_CLOUD_ID, end ) )					{						if ( ( lstrcpynW( ext_cloud_id, pt + end + 1, 128 ) ) == NULL )							return IFC_PLAYLISTLOADER_FAILED;					}					else if ( !wcsncmp( pt, _INFO_NAME_CLOUD_STATUS, end ) )					{						if ( ( lstrcpynW( ext_cloud_status, pt + end + 1, 16 ) ) == NULL )							return IFC_PLAYLISTLOADER_FAILED;					}					else if ( !wcsncmp( pt, _INFO_NAME_CLOUD_DEVICES, end ) )					{						wchar_t *p2 = pt + end + 1;						while ( p2 && *p2 != '\n' )							p2 = CharNextW( p2 );						if ( p2 && *p2 == '\n' )							*p2 = 0;						if ( ( lstrcpynW( ext_cloud_devices, pt + end + 1, 128 ) ) == NULL )							return IFC_PLAYLISTLOADER_FAILED;					}					pt = wcstok( NULL, L"," );				}			}			else			{				ext_len                = -1;				ext_title[ 0 ]         = 0;				ext_mediahash[ 0 ]     = 0;				ext_metahash[ 0 ]      = 0;				ext_cloud_id[ 0 ]      = 0;				ext_cloud_status[ 0 ]  = 0;				ext_cloud_devices[ 0 ] = 0;			}		}	}	if ( fp )		fclose( fp );	return IFC_PLAYLISTLOADER_SUCCESS;}#ifdef CBCLASS#undef CBCLASS#endif#define CBCLASS M3ULoaderSTART_DISPATCH;CB( IFC_PLAYLISTLOADER_LOAD, Load )#if 0VCB( IFC_PLAYLISTLOADER_CLOSE, Close )CB( IFC_PLAYLISTLOADER_GETITEM, GetItem )CB( IFC_PLAYLISTLOADER_GETITEMTITLE, GetItemTitle )CB( IFC_PLAYLISTLOADER_GETITEMLENGTHMILLISECONDS, GetItemLengthMilliseconds )CB( IFC_PLAYLISTLOADER_GETITEMEXTENDEDINFO, GetItemExtendedInfo )CB( IFC_PLAYLISTLOADER_NEXTITEM, NextItem )#endifEND_DISPATCH;
 |