Compare commits

...

17 Commits
0.1.0 ... main

Author SHA1 Message Date
Jonathan Bernard
a6371574a7 Allow multiple commands on one line using and or ;. 2017-01-16 04:22:22 -06:00
Jonathan Bernard
92d384573e Rescan the media library before reporting on absent files. 2016-09-08 14:18:20 -05:00
Jonathan Bernard
4468f606ed Bump H2 DB version. 2016-09-08 14:13:04 -05:00
Jonathan Bernard
4507c6b664 Added and commands to CLI. 2016-08-01 06:25:03 -05:00
Jonathan Bernard
ffcfc7bc77 Simplify the new playlist creation logic. 2016-06-01 11:31:05 -05:00
Jonathan Bernard
451956dc4f Stop the player before clearing the play queue. 2016-06-01 11:30:45 -05:00
Jonathan Bernard
a2c5c13ef4 Disable reporting of repeat mode until a working implementation can be found. 2016-06-01 11:30:02 -05:00
Jonathan Bernard
cbf01b4d73 Fix a bug where playlist media counts where being overwritten with stale data. 2016-06-01 11:29:39 -05:00
Jonathan Bernard
aa640bb4f9 Stop playBookmak from being updated when a song is paused. 2016-06-01 11:29:10 -05:00
Jonathan Bernard
7101f3fd53 Made playlists default to user-created. 2016-06-01 11:28:27 -05:00
Jonathan Bernard
19e21811e2 Switched to db-migrate.groovy configuration. 2016-06-01 11:26:53 -05:00
Jonathan Bernard
4007f8a479 Fix bug in help code. 2016-05-22 07:15:53 -05:00
Jonathan Bernard
981fb51af3 Added installation/build instructions. 2016-04-21 08:27:43 -05:00
Jonathan Bernard
bbf8a019f9 Added maven plugin to build. 2016-04-21 07:25:46 -05:00
Jonathan Bernard
816820c427 Use java system properties to find the user's home directory before looking to ENV properties. 2016-04-21 00:26:50 -05:00
Jonathan Bernard
7bd9c64c44 README title was ugly when displayed by gogs. 2016-04-20 12:46:31 -05:00
Jonathan Bernard
ed678872c9 Updated README for new version. 2016-04-20 12:45:28 -05:00
8 changed files with 181 additions and 52 deletions

View File

@ -1,4 +1,5 @@
# **W**hat **Do** **I** **W**ant **T**o **L**isten **T**o # What Do I Want To Listen To
## A simple, tag-based music library manager. ## A simple, tag-based music library manager.
This project is born out of a frustration I had managing my music library. This project is born out of a frustration I had managing my music library.
@ -6,16 +7,70 @@ I have found playlists, genres, and other ways of organizing my music too
restrictive and cumbersome to keep up with. Here are the main features I want restrictive and cumbersome to keep up with. Here are the main features I want
out of music player, in order of priority: out of music player, in order of priority:
### Implemented
* Song tagging.
* Playlists.
* Bookmarks (temporary song tag marking location in a playlist/album)
* Read meta-data from ID3.
### TODO
* Web-based interface. * Web-based interface.
* Mobile interface (could implement the Subsonic API on the back-end to be able * Mobile interface (could implement the Subsonic API on the back-end to be able
to use pre-made mobile apps) to use pre-made mobile apps)
* Song tagging.
* Playlists.
* Read meta-data from ID3.
* Transcoding on the fly. * Transcoding on the fly.
* Read songs from Amazon S3. * Read songs from Amazon S3.
* Stream from Amazon Cloudfront. * Stream from Amazon Cloudfront.
* Bookmarks (temporary song tag marking location in a playlist/album)
I have not found a music manager that gives me all of the above, so I'm going I have not found a music manager that gives me all of the above, so I'm
to write my own. writing my own.
## Overview
WDIWTLT is currently made up of two subprojects:
* `core`
* `cli`
### `core`
`core` contains the data layer implementation, built with a lightweight ORM
layer over JDBC, and common functionality for managing a media library.
### `cli`
`cli` is a command-line interface built using the WDIWTLT core and VLC for
media playback.
## Install
The current version, 0.1.0, is an alpha version. The CLI client depends on
[VLC](http://www.videolan.org/vlc/) and expects it to already be installed on
the system.
To install the CLI client:
$ wget http://mvn.jdb-labs.com/repo/com/jdbernard/wdiwtlt-cli/0.1.0/wdiwtlt-cli-0.1.0.zip
$ unzip wdiwtlt-cli-0.1.0.zip
The `wdiwtlt-cli` binary is located in `wdiwtlt-cli/bin`, which you can add to
you `PATH` environment variable.
## Building From Source
The `wdiwtlt` project is written in [Groovy](http://www.groovy-lang.org/) uses
the [Gradle](http://gradle.org) build tool. If you do not already have a Groovy
environment installed you can do:
$ curl -s https://get.sdkman.io | bash
$ source "$HOME/.sdkman/bin/sdkman-init.sh"
$ yes | sdkman install groovy
$ yes | sdkman install gradle
Once you have Groovy and Gradle, the `wdiwtlt` project can be built by invoking
`gradle assembleDist`.
$ git clone https://git.jdb-labs.com/jdb/wdiwtlt.git
$ cd wdiwtlt
$ gradle assembleDist

View File

@ -12,11 +12,14 @@ allprojects {
} }
group = 'com.jdbernard' group = 'com.jdbernard'
version = '0.1.0' version = '0.1.3'
repositories { repositories {
mavenLocal() mavenLocal()
mavenCentral() mavenCentral()
maven { url "https://dl.bintray.com/ijabz/maven" } maven { url "https://dl.bintray.com/ijabz/maven" }
maven { url "http://mvn.jdb-labs.com/repo" }
} }
apply plugin: 'maven'
} }

View File

@ -28,7 +28,7 @@ import static com.jdbernard.wdiwtlt.cli.CliErr.*
public class CommandLineInterface { public class CommandLineInterface {
public static final VERSION = "ALPHA" public static final VERSION = "0.1.3"
public static final def DOC = """\ public static final def DOC = """\
wdiwtlt v$VERSION wdiwtlt v$VERSION
@ -97,14 +97,14 @@ Configuration:
private String titleStyle, normalStyle, statusStyle, promptStyle, private String titleStyle, normalStyle, statusStyle, promptStyle,
artistStyle, albumStyle, fileStyle, errorStyle, playlistStyle, artistStyle, albumStyle, fileStyle, errorStyle, playlistStyle,
cmdStyle, optStyle cmdStyle, optStyle
private String eraseToEnd = new ANSI().eraseLine(Erase.ToEnd).toString()
private String clearLine = new ANSI().eraseLine(Erase.All).toString() private String clearLine = new ANSI().eraseLine(Erase.All).toString()
private String afterInput = private String afterInput =
new ANSI().eraseLine(Erase.All).scrollUp().cursorUp().toString() new ANSI().eraseLine(Erase.All).scrollUp().cursorUp().toString()
private String beforeLeader = private String beforeLeader =
new ANSI().saveCursor().cursorPrevLine(3).toString() new ANSI().saveCursor().cursorPrevLine(3).toString()
private String afterLeader = private String afterLeader =
new ANSI().restoreCursor().eraseLine(Erase.ToEnd).toString() new ANSI().restoreCursor().toString()
private String eraseLeader = private String eraseLeader =
new ANSI().eraseLine(Erase.All).cursorPrevLine().eraseLine(Erase.All) new ANSI().eraseLine(Erase.All).cursorPrevLine().eraseLine(Erase.All)
.cursorPrevLine().eraseLine(Erase.All) .cursorPrevLine().eraseLine(Erase.All)
@ -125,6 +125,7 @@ Configuration:
private ScrollText status = new ScrollText(maxWidth: displayWidth) private ScrollText status = new ScrollText(maxWidth: displayWidth)
private Date dismissMsgDate = new Date() private Date dismissMsgDate = new Date()
private SimpleDateFormat sdf = new SimpleDateFormat('EEE-HH-SSS') private SimpleDateFormat sdf = new SimpleDateFormat('EEE-HH-SSS')
private Random rand = new Random()
/// Current play queue and selection data /// Current play queue and selection data
List<Model> currentSelection = [] List<Model> currentSelection = []
@ -151,8 +152,13 @@ Configuration:
if (!cfgFile || !cfgFile.exists() || !cfgFile.isFile()) if (!cfgFile || !cfgFile.exists() || !cfgFile.isFile())
cfgFile = new File("wdiwtlt.cli.properties") cfgFile = new File("wdiwtlt.cli.properties")
if (!cfgFile || !cfgFile.exists() || !cfgFile.isFile()) if (!cfgFile || !cfgFile.exists() || !cfgFile.isFile()) {
cfgFile = new File(System.getenv().HOME, ".wdiwtlt.cli.properties") String userHome = System.getenv().HOME
if (userHome) cfgFile = new File(userHome, ".wdiwtlt.cli.properties")
if (!cfgFile || !cfgFile.exists() || !cfgFile.isFile()) {
userHome = System.getProperty('user.home')
if (userHome) cfgFile = new File(userHome, 'wdiwtlt.cli.properties') } }
if (cfgFile.exists() && cfgFile.isFile()) { if (cfgFile.exists() && cfgFile.isFile()) {
try { cfgFile.withInputStream { givenCfg.load(it) } } try { cfgFile.withInputStream { givenCfg.load(it) } }
@ -273,6 +279,9 @@ Configuration:
// Try to discover the VLC native libraries // Try to discover the VLC native libraries
def vlcj def vlcj
try { try {
String vlcLibDir = opts['--vlc-lib-dir'] || givenCfg['vlc.lib.dir']
//if (vlcLibDir) NativeLibrary.addSearchPath('libvlc', vlcLibDir)
new NativeDiscovery().discover() new NativeDiscovery().discover()
vlcj = new AudioMediaListPlayerComponent() } vlcj = new AudioMediaListPlayerComponent() }
catch (Exception e) { catch (Exception e) {
@ -305,7 +314,8 @@ Configuration:
library.clean() library.clean()
playQueue = library.save(new Playlist( playQueue = library.save(new Playlist(
name: "CLI Queue ${sdf.format(new Date())}")) name: "CLI Queue ${sdf.format(new Date())}",
userCreated: false))
String user = System.getProperty('user.name') String user = System.getProperty('user.name')
String os = System.getProperty('os.name') String os = System.getProperty('os.name')
@ -381,10 +391,14 @@ Configuration:
printLongMessage(errorStyle + errMsg + normalStyle) } printLongMessage(errorStyle + errMsg + normalStyle) }
else { else {
status.text = errorStyle + errMsg + normalStyle status.text = errorStyle + errMsg + normalStyle
dismissMsgDate = new Date(new Date().time + msgTimeout) } } } dismissMsgDate = new Date(new Date().time + msgTimeout) } }
outStream.print eraseToEnd }
else { else {
drawLeader() drawLeader()
if (curMediaFile && System.currentTimeMillis() > nextBookmarkUpdate) { if (curMediaFile &&
vlcj.mediaListPlayer.mediaPlayer.isPlaying() &&
System.currentTimeMillis() > nextBookmarkUpdate) {
playBookmark.playTimeMs = playBookmark.playTimeMs =
vlcj.mediaListPlayer.mediaPlayer.time vlcj.mediaListPlayer.mediaPlayer.time
library.save(playBookmark) library.save(playBookmark)
@ -392,8 +406,13 @@ Configuration:
Thread.sleep(250) } } } Thread.sleep(250) } } }
private def processInput(String line) { private def processInput(String line) {
line = line.trim()
logger.debug("line: $line") logger.debug("line: $line")
if (line.indexOf(' and ') > 0 || line.indexOf(';')> 0) {
String[] cmds = line.split(/ and |;/)
return cmds.collect(this.&processInput) }
String[] parts = line.split(' ', 2) String[] parts = line.split(' ', 2)
String command = parts[0]?.toLowerCase() String command = parts[0]?.toLowerCase()
String rest = parts.size() == 2 ? parts[1]?.trim() : null String rest = parts.size() == 2 ? parts[1]?.trim() : null
@ -491,6 +510,7 @@ Configuration:
Class modelClass Class modelClass
switch (options) { switch (options) {
case ~/selection/: return selection;
case ~/playing ($selectableModels)s?/: case ~/playing ($selectableModels)s?/:
if (!curMediaFile) err "No media is currently playing." if (!curMediaFile) err "No media is currently playing."
@ -501,10 +521,29 @@ Configuration:
else return library.getWhere(modelClass, else return library.getWhere(modelClass,
[mediaFileId: curMediaFile.id]) [mediaFileId: curMediaFile.id])
/* TODO case ~/(\d+ )?random ($selectableModels)s?( from (.+$))?/:
case ~/files tagged as((\s\w+)+?) and not as((\s\w+)+)/: modelClass = modelClasses[Matcher.lastMatcher[0][2]]
excludedTags = lastMatcher[0][3].split(/\s/) def sourceCriteria = Matcher.lastMatcher[0][4]?.trim()
*/ int count = (Matcher.lastMatcher[0][1] ?: 1) as int
def source
if (sourceCriteria) {
source = select(sourceCriteria, selection)
if (modelClass != source[0].class) {
source = source.collectMany { library.getWhere(modelClass,
[(idKeyFor(source[0].class)): it.id]) }
.findAll().unique() } }
else source = library.getAll(modelClass);
if (source.size() < count)
err "There are not ${count} ${toEnglish(modelClass)}s to select."
def selected = []
(0..<count).each {
selected << source.remove(rand.nextInt(source.size())) }
return selected
case ~/selected ($selectableModels)s?/: case ~/selected ($selectableModels)s?/:
@ -518,6 +557,7 @@ Configuration:
[(idKeyFor(selectionClass)): it.id]) }.findAll() [(idKeyFor(selectionClass)): it.id]) }.findAll()
case ~/absent files/: case ~/absent files/:
scanMediaLibrary()
return library.getMediaFilesWhere(presentLocally: false) return library.getMediaFilesWhere(presentLocally: false)
case ~/files tagged( as){0,1}((\s[^\s]+)+)/: case ~/files tagged( as){0,1}((\s[^\s]+)+)/:
@ -531,6 +571,18 @@ Configuration:
// TODO // TODO
err "select <thing> where ... is not yet implemented." err "select <thing> where ... is not yet implemented."
case ~/($selectableModels)s? from (.+)/:
modelClass = modelClasses[Matcher.lastMatcher[0][1]]
def sourceCriteria = Matcher.lastMatcher[0][2].trim()
def models = select(sourceCriteria, selection)
if (modelClass != models[0].class) {
models = models.collectMany { library.getWhere(modelClass,
[(idKeyFor(models[0].class)): it.id]) }
.findAll().unique() }
return models;
case ~/($selectableModels)s((\s\d+)+)/: case ~/($selectableModels)s((\s\d+)+)/:
modelClass = modelClasses[Matcher.lastMatcher[0][1]] modelClass = modelClasses[Matcher.lastMatcher[0][1]]
return Matcher.lastMatcher[0][2].split(/\s/) return Matcher.lastMatcher[0][2].split(/\s/)
@ -561,12 +613,17 @@ Configuration:
switch (options) { switch (options) {
case ~/bookmark named (.+) on playlist (.+) at (.+)/: case ~/bookmark named (.+) on playlist (.+) at (.+)/:
Playlist p = getExactlyOne(Matcher.lastMatcher[0][2].trim()) Playlist p = getExactlyOne(
MediaFile mf = getExactlyOne(Matcher.lastMatcher[0][3].trim()) Playlist, Matcher.lastMatcher[0][2].trim())
MediaFile mf = getExactlyOne(
MediaFile, Matcher.lastMatcher[0][3].trim())
Bookmark b = new Bookmark(name: Matcher.lastMatcher[0][1].trim(), Bookmark b = new Bookmark(name: Matcher.lastMatcher[0][1].trim(),
playlistId: p.id, mediaFileId: mf.id) playlistId: p.id, mediaFileId: mf.id)
if (!p.userCreated) {
p.userCreated = true
p = library.save(p) }
b = library.save(b) b = library.save(b)
msg "New bookmark: ${b.id}: ${b.name}" msg "New bookmark: ${b.id}: ${b.name}"
return b return b
@ -577,33 +634,29 @@ Configuration:
b.name = Matcher.lastMatcher[0][1].trim() b.name = Matcher.lastMatcher[0][1].trim()
b.id = null; b.id = null;
Playlist p = library.getById(Playlist, b.playlistId)
p.userCreated = true
p = library.save(p)
b = library.save(b) b = library.save(b)
msg "New bookmark: ${b.id}: ${b.name}" msg "New bookmark: ${b.id}: ${b.name}"
return b return b
case ~/playlist named (.+) from (queue|selection|.+)/: case ~/playlist named (.+) from (queue|selection|.+)/:
Playlist p Playlist p = library.save(
if (Matcher.lastMatcher[0][2] == 'queue') { new Playlist( name: Matcher.lastMatcher[0][1].trim()))
p = playQueue.clone()
p.name = Matcher.lastMatcher[0][1].trim()
p.id = null
p = library.save(p) }
else {
p = new Playlist(name: Matcher.lastMatcher[0][1])
p = library.save(p)
if (Matcher.lastMatcher[0][2] != 'selection') if (Matcher.lastMatcher[0][2] != 'selection')
selection = select(Matcher.lastMatcher[0][2], selection) selection = select(Matcher.lastMatcher[0][2], selection)
library.addToPlaylist(p.id, library.addToPlaylist(p.id,
library.collectMediaFiles(selection).collect { it.id }) } library.collectMediaFiles(selection).collect { it.id })
msg "New playlist: ${p.id}: ${p.name}"
return p return p
case ~/playlist named (.+)/: case ~/playlist named (.+)/:
Playlist p = new Playlist( Playlist p = new Playlist(
name: Matcher.lastMatcher[0][1].trim(), name: Matcher.lastMatcher[0][1].trim())
userCreated: true)
p = library.save(p) p = library.save(p)
msg "New playlist: ${p.id}: ${p.name}" msg "New playlist: ${p.id}: ${p.name}"
@ -653,7 +706,7 @@ Configuration:
if (selection) { if (selection) {
List<MediaFile> mediaFiles = library.collectMediaFiles(selection) List<MediaFile> mediaFiles = library.collectMediaFiles(selection)
playQueue = library.removeAllFromPlaylist(playQueue.id) playQueue = library.removeAllFromPlaylist(playQueue.id)
library.addToPlaylist(playQueue.id, mediaFiles.collect { it.id }) playQueue = library.addToPlaylist(playQueue.id, mediaFiles.collect { it.id })
setPlayQueue(playQueue) } setPlayQueue(playQueue) }
vlcj.mediaListPlayer.play() vlcj.mediaListPlayer.play()
@ -927,10 +980,11 @@ Configuration:
private def processRepeat(String option) { private def processRepeat(String option) {
switch(option) { switch(option) {
case null: case null:
def mode = vlcj.mediaListPlayer.mode err 'Reading current repeat mode is not yet implemented.'
/*def mode = vlcj.mediaListPlayer.mode
msg("Repeat mode is: " + msg("Repeat mode is: " +
mode == MediaListPlayerMode.LOOP ? 'all.' : mode == MediaListPlayerMode.LOOP ? 'all.' :
(mode == MediaListPlayerMode.REPEAT ? 'one.' : 'off.')) (mode == MediaListPlayerMode.REPEAT ? 'one.' : 'off.'))*/
return return
case 'off': case 'off':
@ -1024,10 +1078,18 @@ ${cmdStyle}select playing { album | artist | file | playlist | tag }${normalStyl
this selects items that are associated with the currently playing media this selects items that are associated with the currently playing media
file. file.
${cmdStyle}queued { albums | artists | files | playlists | tags }${normalStyle} ${cmdStyle}select queued { albums | artists | files | playlists | tags }${normalStyle}
Select the items currently in the queue. Select the items currently in the queue.
${cmdStyle}select <count> random {albums | artists | files | playlists | tags }${normalStyle}
Select one or more items randomly.
${cmdStyle}select <count> random {albums | artists | files | playlists | tags } from <select-criteria>${normalStyle}
Make a selection, then select one or more items randomly from it.
${cmdStyle}select selected { album | artist | file | playlist | tag }${normalStyle} ${cmdStyle}select selected { album | artist | file | playlist | tag }${normalStyle}
Select the items associated with the current selection buffer into the Select the items associated with the current selection buffer into the
@ -1409,7 +1471,7 @@ Library Management:
""" """
default: default:
err "Unrecognized command: '$line'" err "Unrecognized command: '$options'"
drawLeader() drawLeader()
Thread.sleep(250) Thread.sleep(250)
break } } break } }
@ -1454,6 +1516,7 @@ Library Management:
p.lastUsed = new Timestamp(new Date().time) p.lastUsed = new Timestamp(new Date().time)
playQueue = library.update(p) playQueue = library.update(p)
vlcj.mediaListPlayer.stop()
vlcj.mediaList.clear() vlcj.mediaList.clear()
library.getMediaFilesWhere(playlistId: playQueue.id).each { library.getMediaFilesWhere(playlistId: playQueue.id).each {

View File

@ -12,7 +12,6 @@ buildscript {
apply plugin: 'java' apply plugin: 'java'
apply plugin: 'groovy' apply plugin: 'groovy'
apply plugin: 'ch.raffael.pegdown-doclet' apply plugin: 'ch.raffael.pegdown-doclet'
apply plugin: 'war'
dependencies { dependencies {
compile localGroovy() compile localGroovy()
@ -26,6 +25,6 @@ dependencies {
testCompile 'junit:junit:4.12' testCompile 'junit:junit:4.12'
runtime 'com.h2database:h2:1.4.185' runtime 'com.h2database:h2:1.4.192'
runtime 'org.postgresql:postgresql:9.4.1207.jre7' runtime 'org.postgresql:postgresql:9.4.1207.jre7'
} }

View File

@ -1,5 +0,0 @@
{
"driver": "postgres",
"sqlDir": "src/main/sql/migrations",
"connectionString": "host=localhost port=5432 dbname=wdiwtlt user=jdbernard password="
}

5
core/database.properties Normal file
View File

@ -0,0 +1,5 @@
dataSourceClassName=org.h2.jdbcx.JdbcDataSource
dataSource.url=jdbc:h2:/Users/jonathan/programs/wdiwtlt/db;DATABASE_TO_UPPER=FALSE;AUTO_SERVER=TRUE
dataSource.user=sa
dataSource.password=
migrations.dir=src/main/sql/migrations

View File

@ -12,10 +12,15 @@ public class ConfigWrapper {
public ConfigWrapper() { public ConfigWrapper() {
if (env.WDIWTLT_CONFIG_FILE) { if (env.WDIWTLT_CONFIG_FILE) {
if (tryLoadConfigFile(new File(env.WDIWTLT_CONFIG_FILE))) { if (tryLoadConfigFile(env.WDIWTLT_CONFIG_FILE)) {
return } } return } }
if (tryLoadConfigFile(new File(env.HOME, ".wdiwtlt.properties"))) return if (env.HOME &&
tryLoadConfigFile(new File(env.HOME, ".wdiwtlt.properties"))) return
String userHome = System.getProperty('user.home')
if (userHome &&
tryLoadConfigFile(new File(userHome, '.wdiwtlt.properties'))) return
try { try {
ConfigWrapper.getResourceAsStream("/com/jdbernard/wdiwtlt/db/default.properties") ConfigWrapper.getResourceAsStream("/com/jdbernard/wdiwtlt/db/default.properties")
@ -68,6 +73,10 @@ public class ConfigWrapper {
if (props) return new HikariConfig(props) if (props) return new HikariConfig(props)
else return null } else return null }
private boolean tryLoadConfigFile(String configFilePath) {
if (!configFilePath) return false
return tryLoadConfigFile(new File(configFilePath)) }
private boolean tryLoadConfigFile(File configFile) { private boolean tryLoadConfigFile(File configFile) {
if (!configFile.exists() || !configFile.isFile()) return false if (!configFile.exists() || !configFile.isFile()) return false

View File

@ -5,7 +5,7 @@ import java.util.Date;
import java.util.UUID; import java.util.UUID;
public class Playlist extends Model { public class Playlist extends Model {
public boolean userCreated = false; public boolean userCreated = true;
public String name; public String name;
public int mediaFileCount = 0; public int mediaFileCount = 0;
public UUID copiedFromId = null; public UUID copiedFromId = null;