You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

208 lines
6.9 KiB

#!/usr/bin/env python3
import Mobileclient from gmusicapi
import argparse
import sys
import requests
import json
import urllib.parse
creds_path = sys.path[0] + '/mobileclient.cred'
def authenticate(api):
creds = api.perform_oauth(storage_filepath=creds_path, open_browser=False)
def uploadLibrary(mudbase_api, mudbase_user, mudbase_password, songs):
# First, attempt to login and start a session.
s = requests.Session()
response = s.post(mudbase_api + '/login?username='
+ urllib.parse.quote(mudbase_user)
+ '&password='
+ urllib.parse.quote(mudbase_password))
if response.status_code != 200:
print("Unable to log in to MuDBase API.")
# Helpers
def getArtistStoreIds(song):
if 'artistId' in song:
return [song['artistId'][0]]
return []
def getSongStoreIds(song):
if 'storeId' in song:
return [song['storeId']]
return []
# Create GPM import tag
gpmTagIdResponse = s.post(mudbase_api + '/tag', data={
'name': 'GPM Import'
}).json()
gpmTagId = gpmTagIdResponse['id']
print(f"Created tag \"GPM Import\", response: {gpmTagIdResponse}")
# Create the root genre tag
genreRootResponse = s.post(mudbase_api + '/tag', data={
'name': 'Genre'
}).json()
genreRootTagId = genreRootResponse['id']
print(f"Created tag \"Genre\", response: {genreRootResponse}")
# For keeping track what we have already created.
storedArtists = dict()
storedAlbums = dict()
storedGenreTags = dict()
for song in songs:
# TODO: check if these items already exist
# Determine artist properties.
artist = {
'name': song['artist'],
'storeLinks': ['https://play.google.com/music/m' + id for id in getArtistStoreIds(song)],
'tagIds': [gpmTagId]
} if 'artist' in song else None
# Determine album properties.
album = {
'name': song['album'],
'tagIds': [gpmTagId]
} if 'album' in song else None
# Determine genre properties.
genre = {
'name': song['genre'],
'parentId': genreRootTagId
} if 'genre' in song else None
# Upload artist if not already done
artistId = None
if artist:
for key, value in storedArtists.items():
if value == artist:
artistId = key
break
if not artistId:
response = s.post(mudbase_api + '/artist', json=artist).json()
artistId = response['id']
print(
f"Created artist \"{artist['name']}\", response: {response}")
storedArtists[artistId] = artist
# Upload album if not already done
albumId = None
if album:
for key, value in storedAlbums.items():
if value == album:
albumId = key
break
if not albumId:
response = s.post(mudbase_api + '/album', json=album).json()
albumId = response['id']
print(
f"Created album \"{album['name']}\", response: {response}")
storedAlbums[albumId] = album
# Upload genre if not already done
genreTagId = None
if genre:
for key, value in storedGenreTags.items():
if value == genre:
genreTagId = key
break
if not genreTagId:
response = s.post(mudbase_api + '/tag', json=genre).json()
genreTagId = response['id']
print(
f"Created genre tag \"Genre / {genre['name']}\", response: {response}")
storedGenreTags[genreTagId] = genre
# Upload the song itself
tagIds = [gpmTagId]
if genreTagId:
tagIds.append(genreTagId)
_song = {
'title': song['title'],
'artistIds': [artistId] if artistId != None else [],
'albumIds': [albumId] if albumId != None else [],
'tagIds': tagIds,
'storeLinks': ['https://play.google.com/music/m/' + id for id in getSongStoreIds(song)],
}
response = s.post(mudbase_api + '/song', json=_song).json()
print(
f"Created song \"{song['title']}\" with artist ID {artistId}, album ID {albumId}, response: {response}")
def getData(api):
return {
"songs": api.get_all_songs(),
"playlists": api.get_all_user_playlist_contents()
}
def getSongs(data):
# Get songs from library
songs = [] # data['songs']
# Append songs from playlists
for playlist in data['playlists']:
for track in playlist['tracks']:
if 'track' in track:
songs.append(track['track'])
# Uniquify by using a dict. After all, same song may appear in
# multiple playlists.
def sI(song): return song['artist'] + '-' + \
song['title'] if 'artist' in song and 'title' in song else 'z'
return list(dict((sI(song), song) for song in songs).values())
api = Mobileclient()
parser = argparse.ArgumentParser(
description="Import Google Music library into MudBase.")
parser.add_argument(
'--authenticate', help="Generate credentials for authentication", action="store_true")
parser.add_argument('--store-to', help="Store GPM library to JSON for later upload",
action='store', dest='store_to')
parser.add_argument('--load-from', help="Load GPM library from JSON for upload",
action='store', dest='load_from')
parser.add_argument('--mudbase_api', help="Address for the Mudbase back-end API to upload to",
action='store', dest='mudbase_api')
parser.add_argument('--mudbase_user', help="Username for the Mudbase API",
action='store', dest="mudbase_user")
parser.add_argument('--mudbase_password', help="Password for the Mudbase API",
action='store', dest="mudbase_password")
args = parser.parse_args()
if args.authenticate:
authenticate(api)
data = None
# Determine whether we need to log in to GPM and get songs
if args.store_to or (not args.load_from and args.mudbase_api):
api.oauth_login(Mobileclient.FROM_MAC_ADDRESS,
oauth_credentials=creds_path)
data = getData(api)
# Determine whether to save to a file
if args.store_to:
with open(args.store_to, 'w') as outfile:
json.dump(data, outfile, sort_keys=True, indent=2)
# Determine whether to load from a file
if args.load_from:
with open(args.load_from, 'r') as f:
data = json.load(f)
songs = getSongs(data)
print(f"Found {len(songs)} songs.")
if args.mudbase_api:
api.oauth_login(Mobileclient.FROM_MAC_ADDRESS,
oauth_credentials=creds_path)
uploadLibrary(args.mudbase_api, args.mudbase_user,
args.mudbase_password, songs)