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
2b10ecfa
Commit
2b10ecfa
authored
Jan 21, 2014
by
Max Kellermann
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
IcyMetadataParser: more robust tag parser
Allow semicolons and single quotes in the stream title. This is not part of any specification, but found in real life.
parent
f7eb2b69
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
87 additions
and
21 deletions
+87
-21
NEWS
NEWS
+1
-0
IcyMetaDataParser.cxx
src/IcyMetaDataParser.cxx
+71
-21
test_icy_parser.cxx
test/test_icy_parser.cxx
+15
-0
No files found.
NEWS
View file @
2b10ecfa
ver 0.18.8 (not yet released)
* decoder
- ffmpeg: support libav v10_alpha1
* more robust Icy-Metadata parser
* fix Solaris build failure
ver 0.18.7 (2013/01/13)
...
...
src/IcyMetaDataParser.cxx
View file @
2b10ecfa
...
...
@@ -81,31 +81,85 @@ icy_add_item(Tag &tag, TagType type, const char *value)
}
static
void
icy_parse_tag_item
(
Tag
&
tag
,
const
char
*
item
)
icy_parse_tag_item
(
Tag
&
tag
,
const
char
*
name
,
const
char
*
value
)
{
gchar
**
p
=
g_strsplit
(
item
,
"="
,
0
);
if
(
p
[
0
]
!=
nullptr
&&
p
[
1
]
!=
nullptr
)
{
if
(
strcmp
(
p
[
0
],
"StreamTitle"
)
==
0
)
icy_add_item
(
tag
,
TAG_TITLE
,
p
[
1
]);
else
FormatDebug
(
icy_metadata_domain
,
"unknown icy-tag: '%s'"
,
p
[
0
]);
}
if
(
strcmp
(
name
,
"StreamTitle"
)
==
0
)
icy_add_item
(
tag
,
TAG_TITLE
,
value
);
else
FormatDebug
(
icy_metadata_domain
,
"unknown icy-tag: '%s'"
,
name
);
}
/**
* Find a single quote that is followed by a semicolon (or by the end
* of the string). If that fails, return the first single quote. If
* that also fails, return #end.
*/
static
char
*
find_end_quote
(
char
*
p
,
char
*
const
end
)
{
char
*
fallback
=
std
::
find
(
p
,
end
,
'\''
);
if
(
fallback
>=
end
-
1
||
fallback
[
1
]
==
';'
)
return
fallback
;
g_strfreev
(
p
);
p
=
fallback
+
1
;
while
(
true
)
{
p
=
std
::
find
(
p
,
end
,
'\''
);
if
(
p
==
end
)
return
fallback
;
if
(
p
==
end
-
1
||
p
[
1
]
==
';'
)
return
p
;
++
p
;
}
}
static
Tag
*
icy_parse_tag
(
c
onst
char
*
p
)
icy_parse_tag
(
c
har
*
p
,
char
*
const
end
)
{
assert
(
p
!=
nullptr
);
assert
(
end
!=
nullptr
);
assert
(
p
<=
end
);
Tag
*
tag
=
new
Tag
();
gchar
**
items
=
g_strsplit
(
p
,
";"
,
0
);
for
(
unsigned
i
=
0
;
items
[
i
]
!=
nullptr
;
++
i
)
icy_parse_tag_item
(
*
tag
,
items
[
i
]);
while
(
p
!=
end
)
{
const
char
*
const
name
=
p
;
char
*
eq
=
std
::
find
(
p
,
end
,
'='
);
if
(
eq
==
end
)
break
;
*
eq
=
0
;
p
=
eq
+
1
;
if
(
*
p
!=
'\''
)
{
/* syntax error; skip to the next semicolon,
try to recover */
char
*
semicolon
=
std
::
find
(
p
,
end
,
';'
);
if
(
semicolon
==
end
)
break
;
p
=
semicolon
+
1
;
continue
;
}
++
p
;
const
char
*
const
value
=
p
;
char
*
quote
=
find_end_quote
(
p
,
end
);
if
(
quote
==
end
)
break
;
*
quote
=
0
;
p
=
quote
+
1
;
g_strfreev
(
items
);
icy_parse_tag_item
(
*
tag
,
name
,
value
);
char
*
semicolon
=
std
::
find
(
p
,
end
,
';'
);
if
(
semicolon
==
end
)
break
;
p
=
semicolon
+
1
;
}
return
tag
;
}
...
...
@@ -152,15 +206,11 @@ IcyMetaDataParser::Meta(const void *data, size_t length)
++
length
;
if
(
meta_position
==
meta_size
)
{
/* null-terminate the string */
meta_data
[
meta_size
]
=
0
;
/* parse */
delete
tag
;
tag
=
icy_parse_tag
(
meta_data
);
tag
=
icy_parse_tag
(
meta_data
,
meta_data
+
meta_size
);
g_free
(
meta_data
);
/* change back to normal data mode */
...
...
test/test_icy_parser.cxx
View file @
2b10ecfa
...
...
@@ -14,6 +14,17 @@
#include <string>
#include <string.h>
static
Tag
*
icy_parse_tag
(
const
char
*
p
)
{
char
*
q
=
strdup
(
p
);
Tag
*
tag
=
icy_parse_tag
(
q
,
q
+
strlen
(
q
));
free
(
q
);
return
tag
;
}
static
void
CompareTagTitle
(
const
Tag
&
tag
,
const
std
::
string
&
title
)
{
...
...
@@ -51,10 +62,14 @@ public:
TestIcyParserTitle
(
"StreamTitle='foo bar'"
,
"foo bar"
);
TestIcyParserTitle
(
"StreamTitle='foo bar';"
,
"foo bar"
);
TestIcyParserTitle
(
"StreamTitle='foo
\"
bar';"
,
"foo
\"
bar"
);
TestIcyParserTitle
(
"StreamTitle='foo=bar';"
,
"foo=bar"
);
TestIcyParserTitle
(
"a=b;StreamTitle='foo';"
,
"foo"
);
TestIcyParserTitle
(
"a=;StreamTitle='foo';"
,
"foo"
);
TestIcyParserTitle
(
"a=b;StreamTitle='foo';c=d"
,
"foo"
);
TestIcyParserTitle
(
"a=b;StreamTitle='foo'"
,
"foo"
);
TestIcyParserTitle
(
"a='b;c';StreamTitle='foo;bar'"
,
"foo;bar"
);
TestIcyParserTitle
(
"a='b'c';StreamTitle='foo'bar'"
,
"foo'bar"
);
TestIcyParserTitle
(
"StreamTitle='fo'o'b'ar';a='b'c'd'"
,
"fo'o'b'ar"
);
}
};
...
...
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