[artwork] Refactor artwork_get_dir_image

This commit is contained in:
chme 2020-02-25 06:07:52 +01:00
parent 6ba45e8d94
commit 67fd555b52

View File

@ -723,52 +723,53 @@ artwork_evbuf_rescale(struct evbuffer *artwork, struct evbuffer *raw, int max_w,
return -1; return -1;
} }
/* Looks for an artwork file in a directory. Will rescale if needed. /*
* Checks if an image file with one of the configured artwork_basenames exists in
* the given directory "dir". Returns 0 if an image exists, -1 if no image was
* found or an error occurred.
* *
* @out evbuf Image data * If an image exists, "out_path" will contain the absolute path to this image.
* @in dir Directory to search *
* @in max_w Requested width * @param out_path If return value is 0, contains the absolute path to the image
* @in max_h Requested height * @param len If return value is 0, contains the length of the absolute path
* @out out_path Path to the artwork file if found, must be a char[PATH_MAX] buffer * @param dir The directory to search
* @return ART_FMT_* on success, ART_E_NONE on nothing found, ART_E_ERROR on error * @return 0 if image exists, -1 otherwise
*/ */
static int static int
artwork_get_bydir(struct evbuffer *evbuf, char *dir, int max_w, int max_h, char *out_path) dir_image_find(char *out_path, size_t len, const char *dir)
{ {
char path[PATH_MAX]; char path[PATH_MAX];
char parentdir[PATH_MAX];
int i; int i;
int j; int j;
int len; int path_len;
int ret; int ret;
cfg_t *lib; cfg_t *lib;
int nbasenames; int nbasenames;
int nextensions; int nextensions;
char *ptr;
ret = snprintf(path, sizeof(path), "%s", dir); ret = snprintf(path, sizeof(path), "%s", dir);
if ((ret < 0) || (ret >= sizeof(path))) if ((ret < 0) || (ret >= sizeof(path)))
{ {
DPRINTF(E_LOG, L_ART, "Artwork path exceeds PATH_MAX (%s)\n", dir); DPRINTF(E_LOG, L_ART, "Artwork path exceeds PATH_MAX (%s)\n", dir);
return ART_E_ERROR; return -1;
} }
len = strlen(path); path_len = strlen(path);
lib = cfg_getsec(cfg, "library"); lib = cfg_getsec(cfg, "library");
nbasenames = cfg_size(lib, "artwork_basenames"); nbasenames = cfg_size(lib, "artwork_basenames");
if (nbasenames == 0) if (nbasenames == 0)
return ART_E_NONE; return -1;
nextensions = sizeof(cover_extension) / sizeof(cover_extension[0]); nextensions = ARRAY_SIZE(cover_extension);
for (i = 0; i < nbasenames; i++) for (i = 0; i < nbasenames; i++)
{ {
for (j = 0; j < nextensions; j++) for (j = 0; j < nextensions; j++)
{ {
ret = snprintf(path + len, sizeof(path) - len, "/%s.%s", cfg_getnstr(lib, "artwork_basenames", i), cover_extension[j]); ret = snprintf(path + path_len, sizeof(path) - path_len, "/%s.%s", cfg_getnstr(lib, "artwork_basenames", i), cover_extension[j]);
if ((ret < 0) || (ret >= sizeof(path) - len)) if ((ret < 0) || (ret >= sizeof(path) - path_len))
{ {
DPRINTF(E_LOG, L_ART, "Artwork path will exceed PATH_MAX (%s/%s)\n", dir, cfg_getnstr(lib, "artwork_basenames", i)); DPRINTF(E_LOG, L_ART, "Artwork path will exceed PATH_MAX (%s/%s)\n", dir, cfg_getnstr(lib, "artwork_basenames", i));
continue; continue;
@ -777,39 +778,62 @@ artwork_get_bydir(struct evbuffer *evbuf, char *dir, int max_w, int max_h, char
DPRINTF(E_SPAM, L_ART, "Trying directory artwork file %s\n", path); DPRINTF(E_SPAM, L_ART, "Trying directory artwork file %s\n", path);
ret = access(path, F_OK); ret = access(path, F_OK);
if (ret < 0) if (ret == 0)
continue;
// If artwork file exists (ret == 0), exit the loop
break;
}
// In case the previous loop exited early, we found an existing artwork file and exit the outer loop
if (j < nextensions)
break;
}
// If the loop for directory artwork did not exit early, look for parent directory artwork
if (i == nbasenames)
{ {
ptr = strrchr(path, '/'); snprintf(out_path, len, "%s", path);
if (ptr) return 0;
*ptr = '\0'; }
}
}
return -1;
}
/*
* Checks if an image file exists in the given directory "dir" with the basename
* equal to the directory name. Returns 0 if an image exists, -1 if no image was
* found or an error occurred.
*
* If an image exists, "out_path" will contain the absolute path to this image.
*
* @param out_path If return value is 0, contains the absolute path to the image
* @param len If return value is 0, contains the length of the absolute path
* @param dir The directory to search
* @return 0 if image exists, -1 otherwise
*/
static int
parent_dir_image_find(char *out_path, size_t len, const char *dir)
{
char path[PATH_MAX];
char parentdir[PATH_MAX];
char *ptr;
int i;
int nextensions;
int path_len;
int ret;
ret = snprintf(path, sizeof(path), "%s", dir);
if ((ret < 0) || (ret >= sizeof(path)))
{
DPRINTF(E_LOG, L_ART, "Artwork path exceeds PATH_MAX (%s)\n", dir);
return -1;
}
ptr = strrchr(path, '/'); ptr = strrchr(path, '/');
if ((!ptr) || (strlen(ptr) <= 1)) if ((!ptr) || (strlen(ptr) <= 1))
{ {
DPRINTF(E_LOG, L_ART, "Could not find parent dir name (%s)\n", path); DPRINTF(E_LOG, L_ART, "Could not find parent dir name (%s)\n", path);
return ART_E_ERROR; return -1;
} }
strcpy(parentdir, ptr + 1); strcpy(parentdir, ptr + 1);
len = strlen(path); path_len = strlen(path);
nextensions = ARRAY_SIZE(cover_extension);
for (i = 0; i < nextensions; i++) for (i = 0; i < nextensions; i++)
{ {
ret = snprintf(path + len, sizeof(path) - len, "/%s.%s", parentdir, cover_extension[i]); ret = snprintf(path + path_len, sizeof(path) - path_len, "/%s.%s", parentdir, cover_extension[i]);
if ((ret < 0) || (ret >= sizeof(path) - len)) if ((ret < 0) || (ret >= sizeof(path) - path_len))
{ {
DPRINTF(E_LOG, L_ART, "Artwork path will exceed PATH_MAX (%s)\n", parentdir); DPRINTF(E_LOG, L_ART, "Artwork path will exceed PATH_MAX (%s)\n", parentdir);
continue; continue;
@ -818,19 +842,44 @@ artwork_get_bydir(struct evbuffer *evbuf, char *dir, int max_w, int max_h, char
DPRINTF(E_SPAM, L_ART, "Trying parent directory artwork file %s\n", path); DPRINTF(E_SPAM, L_ART, "Trying parent directory artwork file %s\n", path);
ret = access(path, F_OK); ret = access(path, F_OK);
if (ret < 0) if (ret == 0)
continue; {
snprintf(out_path, len, "%s", path);
break; return 0;
}
}
return -1;
}
/* Looks for an artwork file in a directory. Will rescale if needed.
*
* @out evbuf Image data
* @in dir Directory to search
* @in max_w Requested width
* @in max_h Requested height
* @out out_path Path to the artwork file if found, must be a char[PATH_MAX] buffer
* @in len Max size of "out_path"
* @return ART_FMT_* on success, ART_E_NONE on nothing found, ART_E_ERROR on error
*/
static int
artwork_get_bydir(struct evbuffer *evbuf, char *dir, int max_w, int max_h, char *out_path, size_t len)
{
int ret;
ret = dir_image_find(out_path, len, dir);
if (ret >= 0)
{
return artwork_get(evbuf, out_path, NULL, max_w, max_h, false);
}
ret = parent_dir_image_find(out_path, len, dir);
if (ret >= 0)
{
return artwork_get(evbuf, out_path, NULL, max_w, max_h, false);
} }
if (i == nextensions)
return ART_E_NONE; return ART_E_NONE;
}
snprintf(out_path, PATH_MAX, "%s", path);
return artwork_get(evbuf, path, NULL, max_w, max_h, false);
} }
/* Retrieves artwork from an URL, will rescale if needed. Checks the cache stash /* Retrieves artwork from an URL, will rescale if needed. Checks the cache stash
@ -1314,7 +1363,7 @@ source_group_dir_get(struct artwork_ctx *ctx)
if (access(dir, F_OK) < 0) if (access(dir, F_OK) < 0)
continue; continue;
ret = artwork_get_bydir(ctx->evbuf, dir, ctx->max_w, ctx->max_h, ctx->path); ret = artwork_get_bydir(ctx->evbuf, dir, ctx->max_w, ctx->max_h, ctx->path, sizeof(ctx->path));
if (ret > 0) if (ret > 0)
{ {
db_query_end(&qp); db_query_end(&qp);