[vlc-commits] [Git][videolan/vlc][master] 6 commits: art: factor cache file path helpers

Steve Lhomme (@robUx4) gitlab at videolan.org
Wed May 13 08:10:25 UTC 2026



Steve Lhomme pushed to branch master at VideoLAN / VLC


Commits:
69e196b2 by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: factor cache file path helpers

- - - - -
4ce0185e by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: read and write cached art URLs with helpers

- - - - -
17b47e55 by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: harden album art save failures

- - - - -
836e6217 by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: make attachment cache keys item-specific

- - - - -
ea171b49 by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: store embedded artwork blobs by content hash

- - - - -
297b8f9e by Niklas Burtscher at 2026-05-13T07:52:44+00:00
art: probe item-scoped attachment cache first

- - - - -


1 changed file:

- src/preparser/art.c


Changes:

=====================================
src/preparser/art.c
=====================================
@@ -38,14 +38,74 @@
 
 #include "art.h"
 
-static void ArtCacheCreateDir( char *psz_dir )
+static void ArtCacheCreateDir( const char *psz_dir )
 {
     vlc_mkdir_parent(psz_dir, 0700);
 }
 
+static bool ArtUrlIsAttachment( const char *psz_arturl )
+{
+    return psz_arturl != NULL && !strncmp( psz_arturl, "attachment://", 13 );
+}
+
+static char *ArtCacheGetAttachmentDirPath( const char *psz_title,
+                                           const char *psz_uri )
+{
+    char *psz_cachedir = config_GetUserDir( VLC_CACHE_DIR );
+    char *psz_dir = NULL;
+
+    if( unlikely( psz_cachedir == NULL ) )
+        return NULL;
+
+    if( EMPTY_STR( psz_uri ) && EMPTY_STR( psz_title ) )
+    {
+        free( psz_cachedir );
+        return NULL;
+    }
+
+    char psz_hash[VLC_HASH_MD5_DIGEST_HEX_SIZE];
+    vlc_hash_md5_t md5;
+    vlc_hash_md5_Init( &md5 );
+    vlc_hash_md5_Update( &md5, "attachment://", 13 );
+
+    if( !EMPTY_STR( psz_uri ) )
+        vlc_hash_md5_Update( &md5, psz_uri, strlen( psz_uri ) );
+    else
+        vlc_hash_md5_Update( &md5, psz_title, strlen( psz_title ) );
+
+    vlc_hash_FinishHex( &md5, psz_hash );
+
+    if( asprintf( &psz_dir, "%s" DIR_SEP "art" DIR_SEP "arturl" DIR_SEP
+                  "attachment-%s", psz_cachedir, psz_hash ) == -1 )
+        psz_dir = NULL;
+
+    free( psz_cachedir );
+    return psz_dir;
+}
+
+static char *ArtCacheAttachmentPath( input_item_t *p_item )
+{
+    char *psz_path = NULL;
+    const char *psz_title = NULL;
+    const char *psz_uri = NULL;
+
+    vlc_mutex_lock( &p_item->lock );
+
+    if( p_item->p_meta )
+        psz_title = vlc_meta_Get( p_item->p_meta, vlc_meta_Title );
+    psz_uri = p_item->psz_uri;
+    if( !psz_title )
+        psz_title = p_item->psz_name;
+
+    psz_path = ArtCacheGetAttachmentDirPath( psz_title, psz_uri );
+
+    vlc_mutex_unlock( &p_item->lock );
+    return psz_path;
+}
+
 static char* ArtCacheGetDirPath( const char *psz_arturl, const char *psz_artist,
                                  const char *psz_album,  const char *psz_date,
-                                 const char *psz_title )
+                                 const char *psz_title, const char *psz_uri )
 {
     char *psz_dir;
     char *psz_cachedir = config_GetUserDir(VLC_CACHE_DIR);
@@ -53,7 +113,8 @@ static char* ArtCacheGetDirPath( const char *psz_arturl, const char *psz_artist,
     if (unlikely(psz_cachedir == NULL))
         return NULL;
 
-    if( !EMPTY_STR(psz_artist) && !EMPTY_STR(psz_album) )
+    if( !ArtUrlIsAttachment( psz_arturl ) &&
+        !EMPTY_STR(psz_artist) && !EMPTY_STR(psz_album) )
     {
         char *psz_album_sanitized = strdup( psz_album );
         if (!psz_album_sanitized)
@@ -89,19 +150,24 @@ static char* ArtCacheGetDirPath( const char *psz_arturl, const char *psz_artist,
     else
     {
         /* If artist or album are missing, cache by art download URL.
-         * If the URL is an attachment://, add the title to the cache name.
-         * It will be md5 hashed to form a valid cache filename.
-         * We assume that psz_arturl is always the download URL and not the
-         * already hashed filename.
-         * (We should never need to call this function if art has already been
-         * downloaded anyway).
+         *
+         * Embedded artwork ("attachment://") must stay item-specific even when
+         * metadata matches across different tracks, otherwise a previous track
+         * can populate a shared cache entry and be reused later.
          */
         char psz_arturl_sanitized[VLC_HASH_MD5_DIGEST_HEX_SIZE];
 
         vlc_hash_md5_t md5;
         vlc_hash_md5_Init( &md5 );
         vlc_hash_md5_Update( &md5, psz_arturl, strlen( psz_arturl ) );
-        if( !strncmp( psz_arturl, "attachment://", 13 ) && psz_title != NULL )
+        if( ArtUrlIsAttachment( psz_arturl ) )
+        {
+            if( !EMPTY_STR( psz_uri ) )
+                vlc_hash_md5_Update( &md5, psz_uri, strlen( psz_uri ) );
+            else if( !EMPTY_STR( psz_title ) )
+                vlc_hash_md5_Update( &md5, psz_title, strlen( psz_title ) );
+        }
+        else if( !EMPTY_STR( psz_title ) )
             vlc_hash_md5_Update( &md5, psz_title, strlen( psz_title ) );
         vlc_hash_FinishHex( &md5, psz_arturl_sanitized );
         if( asprintf( &psz_dir, "%s" DIR_SEP "art" DIR_SEP "arturl" DIR_SEP
@@ -120,6 +186,7 @@ static char *ArtCachePath( input_item_t *p_item )
     const char *psz_arturl;
     const char *psz_title;
     const char *psz_date;
+    const char *psz_uri;
 
     vlc_mutex_lock( &p_item->lock );
 
@@ -133,13 +200,15 @@ static char *ArtCachePath( input_item_t *p_item )
     psz_arturl = vlc_meta_Get( p_item->p_meta, vlc_meta_ArtworkURL );
     psz_title = vlc_meta_Get( p_item->p_meta, vlc_meta_Title );
     psz_date = vlc_meta_Get( p_item->p_meta, vlc_meta_Date );
+    psz_uri = p_item->psz_uri;
     if( !psz_title )
         psz_title = p_item->psz_name;
 
     if( (EMPTY_STR(psz_artist) || EMPTY_STR(psz_album) ) && !psz_arturl )
         goto end;
 
-    psz_path = ArtCacheGetDirPath( psz_arturl, psz_artist, psz_album, psz_date, psz_title );
+    psz_path = ArtCacheGetDirPath( psz_arturl, psz_artist, psz_album,
+                                   psz_date, psz_title, psz_uri );
 
 end:
     vlc_mutex_unlock( &p_item->lock );
@@ -168,25 +237,137 @@ end:
     return psz_filename;
 }
 
-/* */
-int input_FindArtInCache( input_item_t *p_item )
+static char *ArtCacheNameFromDirPath( const char *psz_path, const char *psz_type )
 {
-    char *psz_path = ArtCachePath( p_item );
+    char *psz_ext = strdup( psz_type ? psz_type : "" );
+    char *psz_filename = NULL;
 
-    if( !psz_path )
-        return VLC_EGENERIC;
+    if( unlikely( !psz_path || !psz_ext ) )
+        goto end;
 
-    /* Check if file exists */
-    vlc_DIR *p_dir = vlc_opendir( psz_path );
-    if( !p_dir )
+    ArtCacheCreateDir( psz_path );
+    filename_sanitize( psz_ext );
+
+    if( asprintf( &psz_filename, "%s" DIR_SEP "art%s", psz_path, psz_ext ) < 0 )
+        psz_filename = NULL;
+
+end:
+    free( psz_ext );
+
+    return psz_filename;
+}
+
+static char *ArtCacheFilePath( const char *psz_dir, const char *psz_name )
+{
+    char *psz_file = NULL;
+
+    if( !psz_dir || !psz_name )
+        return NULL;
+
+    if( asprintf( &psz_file, "%s" DIR_SEP "%s", psz_dir, psz_name ) == -1 )
+        psz_file = NULL;
+
+    return psz_file;
+}
+
+static char *ArtCacheReadUriFromFile( const char *psz_file )
+{
+    if( !psz_file )
+        return NULL;
+
+    FILE *fd = vlc_fopen( psz_file, "rb" );
+    if( !fd )
+        return NULL;
+
+    char *psz_uri = NULL;
+    char sz_cachefile[2049];
+
+    if( fgets( sz_cachefile, sizeof( sz_cachefile ), fd ) != NULL )
     {
-        free( psz_path );
+        size_t i_len = strlen( sz_cachefile );
+        while( i_len > 0 &&
+               ( sz_cachefile[i_len - 1] == '\n' ||
+                 sz_cachefile[i_len - 1] == '\r' ) )
+            sz_cachefile[--i_len] = '\0';
+
+        if( i_len > 0 )
+            psz_uri = strdup( sz_cachefile );
+    }
+
+    fclose( fd );
+    return psz_uri;
+}
+
+static char *ArtCacheReadUriFromDirPath( const char *psz_dir )
+{
+    char *psz_uri = NULL;
+    char *psz_file = ArtCacheFilePath( psz_dir, "arturl" );
+
+    if( psz_file )
+    {
+        psz_uri = ArtCacheReadUriFromFile( psz_file );
+        free( psz_file );
+    }
+
+    return psz_uri;
+}
+
+static int ArtCacheWriteUriToFile( vlc_object_t *obj, const char *psz_file,
+                                   const char *psz_uri )
+{
+    if( !psz_file || !psz_uri )
+        return VLC_EGENERIC;
+
+    FILE *f = vlc_fopen( psz_file, "wb" );
+    if( !f )
         return VLC_EGENERIC;
+
+    int ret = VLC_SUCCESS;
+    if( fputs( psz_uri, f ) < 0 )
+    {
+        msg_Err( obj, "Error writing %s: %s", psz_file, vlc_strerror_c(errno) );
+        ret = VLC_EGENERIC;
     }
 
     bool b_found = false;
+    fclose( f );
+    return ret;
+}
+
+static char *ArtCacheGetAttachmentBlobDirPath( const void *p_data, size_t i_data )
+{
+    char *psz_cachedir = config_GetUserDir( VLC_CACHE_DIR );
+    char *psz_dir = NULL;
+
+    if( unlikely( psz_cachedir == NULL ) )
+        return NULL;
+
+    char psz_hash[VLC_HASH_MD5_DIGEST_HEX_SIZE];
+    vlc_hash_md5_t md5;
+    vlc_hash_md5_Init( &md5 );
+    vlc_hash_md5_Update( &md5, p_data, i_data );
+    vlc_hash_FinishHex( &md5, psz_hash );
+
+    if( asprintf( &psz_dir, "%s" DIR_SEP "art" DIR_SEP "artblob" DIR_SEP "%s",
+                  psz_cachedir, psz_hash ) == -1 )
+        psz_dir = NULL;
+
+    free( psz_cachedir );
+    return psz_dir;
+}
+
+static char *ArtCacheUriInPath( const char *psz_path )
+{
+    if( !psz_path )
+        return NULL;
+
+    vlc_DIR *p_dir = vlc_opendir( psz_path );
+    if( !p_dir )
+        return NULL;
+
+    char *psz_uri = NULL;
     const char *psz_filename;
-    while( !b_found && (psz_filename = vlc_readdir( p_dir )) )
+    while( psz_uri == NULL && (psz_filename = vlc_readdir( p_dir )) )
     {
         if( !strncmp( psz_filename, "art", 3 ) )
         {
@@ -194,23 +375,54 @@ int input_FindArtInCache( input_item_t *p_item )
             if( asprintf( &psz_file, "%s" DIR_SEP "%s",
                           psz_path, psz_filename ) != -1 )
             {
-                char *psz_uri = vlc_path2uri( psz_file, "file" );
-                if( psz_uri )
-                {
-                    input_item_SetArtURL( p_item, psz_uri );
-                    free( psz_uri );
-                }
+                psz_uri = vlc_path2uri( psz_file, "file" );
                 free( psz_file );
             }
-
-            b_found = true;
         }
     }
 
-    /* */
     vlc_closedir( p_dir );
+    return psz_uri;
+}
+
+static int ArtCacheFindInPath( input_item_t *p_item, char *psz_path )
+{
+    char *psz_uri = ArtCacheUriInPath( psz_path );
     free( psz_path );
-    return b_found ? VLC_SUCCESS : VLC_EGENERIC;
+    if( !psz_uri )
+        return VLC_EGENERIC;
+
+    input_item_SetArtURL( p_item, psz_uri );
+    free( psz_uri );
+    return VLC_SUCCESS;
+}
+
+static int ArtCacheFindAttachment( input_item_t *p_item )
+{
+    char *psz_path = ArtCacheAttachmentPath( p_item );
+    char *psz_uri = ArtCacheReadUriFromDirPath( psz_path );
+
+    if( psz_uri )
+    {
+        input_item_SetArtURL( p_item, psz_uri );
+        free( psz_uri );
+        free( psz_path );
+        return VLC_SUCCESS;
+    }
+
+    return ArtCacheFindInPath( p_item, psz_path );
+}
+
+/* */
+int input_FindArtInCache( input_item_t *p_item )
+{
+    /* Probe the item-scoped attachment cache even before metadata repopulates
+     * the current art URL. Otherwise cached embedded art is missed until a
+     * later parse re-discovers the attachment:// URL. */
+    if( ArtCacheFindAttachment( p_item ) == VLC_SUCCESS )
+        return VLC_SUCCESS;
+
+    return ArtCacheFindInPath( p_item, ArtCachePath( p_item ) );
 }
 
 static char * GetDirByItemUIDs( char *psz_uid )
@@ -230,16 +442,6 @@ static char * GetDirByItemUIDs( char *psz_uid )
     return psz_dir;
 }
 
-static char * GetFileByItemUID( char *psz_dir, const char *psz_type )
-{
-    char *psz_file;
-    if( asprintf( &psz_file, "%s" DIR_SEP "%s", psz_dir, psz_type ) == -1 )
-    {
-        psz_file = NULL;
-    }
-    return psz_file;
-}
-
 int input_FindArtInCacheUsingItemUID( input_item_t *p_item )
 {
     char *uid = input_item_GetInfo( p_item, "uid", "md5" );
@@ -252,21 +454,16 @@ int input_FindArtInCacheUsingItemUID( input_item_t *p_item )
     /* we have an input item uid set */
     bool b_done = false;
     char *psz_byuiddir = GetDirByItemUIDs( uid );
-    char *psz_byuidfile = GetFileByItemUID( psz_byuiddir, "arturl" );
+    char *psz_byuidfile = ArtCacheFilePath( psz_byuiddir, "arturl" );
     free( psz_byuiddir );
     if( psz_byuidfile )
     {
-        FILE *fd = vlc_fopen( psz_byuidfile, "rb" );
-        if ( fd )
+        char *psz_uri = ArtCacheReadUriFromFile( psz_byuidfile );
+        if( psz_uri )
         {
-            char sz_cachefile[2049];
-            /* read the cache hash url */
-            if ( fgets( sz_cachefile, 2048, fd ) != NULL )
-            {
-                input_item_SetArtURL( p_item, sz_cachefile );
-                b_done = true;
-            }
-            fclose( fd );
+            input_item_SetArtURL( p_item, psz_uri );
+            free( psz_uri );
+            b_done = true;
         }
         free( psz_byuidfile );
     }
@@ -280,46 +477,105 @@ int input_FindArtInCacheUsingItemUID( input_item_t *p_item )
 int input_SaveArt( vlc_object_t *obj, input_item_t *p_item,
                    const void *data, size_t length, const char *psz_type )
 {
-    char *psz_filename = ArtCacheName( p_item, psz_type );
+    char *psz_arturl = input_item_GetArtURL( p_item );
+    const bool b_attachment = ArtUrlIsAttachment( psz_arturl );
+    int i_ret = VLC_EGENERIC;
+    char *psz_filename = NULL;
+    char *psz_uri = NULL;
+    char *psz_attachment_dir = NULL;
+    char *psz_attachment_urlfile = NULL;
+
+    if( b_attachment )
+    {
+        psz_attachment_dir = ArtCacheAttachmentPath( p_item );
+        psz_attachment_urlfile = ArtCacheFilePath( psz_attachment_dir, "arturl" );
+
+        char *psz_blob_dir = ArtCacheGetAttachmentBlobDirPath( data, length );
+        if( psz_blob_dir )
+        {
+            psz_uri = ArtCacheUriInPath( psz_blob_dir );
+            if( psz_uri )
+                psz_filename = vlc_uri2path( psz_uri );
+            else
+                psz_filename = ArtCacheNameFromDirPath( psz_blob_dir, psz_type );
+
+            free( psz_blob_dir );
+        }
+    }
+    else
+        psz_filename = ArtCacheName( p_item, psz_type );
+    free( psz_arturl );
 
     if( !psz_filename )
+    {
+        free( psz_attachment_urlfile );
+        free( psz_attachment_dir );
         return VLC_EGENERIC;
+    }
 
-    char *psz_uri = vlc_path2uri( psz_filename, "file" );
     if( !psz_uri )
     {
-        free( psz_filename );
-        return VLC_EGENERIC;
+        psz_uri = vlc_path2uri( psz_filename, "file" );
+        if( !psz_uri )
+        {
+            free( psz_filename );
+            free( psz_attachment_urlfile );
+            free( psz_attachment_dir );
+            return VLC_EGENERIC;
+        }
     }
 
     /* Check if we already dumped it */
     struct stat s;
     if( !vlc_stat( psz_filename, &s ) )
     {
+        if( b_attachment && psz_attachment_dir && psz_attachment_urlfile )
+        {
+            ArtCacheCreateDir( psz_attachment_dir );
+            ArtCacheWriteUriToFile( obj, psz_attachment_urlfile, psz_uri );
+        }
         input_item_SetArtURL( p_item, psz_uri );
-        free( psz_filename );
-        free( psz_uri );
-        return VLC_SUCCESS;
+        i_ret = VLC_SUCCESS;
+        goto save_uid;
     }
 
     /* Dump it otherwise */
     FILE *f = vlc_fopen( psz_filename, "wb" );
-    if( f )
+    if( !f )
     {
-        if( fwrite( data, 1, length, f ) != length )
-        {
-            msg_Err( obj, "%s: %s", psz_filename, vlc_strerror_c(errno) );
-        }
-        else
-        {
-            msg_Dbg( obj, "album art saved to %s", psz_filename );
-            input_item_SetArtURL( p_item, psz_uri );
-        }
+        msg_Err( obj, "%s: %s", psz_filename, vlc_strerror_c(errno) );
+        goto end;
+    }
+
+    if( fwrite( data, 1, length, f ) != length )
+    {
+        msg_Err( obj, "%s: %s", psz_filename, vlc_strerror_c(errno) );
         fclose( f );
+        vlc_unlink( psz_filename );
+        goto end;
     }
-    free( psz_uri );
 
+    if( fclose( f ) )
+    {
+        msg_Err( obj, "%s: %s", psz_filename, vlc_strerror_c(errno) );
+        vlc_unlink( psz_filename );
+        goto end;
+    }
+
+    msg_Dbg( obj, "album art saved to %s", psz_filename );
+    if( b_attachment && psz_attachment_dir && psz_attachment_urlfile )
+    {
+        ArtCacheCreateDir( psz_attachment_dir );
+        ArtCacheWriteUriToFile( obj, psz_attachment_urlfile, psz_uri );
+    }
+    input_item_SetArtURL( p_item, psz_uri );
+    i_ret = VLC_SUCCESS;
+
+save_uid:
     /* save uid info */
+    if( i_ret != VLC_SUCCESS )
+        goto end;
+
     char *uid = input_item_GetInfo( p_item, "uid", "md5" );
     if ( ! *uid )
     {
@@ -328,26 +584,21 @@ int input_SaveArt( vlc_object_t *obj, input_item_t *p_item,
     }
 
     char *psz_byuiddir = GetDirByItemUIDs( uid );
-    char *psz_byuidfile = GetFileByItemUID( psz_byuiddir, "arturl" );
+    char *psz_byuidfile = ArtCacheFilePath( psz_byuiddir, "arturl" );
     ArtCacheCreateDir( psz_byuiddir );
     free( psz_byuiddir );
 
     if ( psz_byuidfile )
     {
-        f = vlc_fopen( psz_byuidfile, "wb" );
-        if ( f )
-        {
-            if( fputs( "file://", f ) < 0 || fputs( psz_filename, f ) < 0 )
-                msg_Err( obj, "Error writing %s: %s", psz_byuidfile,
-                         vlc_strerror_c(errno) );
-            fclose( f );
-        }
+        ArtCacheWriteUriToFile( obj, psz_byuidfile, psz_uri );
         free( psz_byuidfile );
     }
     free( uid );
     /* !save uid info */
 end:
+    free( psz_uri );
     free( psz_filename );
-    return VLC_SUCCESS;
+    free( psz_attachment_urlfile );
+    free( psz_attachment_dir );
+    return i_ret;
 }
-



View it on GitLab: https://code.videolan.org/videolan/vlc/-/compare/eaa69c55aa708e33aff1571ca7e98449b9c35194...297b8f9ed45d9e08837fe93bd8dea6ac6286442f

-- 
View it on GitLab: https://code.videolan.org/videolan/vlc/-/compare/eaa69c55aa708e33aff1571ca7e98449b9c35194...297b8f9ed45d9e08837fe93bd8dea6ac6286442f
You're receiving this email because of your account on code.videolan.org.




More information about the vlc-commits mailing list