blob: 3a091c3f401de0535000c9b1d44650c35a57d096 [file] [log] [blame]
#include "common.h"
#include <string.h>
#include <sys/stat.h>
#ifdef HAVE_LIBGEN_H
#include <libgen.h> /* basename() */
#endif
/*
* This program is derived from the exact equivalent in libnjb.
*
* This is an improved commandline track transfer program
* based on Enrique Jorreto Ledesma's work on the original program by
* Shaun Jackman and Linus Walleij.
*/
/* Function that compensate for missing libgen.h on Windows */
#ifndef HAVE_LIBGEN_H
static char *basename(char *in) {
char *p;
if (in == NULL)
return NULL;
p = in + strlen(in) - 1;
while (*p != '\\' && *p != '/' && *p != ':')
{ p--; }
return ++p;
}
#endif
static int progress (uint64_t const sent, uint64_t const total, void const * const data)
{
int percent = (sent*100)/total;
#ifdef __WIN32__
printf("Progress: %I64u of %I64u (%d%%)\r", sent, total, percent);
#else
printf("Progress: %llu of %llu (%d%%)\r", sent, total, percent);
#endif
fflush(stdout);
return 0;
}
static char *prompt (const char *prompt, char *buffer, size_t bufsz, int required)
{
char *cp, *bp;
while (1) {
fprintf(stdout, "%s> ", prompt);
if ( fgets(buffer, bufsz, stdin) == NULL ) {
if (ferror(stdin)) {
perror("fgets");
} else {
fprintf(stderr, "EOF on stdin\n");
}
return NULL;
}
cp = strrchr(buffer, '\n');
if ( cp != NULL ) *cp = '\0';
bp = buffer;
while ( bp != cp ) {
if ( *bp != ' ' && *bp != '\t' ) return bp;
bp++;
}
if (! required) return bp;
}
}
static void usage(void)
{
fprintf(stderr, "usage: sendtr [ -D debuglvl ] [ -q ] -t <title> -a <artist> -l <album>\n");
fprintf(stderr, " -c <codec> -g <genre> -n <track number> -y <year> \n");
fprintf(stderr, " -d <duration in seconds> -f \"Folder Name\" <path>\n");
fprintf(stderr, "(-q means the program will not ask for missing information.)\n");
exit(1);
}
static uint32_t find_folder_list(char *name, LIBMTP_folder_t *folderlist, int level)
{
uint32_t i;
if(folderlist==NULL) {
return 0;
}
if(!strcasecmp(name, folderlist->name))
return folderlist->folder_id;
if ((i = (find_folder_list(name, folderlist->child, level+1))))
return i;
if ((i = (find_folder_list(name, folderlist->sibling, level))))
return i;
return 0;
}
int main(int argc, char **argv)
{
int opt;
extern int optind;
extern char *optarg;
char *path, *filename;
char artist[80], title[80], genre[80], codec[80], album[80];
char num[80];
char *partist = NULL;
char *ptitle = NULL;
char *pgenre = NULL;
char *pcodec = NULL;
char *palbum = NULL;
char *pfolder = NULL;
uint16_t tracknum = 0;
uint16_t length = 0;
uint16_t year = 0;
uint16_t quiet = 0;
uint64_t filesize;
uint32_t parent_id = 0;
struct stat sb;
char *lang;
LIBMTP_mtpdevice_t *device;
LIBMTP_folder_t *folders = NULL;
LIBMTP_track_t *trackmeta;
int ret;
LIBMTP_Init();
while ( (opt = getopt(argc, argv, "qD:t:a:l:c:g:n:d:y:f:")) != -1 ) {
switch (opt) {
case 't':
ptitle = strdup(optarg);
break;
case 'a':
partist = strdup(optarg);
break;
case 'l':
palbum = strdup(optarg);
break;
case 'c':
pcodec = strdup(optarg); // FIXME: DSM check for MP3, WAV or WMA
break;
case 'g':
pgenre = strdup(optarg);
break;
case 'n':
tracknum = atoi(optarg);
break;
case 'd':
length = atoi(optarg);
break;
case 'y':
year = atoi(optarg);
break;
case 'f':
pfolder = strdup(optarg);
break;
case 'q':
quiet = 1;
break;
default:
usage();
}
}
argc -= optind;
argv += optind;
if ( argc != 1 ) {
printf("You need to pass a filename.\n");
usage();
}
/*
* Check environment variables $LANG and $LC_CTYPE
* to see if we want to support UTF-8 unicode
*/
lang = getenv("LANG");
if (lang != NULL) {
if (strlen(lang) > 5) {
char *langsuff = &lang[strlen(lang)-5];
if (strcmp(langsuff, "UTF-8")) {
printf("Your system does not appear to have UTF-8 enabled ($LANG=\"%s\")\n", lang);
printf("If you want to have support for diacritics and Unicode characters,\n");
printf("please switch your locale to an UTF-8 locale, e.g. \"en_US.UTF-8\".\n");
}
}
}
path = argv[0];
filename = basename(path);
if (filename == NULL) {
printf("Error: filename could not be based.\n");
exit(1);
}
if ( stat(path, &sb) == -1 ) {
fprintf(stderr, "%s: ", path);
perror("stat");
exit(1);
}
filesize = (uint64_t) sb.st_size;
/* Ask for missing parameters if not quiet */
if (!quiet) {
if (pcodec == NULL) {
if ( (pcodec = prompt("Codec", codec, 80, 1)) == NULL ) {
printf("A codec name is required.\n");
usage();
}
}
if (!strlen(pcodec)) {
printf("A codec name is required.\n");
usage();
}
if (ptitle == NULL) {
if ( (ptitle = prompt("Title", title, 80, 0)) == NULL )
usage();
}
if (!strlen(ptitle))
ptitle = NULL;
if (palbum == NULL) {
if ( (palbum = prompt("Album", album, 80, 0)) == NULL )
usage();
}
if (!strlen(palbum))
palbum = NULL;
if (partist == NULL) {
if ( (partist = prompt("Artist", artist, 80, 0)) == NULL )
usage();
}
if (!strlen(partist))
partist = NULL;
if (pgenre == NULL) {
if ( (pgenre = prompt("Genre", genre, 80, 0)) == NULL )
usage();
}
if (!strlen(pgenre))
pgenre = NULL;
if (tracknum == 0) {
char *pnum;
if ( (pnum = prompt("Track number", num, 80, 0)) == NULL )
tracknum = 0;
if ( strlen(pnum) ) {
tracknum = strtoul(pnum, 0, 10);
} else {
tracknum = 0;
}
}
if (year == 0) {
char *pnum;
if ( (pnum = prompt("Year", num, 80, 0)) == NULL )
year = 0;
if ( strlen(pnum) ) {
year = strtoul(pnum, 0, 10);
} else {
year = 0;
}
}
if (length == 0) {
char *pnum;
if ( (pnum = prompt("Length", num, 80, 0)) == NULL )
length = 0;
if ( strlen(pnum) ) {
length = strtoul(pnum, 0, 10);
} else {
length = 0;
}
}
}
trackmeta = LIBMTP_new_track_t();
printf("Sending track:\n");
printf("Codec: %s\n", pcodec);
if (!strcasecmp(pcodec,"mp3")) {
trackmeta->filetype = LIBMTP_FILETYPE_MP3;
} else if (!strcasecmp(pcodec,"wav")) {
trackmeta->filetype = LIBMTP_FILETYPE_WAV;
} else if (!strcasecmp(pcodec,"ogg")) {
trackmeta->filetype = LIBMTP_FILETYPE_OGG;
} else if (!strcasecmp(pcodec,"mp4")) {
trackmeta->filetype = LIBMTP_FILETYPE_MP4;
} else if (!strcasecmp(pcodec,"wma") || !strcasecmp(pcodec,"asf")) {
trackmeta->filetype = LIBMTP_FILETYPE_WMA;
} else {
printf("Not a valid codec: \"%s\"\n", pcodec);
printf("Supported formats: MP3, WAV, OGG, MP4, WMA\n");
exit(1);
}
printf("Codec: %s\n", LIBMTP_Get_Filetype_Description(trackmeta->filetype));
if (ptitle) {
printf("Title: %s\n", ptitle);
trackmeta->title = strdup(ptitle);
}
if (palbum) {
printf("Album: %s\n", palbum);
trackmeta->album = strdup(palbum);
}
if (partist) {
printf("Artist: %s\n", partist);
trackmeta->artist = strdup(partist);
}
if (pgenre) {
printf("Genre: %s\n", pgenre);
trackmeta->genre = strdup(pgenre);
}
if (year > 0) {
char tmp[80];
printf("Year: %d\n", year);
snprintf(tmp, sizeof(tmp)-1, "%4d0101T0000.0", year);
tmp[sizeof(tmp)-1] = '\0';
trackmeta->date = strdup(tmp);
}
if (tracknum > 0) {
printf("Track no: %d\n", tracknum);
trackmeta->tracknumber = tracknum;
}
if (length > 0) {
printf("Length: %d\n", length);
// Multiply by 1000 since this is in milliseconds
trackmeta->duration = length * 1000;
}
if (pfolder != NULL && parent_id != 0) {
printf("Folder: %s (ID: %d)\n", pfolder, parent_id);
}
// We should always have this
if (filename != NULL) {
trackmeta->filename = strdup(filename);
}
trackmeta->filesize = filesize;
device = LIBMTP_Get_First_Device();
if (device == NULL) {
printf("No MTP device found.\n");
LIBMTP_destroy_track_t(trackmeta);
exit(1);
}
// If a folder argument was passed in, try to locate the folder.
if (pfolder) {
folders = LIBMTP_Get_Folder_List(device);
if(folders == NULL) {
printf("No folders found, ignoring folder argument.\n");
} else {
parent_id = find_folder_list(pfolder, folders, 0);
if (!parent_id) {
printf("Parent folder could not be found, ignoring folder argument.\n");
}
}
}
printf("Sending track...\n");
ret = LIBMTP_Send_Track_From_File(device, path, trackmeta, progress, NULL, parent_id);
printf("\n");
LIBMTP_Release_Device(device);
printf("New track ID: %d\n", trackmeta->item_id);
LIBMTP_destroy_track_t(trackmeta);
printf("OK.\n");
return 0;
}