1 package App::MPD::Feeder;
8 use App::MPD::Feeder::Options;
9 use App::MPD::Feeder::DB;
13 use IO::Async::Signal;
14 use IO::Async::Timer::Periodic;
15 use Log::Any qw($log);
18 use Syntax::Keyword::Try;
21 class App::MPD::Feeder {
22 has $cfg_file :reader;
25 has $db_needs_update :writer = 1;
29 use constant DEFAULT_CONFIG_FILE => '/etc/mpd-feeder/mpd-feeder.conf';
32 Getopt::Long::Configure('pass_through');
33 Getopt::Long::GetOptions('cfg|config=s' => \$cfg_file);
34 Getopt::Long::Configure('no_pass_through');
36 $cfg_file //= DEFAULT_CONFIG_FILE if -e DEFAULT_CONFIG_FILE;
40 $db_needs_update = 0 if $opt->skip_db_update;
44 my $new_opt = App::MPD::Feeder::Options->new;
46 $new_opt->parse_config_file($cfg_file) if $cfg_file;
48 $new_opt->parse_command_line;
50 Log::Any::Adapter->set( Stderr => log_level => $new_opt->log_level );
54 $db = App::MPD::Feeder::DB->new( opt => $opt );
60 my %conn = ( auto_connect => 1 );
61 $conn{host} = $opt->mpd_host if $opt->mpd_host;
62 $conn{port} = $opt->mpd_port if $opt->mpd_port;
64 $mpd = Net::Async::MPD->new(%conn);
68 die "Connection to MPD lost";
72 my $int_signal_handler = sub {
73 state $signal_count = 0;
75 $log->debug("Signal received. Stopping loop");
76 $mpd->loop->stop('quit');
78 if ( $signal_count > 1 ) {
79 $log->warn("Another signal received (#$signal_count)");
80 $log->warn("Exiting abruptly");
87 IO::Async::Signal->new(
89 on_receipt => $int_signal_handler,
95 IO::Async::Signal->new(
98 $log->debug("SIGHUP received. Scheduling reload");
99 $mpd->loop->stop('reload');
105 IO::Async::Signal->new(
108 $log->debug("SIGUSR1 received. Dumping configuration to STDERR");
109 my $old = select \*STDERR;
126 method update_db($force = undef) {
127 if (!$db_needs_update and !$force) {
128 $log->debug("Skipping DB update");
132 $log->info('Updating song database');
135 my $rows = $mpd->send('listallinfo')->get;
137 $log->trace('got all songs from MPD');
143 foreach my $entry (@$rows) {
144 next unless exists $entry->{file};
146 $self->db->store_song( $entry->{file},
147 $entry->{AlbumArtist} // $entry->{Artist},
153 my ($total_songs, $total_artists, $total_albums,
154 $new_songs, $new_artists, $new_albums
155 ) = $self->db->finish_update;
158 "Updated data about $song_count songs (including $new_songs new), "
159 . "$total_artists artists (including $new_artists new) "
161 . "and $total_albums albums (including $new_albums new)"
164 $db_needs_update = 0;
168 $self->db->cancel_update;
173 method queue_songs($num = undef, $callback = undef) {
177 $mpd->send('playlist')->on_done(
179 my $present = scalar @{ $_[0] // [] };
181 $log->notice( "Playlist contains $present songs. Wanted: "
182 . $opt->target_queue_length );
183 if ( $present < $opt->target_queue_length ) {
185 $opt->target_queue_length - $present, $callback );
188 $callback->() if $callback;
196 my @list = $self->db->find_suitable_songs($num);
198 die "Found no suitable songs" unless @list;
200 if ( @list < $num ) {
203 'Found only %d suitable songs instead of %d',
209 $log->info("About to add $num songs to the playlist");
212 for my $song (@list) {
213 my $path = $song->{song};
218 $log->debug( "Adding " . join( ', ', map {"«$_»"} @paths ) );
219 # MPD needs raw bytes
220 utf8::encode($_) for @paths;
223 push @commands, [ add => "\"$_\"" ];
226 my $f = $mpd->send( \@commands );
227 $f->on_fail( sub { die @_ } );
230 $self->db->note_song_qeued($_) for @list;
231 $callback->(@_) if $callback;
236 method prepare_to_wait_idle {
237 $log->trace('declaring idle mode');
238 $idler = $mpd->send('idle database playlist')->on_done(
244 my $changed = $result->{changed} // '';
246 if ( $changed eq 'database' ) {
247 $db_needs_update = 1;
248 $self->prepare_to_wait_idle;
250 elsif ( $changed eq 'playlist' ) {
251 $self->queue_songs( undef,
252 sub { $self->prepare_to_wait_idle } );
254 elsif ( $changed eq '' ) {
255 $log->debug("got no changes from idle");
256 $self->prepare_to_wait_idle;
261 "Unknown result from idle: " . to_json($result) );
262 $self->prepare_to_wait_idle;
279 IO::Async::Timer::Periodic->new(
283 $log->trace('breaking idle to see if MPD is there');
285 $log->trace("> noidle (direct)");
286 $mpd->{mpd_handle}->write("noidle\n");
293 $self->queue_songs( undef, sub { $self->prepare_to_wait_idle } );
295 $log->debug("Entering event loop. PID=$$");
297 my $result = $mpd->loop->run;
298 $log->trace( "Got loop result of " . ( $result // 'undef' ) );
300 if ( 'reload' eq $result ) {
301 $log->notice("disconnecting");
304 my @exec = ( $0, '--config', $self->cfg_file, '--skip-db-update' );
305 if ( $log->is_trace ) {
307 . join( ' ', map { /\s/ ? "'$_'" : $_ } @exec ) );
312 if ( 'quit' eq $result ) {
313 $log->trace("quitting because of 'quit' loop result");