Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
M
mpd
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Иван Мажукин
mpd
Commits
e939d667
Commit
e939d667
authored
Dec 18, 2015
by
Max Kellermann
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
protocol/Ack: add exception class wrapping enum ack
parent
8bb5a565
Hide whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
286 additions
and
374 deletions
+286
-374
CommandError.cxx
src/command/CommandError.cxx
+2
-0
DatabaseCommands.cxx
src/command/DatabaseCommands.cxx
+1
-2
OtherCommands.cxx
src/command/OtherCommands.cxx
+2
-6
OutputCommands.cxx
src/command/OutputCommands.cxx
+3
-9
PlayerCommands.cxx
src/command/PlayerCommands.cxx
+25
-64
PlaylistCommands.cxx
src/command/PlaylistCommands.cxx
+4
-9
QueueCommands.cxx
src/command/QueueCommands.cxx
+34
-83
Request.hxx
src/command/Request.hxx
+61
-18
TagCommands.cxx
src/command/TagCommands.cxx
+2
-6
Ack.hxx
src/protocol/Ack.hxx
+25
-0
ArgParser.cxx
src/protocol/ArgParser.cxx
+87
-131
ArgParser.hxx
src/protocol/ArgParser.hxx
+30
-21
test_protocol.cxx
test/test_protocol.cxx
+10
-25
No files found.
src/command/CommandError.cxx
View file @
e939d667
...
@@ -165,6 +165,8 @@ PrintError(Response &r, const std::exception &e)
...
@@ -165,6 +165,8 @@ PrintError(Response &r, const std::exception &e)
try
{
try
{
throw
e
;
throw
e
;
}
catch
(
const
ProtocolError
&
pe
)
{
r
.
Error
(
pe
.
GetCode
(),
pe
.
what
());
}
catch
(
const
PlaylistError
&
pe
)
{
}
catch
(
const
PlaylistError
&
pe
)
{
r
.
Error
(
ToAck
(
pe
.
GetCode
()),
pe
.
what
());
r
.
Error
(
ToAck
(
pe
.
GetCode
()),
pe
.
what
());
}
catch
(
const
std
::
system_error
&
)
{
}
catch
(
const
std
::
system_error
&
)
{
...
...
src/command/DatabaseCommands.cxx
View file @
e939d667
...
@@ -69,8 +69,7 @@ handle_match(Client &client, Request args, Response &r, bool fold_case)
...
@@ -69,8 +69,7 @@ handle_match(Client &client, Request args, Response &r, bool fold_case)
{
{
RangeArg
window
;
RangeArg
window
;
if
(
args
.
size
>=
2
&&
StringIsEqual
(
args
[
args
.
size
-
2
],
"window"
))
{
if
(
args
.
size
>=
2
&&
StringIsEqual
(
args
[
args
.
size
-
2
],
"window"
))
{
if
(
!
args
.
Parse
(
args
.
size
-
1
,
window
,
r
))
window
=
args
.
ParseRange
(
args
.
size
-
1
);
return
CommandResult
::
ERROR
;
args
.
pop_back
();
args
.
pop_back
();
args
.
pop_back
();
args
.
pop_back
();
...
...
src/command/OtherCommands.cxx
View file @
e939d667
...
@@ -352,9 +352,7 @@ handle_rescan(Client &client, Request args, Response &r)
...
@@ -352,9 +352,7 @@ handle_rescan(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_setvol
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_setvol
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
level
;
unsigned
level
=
args
.
ParseUnsigned
(
0
,
100
);
if
(
!
args
.
Parse
(
0
,
level
,
r
,
100
))
return
CommandResult
::
ERROR
;
if
(
!
volume_level_change
(
client
.
partition
.
outputs
,
level
))
{
if
(
!
volume_level_change
(
client
.
partition
.
outputs
,
level
))
{
r
.
Error
(
ACK_ERROR_SYSTEM
,
"problems setting volume"
);
r
.
Error
(
ACK_ERROR_SYSTEM
,
"problems setting volume"
);
...
@@ -367,9 +365,7 @@ handle_setvol(Client &client, Request args, Response &r)
...
@@ -367,9 +365,7 @@ handle_setvol(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_volume
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_volume
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
int
relative
;
int
relative
=
args
.
ParseInt
(
0
,
-
100
,
100
);
if
(
!
args
.
Parse
(
0
,
relative
,
r
,
-
100
,
100
))
return
CommandResult
::
ERROR
;
const
int
old_volume
=
volume_level_get
(
client
.
partition
.
outputs
);
const
int
old_volume
=
volume_level_get
(
client
.
partition
.
outputs
);
if
(
old_volume
<
0
)
{
if
(
old_volume
<
0
)
{
...
...
src/command/OutputCommands.cxx
View file @
e939d667
...
@@ -31,9 +31,7 @@ CommandResult
...
@@ -31,9 +31,7 @@ CommandResult
handle_enableoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_enableoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
assert
(
args
.
size
==
1
);
assert
(
args
.
size
==
1
);
unsigned
device
;
unsigned
device
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
device
,
r
))
return
CommandResult
::
ERROR
;
if
(
!
audio_output_enable_index
(
client
.
partition
.
outputs
,
device
))
{
if
(
!
audio_output_enable_index
(
client
.
partition
.
outputs
,
device
))
{
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
...
@@ -47,9 +45,7 @@ CommandResult
...
@@ -47,9 +45,7 @@ CommandResult
handle_disableoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_disableoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
assert
(
args
.
size
==
1
);
assert
(
args
.
size
==
1
);
unsigned
device
;
unsigned
device
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
device
,
r
))
return
CommandResult
::
ERROR
;
if
(
!
audio_output_disable_index
(
client
.
partition
.
outputs
,
device
))
{
if
(
!
audio_output_disable_index
(
client
.
partition
.
outputs
,
device
))
{
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
...
@@ -63,9 +59,7 @@ CommandResult
...
@@ -63,9 +59,7 @@ CommandResult
handle_toggleoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_toggleoutput
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
assert
(
args
.
size
==
1
);
assert
(
args
.
size
==
1
);
unsigned
device
;
unsigned
device
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
device
,
r
))
return
CommandResult
::
ERROR
;
if
(
!
audio_output_toggle_index
(
client
.
partition
.
outputs
,
device
))
{
if
(
!
audio_output_toggle_index
(
client
.
partition
.
outputs
,
device
))
{
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
r
.
Error
(
ACK_ERROR_NO_EXIST
,
"No such audio output"
);
...
...
src/command/PlayerCommands.cxx
View file @
e939d667
...
@@ -57,23 +57,17 @@
...
@@ -57,23 +57,17 @@
#define COMMAND_STATUS_UPDATING_DB "updating_db"
#define COMMAND_STATUS_UPDATING_DB "updating_db"
CommandResult
CommandResult
handle_play
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_play
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
int
song
=
-
1
;
int
song
=
args
.
ParseOptional
(
0
,
-
1
);
if
(
!
args
.
ParseOptional
(
0
,
song
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
PlayPosition
(
song
);
client
.
partition
.
PlayPosition
(
song
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_playid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_playid
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
int
id
=
-
1
;
int
id
=
args
.
ParseOptional
(
0
,
-
1
);
if
(
!
args
.
ParseOptional
(
0
,
id
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
PlayId
(
id
);
client
.
partition
.
PlayId
(
id
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
...
@@ -93,13 +87,10 @@ handle_currentsong(Client &client, gcc_unused Request args, Response &r)
...
@@ -93,13 +87,10 @@ handle_currentsong(Client &client, gcc_unused Request args, Response &r)
}
}
CommandResult
CommandResult
handle_pause
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_pause
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
if
(
!
args
.
IsEmpty
())
{
if
(
!
args
.
IsEmpty
())
{
bool
pause_flag
;
bool
pause_flag
=
args
.
ParseBool
(
0
);
if
(
!
args
.
Parse
(
0
,
pause_flag
,
r
))
return
CommandResult
::
ERROR
;
client
.
player_control
.
LockSetPause
(
pause_flag
);
client
.
player_control
.
LockSetPause
(
pause_flag
);
}
else
}
else
client
.
player_control
.
LockPause
();
client
.
player_control
.
LockPause
();
...
@@ -236,45 +227,33 @@ handle_previous(Client &client, gcc_unused Request args,
...
@@ -236,45 +227,33 @@ handle_previous(Client &client, gcc_unused Request args,
}
}
CommandResult
CommandResult
handle_repeat
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_repeat
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
bool
status
;
bool
status
=
args
.
ParseBool
(
0
);
if
(
!
args
.
Parse
(
0
,
status
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetRepeat
(
status
);
client
.
partition
.
SetRepeat
(
status
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_single
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_single
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
bool
status
;
bool
status
=
args
.
ParseBool
(
0
);
if
(
!
args
.
Parse
(
0
,
status
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetSingle
(
status
);
client
.
partition
.
SetSingle
(
status
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_consume
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_consume
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
bool
status
;
bool
status
=
args
.
ParseBool
(
0
);
if
(
!
args
.
Parse
(
0
,
status
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetConsume
(
status
);
client
.
partition
.
SetConsume
(
status
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_random
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_random
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
bool
status
;
bool
status
=
args
.
ParseBool
(
0
);
if
(
!
args
.
Parse
(
0
,
status
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetRandom
(
status
);
client
.
partition
.
SetRandom
(
status
);
client
.
partition
.
outputs
.
SetReplayGainMode
(
replay_gain_get_real_mode
(
client
.
partition
.
GetRandom
()));
client
.
partition
.
outputs
.
SetReplayGainMode
(
replay_gain_get_real_mode
(
client
.
partition
.
GetRandom
()));
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
...
@@ -291,10 +270,8 @@ handle_clearerror(Client &client, gcc_unused Request args,
...
@@ -291,10 +270,8 @@ handle_clearerror(Client &client, gcc_unused Request args,
CommandResult
CommandResult
handle_seek
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_seek
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
song
;
unsigned
song
=
args
.
ParseUnsigned
(
0
);
SongTime
seek_time
;
SongTime
seek_time
=
args
.
ParseSongTime
(
1
);
if
(
!
args
.
Parse
(
0
,
song
,
r
)
||
!
args
.
Parse
(
1
,
seek_time
,
r
))
return
CommandResult
::
ERROR
;
Error
error
;
Error
error
;
return
client
.
partition
.
SeekSongPosition
(
song
,
seek_time
,
error
)
return
client
.
partition
.
SeekSongPosition
(
song
,
seek_time
,
error
)
...
@@ -305,12 +282,8 @@ handle_seek(Client &client, Request args, Response &r)
...
@@ -305,12 +282,8 @@ handle_seek(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_seekid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_seekid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
id
;
unsigned
id
=
args
.
ParseUnsigned
(
0
);
SongTime
seek_time
;
SongTime
seek_time
=
args
.
ParseSongTime
(
1
);
if
(
!
args
.
Parse
(
0
,
id
,
r
))
return
CommandResult
::
ERROR
;
if
(
!
args
.
Parse
(
1
,
seek_time
,
r
))
return
CommandResult
::
ERROR
;
Error
error
;
Error
error
;
return
client
.
partition
.
SeekSongId
(
id
,
seek_time
,
error
)
return
client
.
partition
.
SeekSongId
(
id
,
seek_time
,
error
)
...
@@ -323,9 +296,7 @@ handle_seekcur(Client &client, Request args, Response &r)
...
@@ -323,9 +296,7 @@ handle_seekcur(Client &client, Request args, Response &r)
{
{
const
char
*
p
=
args
.
front
();
const
char
*
p
=
args
.
front
();
bool
relative
=
*
p
==
'+'
||
*
p
==
'-'
;
bool
relative
=
*
p
==
'+'
||
*
p
==
'-'
;
SignedSongTime
seek_time
;
SignedSongTime
seek_time
=
ParseCommandArgSignedSongTime
(
p
);
if
(
!
ParseCommandArg
(
r
,
seek_time
,
p
))
return
CommandResult
::
ERROR
;
Error
error
;
Error
error
;
return
client
.
partition
.
SeekCurrent
(
seek_time
,
relative
,
error
)
return
client
.
partition
.
SeekCurrent
(
seek_time
,
relative
,
error
)
...
@@ -334,36 +305,26 @@ handle_seekcur(Client &client, Request args, Response &r)
...
@@ -334,36 +305,26 @@ handle_seekcur(Client &client, Request args, Response &r)
}
}
CommandResult
CommandResult
handle_crossfade
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_crossfade
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
xfade_time
;
unsigned
xfade_time
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
xfade_time
,
r
))
return
CommandResult
::
ERROR
;
client
.
player_control
.
SetCrossFade
(
xfade_time
);
client
.
player_control
.
SetCrossFade
(
xfade_time
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_mixrampdb
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_mixrampdb
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
float
db
;
float
db
=
args
.
ParseFloat
(
0
);
if
(
!
args
.
Parse
(
0
,
db
,
r
))
return
CommandResult
::
ERROR
;
client
.
player_control
.
SetMixRampDb
(
db
);
client
.
player_control
.
SetMixRampDb
(
db
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_mixrampdelay
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_mixrampdelay
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
float
delay_secs
;
float
delay_secs
=
args
.
ParseFloat
(
0
);
if
(
!
args
.
Parse
(
0
,
delay_secs
,
r
))
return
CommandResult
::
ERROR
;
client
.
player_control
.
SetMixRampDelay
(
delay_secs
);
client
.
player_control
.
SetMixRampDelay
(
delay_secs
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
...
...
src/command/PlaylistCommands.cxx
View file @
e939d667
...
@@ -70,9 +70,7 @@ handle_save(Client &client, Request args, Response &r)
...
@@ -70,9 +70,7 @@ handle_save(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_load
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_load
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
RangeArg
range
=
RangeArg
::
All
();
RangeArg
range
=
args
.
ParseOptional
(
1
,
RangeArg
::
All
());
if
(
!
args
.
ParseOptional
(
1
,
range
,
r
))
return
CommandResult
::
ERROR
;
const
ScopeBulkEdit
bulk_edit
(
client
.
partition
);
const
ScopeBulkEdit
bulk_edit
(
client
.
partition
);
...
@@ -144,9 +142,7 @@ CommandResult
...
@@ -144,9 +142,7 @@ CommandResult
handle_playlistdelete
(
gcc_unused
Client
&
client
,
Request
args
,
Response
&
r
)
handle_playlistdelete
(
gcc_unused
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
const
char
*
const
name
=
args
[
0
];
const
char
*
const
name
=
args
[
0
];
unsigned
from
;
unsigned
from
=
args
.
ParseUnsigned
(
1
);
if
(
!
args
.
Parse
(
1
,
from
,
r
))
return
CommandResult
::
ERROR
;
Error
error
;
Error
error
;
return
spl_remove_index
(
name
,
from
,
error
)
return
spl_remove_index
(
name
,
from
,
error
)
...
@@ -158,9 +154,8 @@ CommandResult
...
@@ -158,9 +154,8 @@ CommandResult
handle_playlistmove
(
gcc_unused
Client
&
client
,
Request
args
,
Response
&
r
)
handle_playlistmove
(
gcc_unused
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
const
char
*
const
name
=
args
.
front
();
const
char
*
const
name
=
args
.
front
();
unsigned
from
,
to
;
unsigned
from
=
args
.
ParseUnsigned
(
1
);
if
(
!
args
.
Parse
(
1
,
from
,
r
)
||
!
args
.
Parse
(
2
,
to
,
r
))
unsigned
to
=
args
.
ParseUnsigned
(
2
);
return
CommandResult
::
ERROR
;
Error
error
;
Error
error
;
return
spl_move_index
(
name
,
from
,
to
,
error
)
return
spl_move_index
(
name
,
from
,
to
,
error
)
...
...
src/command/QueueCommands.cxx
View file @
e939d667
...
@@ -128,9 +128,7 @@ handle_addid(Client &client, Request args, Response &r)
...
@@ -128,9 +128,7 @@ handle_addid(Client &client, Request args, Response &r)
return
print_error
(
r
,
error
);
return
print_error
(
r
,
error
);
if
(
args
.
size
==
2
)
{
if
(
args
.
size
==
2
)
{
unsigned
to
;
unsigned
to
=
args
.
ParseUnsigned
(
1
);
if
(
!
args
.
Parse
(
1
,
to
,
r
))
return
CommandResult
::
ERROR
;
try
{
try
{
client
.
partition
.
MoveId
(
added_id
,
to
);
client
.
partition
.
MoveId
(
added_id
,
to
);
...
@@ -180,9 +178,7 @@ parse_time_range(const char *p, SongTime &start_r, SongTime &end_r)
...
@@ -180,9 +178,7 @@ parse_time_range(const char *p, SongTime &start_r, SongTime &end_r)
CommandResult
CommandResult
handle_rangeid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_rangeid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
id
;
unsigned
id
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
id
,
r
))
return
CommandResult
::
ERROR
;
SongTime
start
,
end
;
SongTime
start
,
end
;
if
(
!
parse_time_range
(
args
[
1
],
start
,
end
))
{
if
(
!
parse_time_range
(
args
[
1
],
start
,
end
))
{
...
@@ -200,23 +196,17 @@ handle_rangeid(Client &client, Request args, Response &r)
...
@@ -200,23 +196,17 @@ handle_rangeid(Client &client, Request args, Response &r)
}
}
CommandResult
CommandResult
handle_delete
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_delete
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
RangeArg
range
;
RangeArg
range
=
args
.
ParseRange
(
0
);
if
(
!
args
.
Parse
(
0
,
range
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
DeleteRange
(
range
.
start
,
range
.
end
);
client
.
partition
.
DeleteRange
(
range
.
start
,
range
.
end
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_deleteid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_deleteid
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
id
;
unsigned
id
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
id
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
DeleteId
(
id
);
client
.
partition
.
DeleteId
(
id
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
...
@@ -229,12 +219,9 @@ handle_playlist(Client &client, gcc_unused Request args, Response &r)
...
@@ -229,12 +219,9 @@ handle_playlist(Client &client, gcc_unused Request args, Response &r)
}
}
CommandResult
CommandResult
handle_shuffle
(
gcc_unused
Client
&
client
,
Request
args
,
Response
&
r
)
handle_shuffle
(
gcc_unused
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
RangeArg
range
=
RangeArg
::
All
();
RangeArg
range
=
args
.
ParseOptional
(
0
,
RangeArg
::
All
());
if
(
!
args
.
ParseOptional
(
0
,
range
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
Shuffle
(
range
.
start
,
range
.
end
);
client
.
partition
.
Shuffle
(
range
.
start
,
range
.
end
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
...
@@ -249,14 +236,8 @@ handle_clear(Client &client, gcc_unused Request args, gcc_unused Response &r)
...
@@ -249,14 +236,8 @@ handle_clear(Client &client, gcc_unused Request args, gcc_unused Response &r)
CommandResult
CommandResult
handle_plchanges
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_plchanges
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
uint32_t
version
;
uint32_t
version
=
ParseCommandArgU32
(
args
.
front
());
if
(
!
ParseCommandArg32
(
r
,
version
,
args
.
front
()))
RangeArg
range
=
args
.
ParseOptional
(
1
,
RangeArg
::
All
());
return
CommandResult
::
ERROR
;
RangeArg
range
=
RangeArg
::
All
();
if
(
!
args
.
ParseOptional
(
1
,
range
,
r
))
return
CommandResult
::
ERROR
;
playlist_print_changes_info
(
r
,
client
.
partition
,
playlist_print_changes_info
(
r
,
client
.
partition
,
client
.
playlist
,
version
,
client
.
playlist
,
version
,
range
.
start
,
range
.
end
);
range
.
start
,
range
.
end
);
...
@@ -266,14 +247,8 @@ handle_plchanges(Client &client, Request args, Response &r)
...
@@ -266,14 +247,8 @@ handle_plchanges(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_plchangesposid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_plchangesposid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
uint32_t
version
;
uint32_t
version
=
ParseCommandArgU32
(
args
.
front
());
if
(
!
ParseCommandArg32
(
r
,
version
,
args
.
front
()))
RangeArg
range
=
args
.
ParseOptional
(
1
,
RangeArg
::
All
());
return
CommandResult
::
ERROR
;
RangeArg
range
=
RangeArg
::
All
();
if
(
!
args
.
ParseOptional
(
1
,
range
,
r
))
return
CommandResult
::
ERROR
;
playlist_print_changes_position
(
r
,
client
.
playlist
,
version
,
playlist_print_changes_position
(
r
,
client
.
playlist
,
version
,
range
.
start
,
range
.
end
);
range
.
start
,
range
.
end
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
...
@@ -282,9 +257,7 @@ handle_plchangesposid(Client &client, Request args, Response &r)
...
@@ -282,9 +257,7 @@ handle_plchangesposid(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_playlistinfo
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_playlistinfo
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
RangeArg
range
=
RangeArg
::
All
();
RangeArg
range
=
args
.
ParseOptional
(
0
,
RangeArg
::
All
());
if
(
!
args
.
ParseOptional
(
0
,
range
,
r
))
return
CommandResult
::
ERROR
;
if
(
!
playlist_print_info
(
r
,
client
.
partition
,
client
.
playlist
,
if
(
!
playlist_print_info
(
r
,
client
.
partition
,
client
.
playlist
,
range
.
start
,
range
.
end
))
range
.
start
,
range
.
end
))
...
@@ -298,10 +271,7 @@ CommandResult
...
@@ -298,10 +271,7 @@ CommandResult
handle_playlistid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_playlistid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
if
(
!
args
.
IsEmpty
())
{
if
(
!
args
.
IsEmpty
())
{
unsigned
id
;
unsigned
id
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
id
,
r
))
return
CommandResult
::
ERROR
;
bool
ret
=
playlist_print_id
(
r
,
client
.
partition
,
bool
ret
=
playlist_print_id
(
r
,
client
.
partition
,
client
.
playlist
,
id
);
client
.
playlist
,
id
);
if
(
!
ret
)
if
(
!
ret
)
...
@@ -341,17 +311,13 @@ handle_playlistsearch(Client &client, Request args, Response &r)
...
@@ -341,17 +311,13 @@ handle_playlistsearch(Client &client, Request args, Response &r)
}
}
CommandResult
CommandResult
handle_prio
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_prio
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
priority
;
unsigned
priority
=
args
.
ParseUnsigned
(
0
,
0xff
);
if
(
!
args
.
ParseShift
(
0
,
priority
,
r
,
0xff
))
args
.
shift
();
return
CommandResult
::
ERROR
;
for
(
const
char
*
i
:
args
)
{
for
(
const
char
*
i
:
args
)
{
RangeArg
range
;
RangeArg
range
=
ParseCommandArgRange
(
i
);
if
(
!
ParseCommandArg
(
r
,
range
,
i
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetPriorityRange
(
range
.
start
,
range
.
end
,
client
.
partition
.
SetPriorityRange
(
range
.
start
,
range
.
end
,
priority
);
priority
);
}
}
...
@@ -360,17 +326,13 @@ handle_prio(Client &client, Request args, Response &r)
...
@@ -360,17 +326,13 @@ handle_prio(Client &client, Request args, Response &r)
}
}
CommandResult
CommandResult
handle_prioid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_prioid
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
priority
;
unsigned
priority
=
args
.
ParseUnsigned
(
0
,
0xff
);
if
(
!
args
.
ParseShift
(
0
,
priority
,
r
,
0xff
))
args
.
shift
();
return
CommandResult
::
ERROR
;
for
(
const
char
*
i
:
args
)
{
for
(
const
char
*
i
:
args
)
{
unsigned
song_id
;
unsigned
song_id
=
ParseCommandArgUnsigned
(
i
);
if
(
!
ParseCommandArg
(
r
,
song_id
,
i
))
return
CommandResult
::
ERROR
;
client
.
partition
.
SetPriorityId
(
song_id
,
priority
);
client
.
partition
.
SetPriorityId
(
song_id
,
priority
);
}
}
...
@@ -378,48 +340,37 @@ handle_prioid(Client &client, Request args, Response &r)
...
@@ -378,48 +340,37 @@ handle_prioid(Client &client, Request args, Response &r)
}
}
CommandResult
CommandResult
handle_move
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_move
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
RangeArg
range
;
RangeArg
range
=
args
.
ParseRange
(
0
);
int
to
;
int
to
=
args
.
ParseInt
(
1
);
if
(
!
args
.
Parse
(
0
,
range
,
r
)
||
!
args
.
Parse
(
1
,
to
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
MoveRange
(
range
.
start
,
range
.
end
,
to
);
client
.
partition
.
MoveRange
(
range
.
start
,
range
.
end
,
to
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_moveid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_moveid
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
id
;
unsigned
id
=
args
.
ParseUnsigned
(
0
);
int
to
;
int
to
=
args
.
ParseInt
(
1
);
if
(
!
args
.
Parse
(
0
,
id
,
r
)
||
!
args
.
Parse
(
1
,
to
,
r
))
return
CommandResult
::
ERROR
;
client
.
partition
.
MoveId
(
id
,
to
);
client
.
partition
.
MoveId
(
id
,
to
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_swap
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_swap
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
song1
,
song2
;
unsigned
song1
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
song1
,
r
)
||
!
args
.
Parse
(
1
,
song2
,
r
))
unsigned
song2
=
args
.
ParseUnsigned
(
1
);
return
CommandResult
::
ERROR
;
client
.
partition
.
SwapPositions
(
song1
,
song2
);
client
.
partition
.
SwapPositions
(
song1
,
song2
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
CommandResult
CommandResult
handle_swapid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_swapid
(
Client
&
client
,
Request
args
,
gcc_unused
Response
&
r
)
{
{
unsigned
id1
,
id2
;
unsigned
id1
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
id1
,
r
)
||
!
args
.
Parse
(
1
,
id2
,
r
))
unsigned
id2
=
args
.
ParseUnsigned
(
1
);
return
CommandResult
::
ERROR
;
client
.
partition
.
SwapIds
(
id1
,
id2
);
client
.
partition
.
SwapIds
(
id1
,
id2
);
return
CommandResult
::
OK
;
return
CommandResult
::
OK
;
}
}
src/command/Request.hxx
View file @
e939d667
...
@@ -22,6 +22,7 @@
...
@@ -22,6 +22,7 @@
#include "check.h"
#include "check.h"
#include "protocol/ArgParser.hxx"
#include "protocol/ArgParser.hxx"
#include "Chrono.hxx"
#include "util/ConstBuffer.hxx"
#include "util/ConstBuffer.hxx"
#include <utility>
#include <utility>
...
@@ -44,30 +45,72 @@ public:
...
@@ -44,30 +45,72 @@ public:
:
default_value
;
:
default_value
;
}
}
template
<
typename
T
,
typename
...
Args
>
gcc_pure
bool
Parse
(
unsigned
idx
,
T
&
value_r
,
Response
&
r
,
int
ParseInt
(
unsigned
idx
)
const
{
Args
&&
...
args
)
{
assert
(
idx
<
size
);
assert
(
idx
<
size
);
return
ParseCommandArgInt
(
data
[
idx
]);
}
return
ParseCommandArg
(
r
,
value_r
,
data
[
idx
],
gcc_pure
std
::
forward
<
Args
>
(
args
)...);
int
ParseInt
(
unsigned
idx
,
int
min_value
,
int
max_value
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgInt
(
data
[
idx
],
min_value
,
max_value
);
}
}
template
<
typename
T
,
typename
...
Args
>
gcc_pure
bool
ParseOptional
(
unsigned
idx
,
T
&
value_r
,
Response
&
r
,
int
ParseUnsigned
(
unsigned
idx
)
const
{
Args
&&
...
args
)
{
assert
(
idx
<
size
);
return
idx
>=
size
||
return
ParseCommandArgUnsigned
(
data
[
idx
]);
Parse
(
idx
,
value_r
,
r
,
std
::
forward
<
Args
>
(
args
)...);
}
}
template
<
typename
T
,
typename
...
Args
>
gcc_pure
bool
ParseShift
(
unsigned
idx
,
T
&
value_r
,
Response
&
r
,
int
ParseUnsigned
(
unsigned
idx
,
unsigned
max_value
)
const
{
Args
&&
...
args
)
{
assert
(
idx
<
size
);
bool
success
=
Parse
(
idx
,
value_r
,
r
,
return
ParseCommandArgUnsigned
(
data
[
idx
],
max_value
);
std
::
forward
<
Args
>
(
args
)...);
}
shift
();
return
success
;
gcc_pure
bool
ParseBool
(
unsigned
idx
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgBool
(
data
[
idx
]);
}
gcc_pure
RangeArg
ParseRange
(
unsigned
idx
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgRange
(
data
[
idx
]);
}
gcc_pure
float
ParseFloat
(
unsigned
idx
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgFloat
(
data
[
idx
]);
}
gcc_pure
SongTime
ParseSongTime
(
unsigned
idx
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgSongTime
(
data
[
idx
]);
}
gcc_pure
SignedSongTime
ParseSignedSongTime
(
unsigned
idx
)
const
{
assert
(
idx
<
size
);
return
ParseCommandArgSignedSongTime
(
data
[
idx
]);
}
gcc_pure
int
ParseOptional
(
unsigned
idx
,
int
default_value
)
const
{
return
idx
<
size
?
ParseInt
(
idx
)
:
default_value
;
}
gcc_pure
RangeArg
ParseOptional
(
unsigned
idx
,
RangeArg
default_value
)
const
{
return
idx
<
size
?
ParseRange
(
idx
)
:
default_value
;
}
}
};
};
...
...
src/command/TagCommands.cxx
View file @
e939d667
...
@@ -30,9 +30,7 @@
...
@@ -30,9 +30,7 @@
CommandResult
CommandResult
handle_addtagid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_addtagid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
song_id
;
unsigned
song_id
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
song_id
,
r
))
return
CommandResult
::
ERROR
;
const
char
*
const
tag_name
=
args
[
1
];
const
char
*
const
tag_name
=
args
[
1
];
const
TagType
tag_type
=
tag_name_parse_i
(
tag_name
);
const
TagType
tag_type
=
tag_name_parse_i
(
tag_name
);
...
@@ -54,9 +52,7 @@ handle_addtagid(Client &client, Request args, Response &r)
...
@@ -54,9 +52,7 @@ handle_addtagid(Client &client, Request args, Response &r)
CommandResult
CommandResult
handle_cleartagid
(
Client
&
client
,
Request
args
,
Response
&
r
)
handle_cleartagid
(
Client
&
client
,
Request
args
,
Response
&
r
)
{
{
unsigned
song_id
;
unsigned
song_id
=
args
.
ParseUnsigned
(
0
);
if
(
!
args
.
Parse
(
0
,
song_id
,
r
))
return
CommandResult
::
ERROR
;
TagType
tag_type
=
TAG_NUM_OF_ITEM_TYPES
;
TagType
tag_type
=
TAG_NUM_OF_ITEM_TYPES
;
if
(
args
.
size
>=
2
)
{
if
(
args
.
size
>=
2
)
{
...
...
src/protocol/Ack.hxx
View file @
e939d667
...
@@ -20,6 +20,10 @@
...
@@ -20,6 +20,10 @@
#ifndef MPD_ACK_H
#ifndef MPD_ACK_H
#define MPD_ACK_H
#define MPD_ACK_H
#include <stdexcept>
#include <stdio.h>
class
Domain
;
class
Domain
;
enum
ack
{
enum
ack
{
...
@@ -40,4 +44,25 @@ enum ack {
...
@@ -40,4 +44,25 @@ enum ack {
extern
const
Domain
ack_domain
;
extern
const
Domain
ack_domain
;
class
ProtocolError
:
public
std
::
runtime_error
{
enum
ack
code
;
public
:
ProtocolError
(
enum
ack
_code
,
const
char
*
msg
)
:
std
::
runtime_error
(
msg
),
code
(
_code
)
{}
enum
ack
GetCode
()
const
{
return
code
;
}
};
template
<
typename
...
Args
>
static
inline
ProtocolError
FormatProtocolError
(
enum
ack
code
,
const
char
*
fmt
,
Args
&&
...
args
)
noexcept
{
char
buffer
[
256
];
snprintf
(
buffer
,
sizeof
(
buffer
),
fmt
,
std
::
forward
<
Args
>
(
args
)...);
return
ProtocolError
(
code
,
buffer
);
}
#endif
#endif
src/protocol/ArgParser.cxx
View file @
e939d667
...
@@ -19,199 +19,155 @@
...
@@ -19,199 +19,155 @@
#include "config.h"
#include "config.h"
#include "ArgParser.hxx"
#include "ArgParser.hxx"
#include "Ack.hxx"
#include "Chrono.hxx"
#include "Chrono.hxx"
#include "client/Response.hxx"
#include <stdlib.h>
#include <stdlib.h>
bool
uint32_t
ParseCommandArg
32
(
Response
&
r
,
uint32_t
&
value_r
,
const
char
*
s
)
ParseCommandArg
U32
(
const
char
*
s
)
{
{
char
*
test
;
char
*
test
;
auto
value
=
strtoul
(
s
,
&
test
,
10
);
if
(
test
==
s
||
*
test
!=
'\0'
)
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
value_r
=
strtoul
(
s
,
&
test
,
10
);
return
value
;
if
(
test
==
s
||
*
test
!=
'\0'
)
{
r
.
FormatError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
return
false
;
}
return
true
;
}
}
bool
int
ParseCommandArg
(
Response
&
r
,
int
&
value_r
,
const
char
*
s
,
ParseCommandArgInt
(
const
char
*
s
,
int
min_value
,
int
max_value
)
int
min_value
,
int
max_value
)
{
{
char
*
test
;
char
*
test
;
long
value
;
auto
value
=
strtol
(
s
,
&
test
,
10
);
if
(
test
==
s
||
*
test
!=
'\0'
)
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
value
=
strtol
(
s
,
&
test
,
10
);
if
(
value
<
min_value
||
value
>
max_value
)
if
(
test
==
s
||
*
test
!=
'\0'
)
{
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
r
.
FormatError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
"Number too large: %s"
,
s
);
return
false
;
}
if
(
value
<
min_value
||
value
>
max_value
)
{
return
(
int
)
value
;
r
.
FormatError
(
ACK_ERROR_ARG
,
"Number too large: %s"
,
s
);
return
false
;
}
value_r
=
(
int
)
value
;
return
true
;
}
}
bool
int
ParseCommandArg
(
Response
&
r
,
int
&
value_r
,
const
char
*
s
)
ParseCommandArg
Int
(
const
char
*
s
)
{
{
return
ParseCommandArg
(
r
,
value_r
,
s
,
return
ParseCommandArg
Int
(
s
,
std
::
numeric_limits
<
int
>::
min
(),
std
::
numeric_limits
<
int
>::
min
(),
std
::
numeric_limits
<
int
>::
max
());
std
::
numeric_limits
<
int
>::
max
());
}
}
bool
RangeArg
ParseCommandArg
(
Response
&
r
,
RangeArg
&
value_r
,
const
char
*
s
)
ParseCommandArg
Range
(
const
char
*
s
)
{
{
char
*
test
,
*
test2
;
char
*
test
,
*
test2
;
long
value
;
auto
value
=
strtol
(
s
,
&
test
,
10
);
if
(
test
==
s
||
(
*
test
!=
'\0'
&&
*
test
!=
':'
))
value
=
strtol
(
s
,
&
test
,
10
);
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
if
(
test
==
s
||
(
*
test
!=
'\0'
&&
*
test
!=
':'
))
{
"Integer or range expected: %s"
,
s
);
r
.
FormatError
(
ACK_ERROR_ARG
,
"Integer or range expected: %s"
,
s
);
return
false
;
}
if
(
value
==
-
1
&&
*
test
==
0
)
{
if
(
value
==
-
1
&&
*
test
==
0
)
/* compatibility with older MPD versions: specifying
/* compatibility with older MPD versions: specifying
"-1" makes MPD display the whole list */
"-1" makes MPD display the whole list */
value_r
.
start
=
0
;
return
RangeArg
::
All
();
value_r
.
end
=
std
::
numeric_limits
<
int
>::
max
();
return
true
;
}
if
(
value
<
0
)
{
if
(
value
<
0
)
r
.
FormatError
(
ACK_ERROR_ARG
,
"Number is negative: %s"
,
s
);
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
return
false
;
"Number is negative: %s"
,
s
);
}
if
(
unsigned
(
value
)
>
std
::
numeric_limits
<
unsigned
>::
max
())
{
if
(
unsigned
(
value
)
>
std
::
numeric_limits
<
unsigned
>::
max
())
r
.
FormatError
(
ACK_ERROR_ARG
,
"Number too large: %s"
,
s
);
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
return
false
;
"Number too large: %s"
,
s
);
}
value_r
.
start
=
(
unsigned
)
value
;
RangeArg
range
;
range
.
start
=
(
unsigned
)
value
;
if
(
*
test
==
':'
)
{
if
(
*
test
==
':'
)
{
value
=
strtol
(
++
test
,
&
test2
,
10
);
value
=
strtol
(
++
test
,
&
test2
,
10
);
if
(
*
test2
!=
'\0'
)
{
if
(
*
test2
!=
'\0'
)
r
.
FormatError
(
ACK_ERROR_ARG
,
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Integer or range expected: %s"
,
s
);
"Integer or range expected: %s"
,
return
false
;
s
);
}
if
(
test
==
test2
)
if
(
test
==
test2
)
value
=
std
::
numeric_limits
<
int
>::
max
();
value
=
std
::
numeric_limits
<
int
>::
max
();
if
(
value
<
0
)
{
if
(
value
<
0
)
r
.
FormatError
(
ACK_ERROR_ARG
,
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Number is negative: %s"
,
s
);
"Number is negative: %s"
,
s
);
return
false
;
}
if
(
unsigned
(
value
)
>
std
::
numeric_limits
<
unsigned
>::
max
())
{
if
(
unsigned
(
value
)
>
std
::
numeric_limits
<
unsigned
>::
max
())
r
.
FormatError
(
ACK_ERROR_ARG
,
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Number too large: %s"
,
s
);
"Number too large: %s"
,
s
);
return
false
;
}
value_r
.
end
=
(
unsigned
)
value
;
range
.
end
=
(
unsigned
)
value
;
}
else
{
}
else
{
value_r
.
end
=
(
unsigned
)
value
+
1
;
range
.
end
=
(
unsigned
)
value
+
1
;
}
}
return
tru
e
;
return
rang
e
;
}
}
bool
unsigned
ParseCommandArg
(
Response
&
r
,
unsigned
&
value_r
,
const
char
*
s
,
ParseCommandArgUnsigned
(
const
char
*
s
,
unsigned
max_value
)
unsigned
max_value
)
{
{
unsigned
long
value
;
char
*
endptr
;
char
*
endptr
;
auto
value
=
strtoul
(
s
,
&
endptr
,
10
);
if
(
endptr
==
s
||
*
endptr
!=
0
)
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
value
=
strtoul
(
s
,
&
endptr
,
10
);
if
(
value
>
max_value
)
if
(
endptr
==
s
||
*
endptr
!=
0
)
{
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
r
.
FormatError
(
ACK_ERROR_ARG
,
"Integer expected: %s"
,
s
);
"Number too large: %s"
,
s
);
return
false
;
}
if
(
value
>
max_value
)
{
r
.
FormatError
(
ACK_ERROR_ARG
,
"Number too large: %s"
,
s
);
return
false
;
}
value_r
=
(
unsigned
)
value
;
return
(
unsigned
)
value
;
return
true
;
}
}
bool
unsigned
ParseCommandArg
(
Response
&
r
,
unsigned
&
value_r
,
const
char
*
s
)
ParseCommandArg
Unsigned
(
const
char
*
s
)
{
{
return
ParseCommandArg
(
r
,
value_r
,
s
,
return
ParseCommandArg
Unsigned
(
s
,
std
::
numeric_limits
<
unsigned
>::
max
());
std
::
numeric_limits
<
unsigned
>::
max
());
}
}
bool
bool
ParseCommandArg
(
Response
&
r
,
bool
&
value_r
,
const
char
*
s
)
ParseCommandArg
Bool
(
const
char
*
s
)
{
{
long
value
;
char
*
endptr
;
char
*
endptr
;
auto
value
=
strtol
(
s
,
&
endptr
,
10
);
if
(
endptr
==
s
||
*
endptr
!=
0
||
(
value
!=
0
&&
value
!=
1
))
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Boolean (0/1) expected: %s"
,
s
);
value
=
strtol
(
s
,
&
endptr
,
10
);
return
!!
value
;
if
(
endptr
==
s
||
*
endptr
!=
0
||
(
value
!=
0
&&
value
!=
1
))
{
r
.
FormatError
(
ACK_ERROR_ARG
,
"Boolean (0/1) expected: %s"
,
s
);
return
false
;
}
value_r
=
!!
value
;
return
true
;
}
}
bool
float
ParseCommandArg
(
Response
&
r
,
float
&
value_r
,
const
char
*
s
)
ParseCommandArg
Float
(
const
char
*
s
)
{
{
float
value
;
char
*
endptr
;
char
*
endptr
;
auto
value
=
strtof
(
s
,
&
endptr
);
if
(
endptr
==
s
||
*
endptr
!=
0
)
throw
FormatProtocolError
(
ACK_ERROR_ARG
,
"Float expected: %s"
,
s
);
value
=
strtof
(
s
,
&
endptr
);
return
value
;
if
(
endptr
==
s
||
*
endptr
!=
0
)
{
r
.
FormatError
(
ACK_ERROR_ARG
,
"Float expected: %s"
,
s
);
return
false
;
}
value_r
=
value
;
return
true
;
}
}
bool
SongTime
ParseCommandArg
(
Response
&
r
,
SongTime
&
value_r
,
const
char
*
s
)
ParseCommandArg
SongTime
(
const
char
*
s
)
{
{
float
value
;
auto
value
=
ParseCommandArgFloat
(
s
);
bool
success
=
ParseCommandArg
(
r
,
value
,
s
)
&&
value
>=
0
;
return
SongTime
::
FromS
(
value
);
if
(
success
)
value_r
=
SongTime
::
FromS
(
value
);
return
success
;
}
}
bool
SignedSongTime
ParseCommandArg
(
Response
&
r
,
SignedSongTime
&
value_r
,
const
char
*
s
)
ParseCommandArg
SignedSongTime
(
const
char
*
s
)
{
{
float
value
;
auto
value
=
ParseCommandArgFloat
(
s
);
bool
success
=
ParseCommandArg
(
r
,
value
,
s
);
return
SongTime
::
FromS
(
value
);
if
(
success
)
value_r
=
SignedSongTime
::
FromS
(
value
);
return
success
;
}
}
src/protocol/ArgParser.hxx
View file @
e939d667
...
@@ -21,6 +21,7 @@
...
@@ -21,6 +21,7 @@
#define MPD_PROTOCOL_ARGPARSER_HXX
#define MPD_PROTOCOL_ARGPARSER_HXX
#include "check.h"
#include "check.h"
#include "Compiler.h"
#include <limits>
#include <limits>
...
@@ -30,15 +31,17 @@ class Response;
...
@@ -30,15 +31,17 @@ class Response;
class
SongTime
;
class
SongTime
;
class
SignedSongTime
;
class
SignedSongTime
;
bool
gcc_pure
ParseCommandArg32
(
Response
&
r
,
uint32_t
&
value_r
,
const
char
*
s
);
uint32_t
ParseCommandArgU32
(
const
char
*
s
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
int
&
value_r
,
const
char
*
s
,
int
int
min_value
,
int
max_value
);
ParseCommandArgInt
(
const
char
*
s
,
int
min_value
,
int
max_value
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
int
&
value_r
,
const
char
*
s
);
int
ParseCommandArgInt
(
const
char
*
s
);
struct
RangeArg
{
struct
RangeArg
{
unsigned
start
,
end
;
unsigned
start
,
end
;
...
@@ -53,26 +56,32 @@ struct RangeArg {
...
@@ -53,26 +56,32 @@ struct RangeArg {
}
}
};
};
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
RangeArg
&
value_r
,
const
char
*
s
);
RangeArg
ParseCommandArgRange
(
const
char
*
s
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
unsigned
&
value_r
,
const
char
*
s
,
unsigned
unsigned
max_value
);
ParseCommandArgUnsigned
(
const
char
*
s
,
unsigned
max_value
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
unsigned
&
value_r
,
const
char
*
s
);
unsigned
ParseCommandArgUnsigned
(
const
char
*
s
);
gcc_pure
bool
bool
ParseCommandArg
(
Response
&
r
,
bool
&
value_r
,
const
char
*
s
);
ParseCommandArg
Bool
(
const
char
*
s
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
float
&
value_r
,
const
char
*
s
);
float
ParseCommandArgFloat
(
const
char
*
s
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
SongTime
&
value_r
,
const
char
*
s
);
SongTime
ParseCommandArgSongTime
(
const
char
*
s
);
bool
gcc_pure
ParseCommandArg
(
Response
&
r
,
SignedSongTime
&
value_r
,
const
char
*
s
);
SignedSongTime
ParseCommandArgSignedSongTime
(
const
char
*
s
);
#endif
#endif
test/test_protocol.cxx
View file @
e939d667
#include "config.h"
#include "config.h"
#include "protocol/ArgParser.hxx"
#include "protocol/ArgParser.hxx"
#include "
client/Response
.hxx"
#include "
protocol/Ack
.hxx"
#include "Compiler.h"
#include "Compiler.h"
#include <cppunit/TestFixture.h>
#include <cppunit/TestFixture.h>
...
@@ -10,20 +10,6 @@
...
@@ -10,20 +10,6 @@
#include <stdlib.h>
#include <stdlib.h>
static
enum
ack
last_error
=
ack
(
-
1
);
void
Response
::
Error
(
enum
ack
code
,
gcc_unused
const
char
*
msg
)
{
last_error
=
code
;
}
void
Response
::
FormatError
(
enum
ack
code
,
gcc_unused
const
char
*
fmt
,
...)
{
last_error
=
code
;
}
class
ArgParserTest
:
public
CppUnit
::
TestFixture
{
class
ArgParserTest
:
public
CppUnit
::
TestFixture
{
CPPUNIT_TEST_SUITE
(
ArgParserTest
);
CPPUNIT_TEST_SUITE
(
ArgParserTest
);
CPPUNIT_TEST
(
TestRange
);
CPPUNIT_TEST
(
TestRange
);
...
@@ -36,25 +22,24 @@ public:
...
@@ -36,25 +22,24 @@ public:
void
void
ArgParserTest
::
TestRange
()
ArgParserTest
::
TestRange
()
{
{
Client
&
client
=
*
(
Client
*
)
nullptr
;
RangeArg
range
=
ParseCommandArgRange
(
"1"
);
Response
r
(
client
,
0
);
RangeArg
range
;
CPPUNIT_ASSERT
(
ParseCommandArg
(
r
,
range
,
"1"
));
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT_EQUAL
(
2u
,
range
.
end
);
CPPUNIT_ASSERT_EQUAL
(
2u
,
range
.
end
);
CPPUNIT_ASSERT
(
ParseCommandArg
(
r
,
range
,
"1:5"
)
);
range
=
ParseCommandArgRange
(
"1:5"
);
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT_EQUAL
(
5u
,
range
.
end
);
CPPUNIT_ASSERT_EQUAL
(
5u
,
range
.
end
);
CPPUNIT_ASSERT
(
ParseCommandArg
(
r
,
range
,
"1:"
)
);
range
=
ParseCommandArgRange
(
"1:"
);
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT_EQUAL
(
1u
,
range
.
start
);
CPPUNIT_ASSERT
(
range
.
end
>=
999999u
);
CPPUNIT_ASSERT
(
range
.
end
>=
999999u
);
CPPUNIT_ASSERT
(
!
ParseCommandArg
(
r
,
range
,
"-2"
));
try
{
CPPUNIT_ASSERT_EQUAL
(
ACK_ERROR_ARG
,
last_error
);
range
=
ParseCommandArgRange
(
"-2"
);
CPPUNIT_ASSERT
(
false
);
}
catch
(
ProtocolError
)
{
CPPUNIT_ASSERT
(
true
);
}
}
}
CPPUNIT_TEST_SUITE_REGISTRATION
(
ArgParserTest
);
CPPUNIT_TEST_SUITE_REGISTRATION
(
ArgParserTest
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment