Filter.cxx 11.4 KB
Newer Older
1
/*
2
 * Copyright 2003-2018 The Music Player Daemon Project
3
 * http://www.musicpd.org
4 5 6 7 8 9 10 11 12 13
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
14 15 16 17
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 19
 */

20
#include "config.h"
21
#include "Filter.hxx"
22
#include "NotSongFilter.hxx"
23 24 25 26
#include "UriSongFilter.hxx"
#include "BaseSongFilter.hxx"
#include "TagSongFilter.hxx"
#include "ModifiedSinceSongFilter.hxx"
27
#include "AudioFormatSongFilter.hxx"
28
#include "LightSong.hxx"
29
#include "AudioParser.hxx"
30
#include "tag/ParseName.hxx"
31
#include "tag/Tag.hxx"
32 33
#include "time/ChronoUtil.hxx"
#include "time/ISO8601.hxx"
34
#include "util/CharUtil.hxx"
35
#include "util/ConstBuffer.hxx"
36
#include "util/RuntimeError.hxx"
37
#include "util/StringAPI.hxx"
38
#include "util/StringCompare.hxx"
39
#include "util/StringStrip.hxx"
40
#include "util/StringView.hxx"
41
#include "util/ASCII.hxx"
42
#include "util/UriUtil.hxx"
43
#include "lib/icu/CaseFold.hxx"
44

45
#include <exception>
46

47
#include <assert.h>
48 49
#include <stdlib.h>

50 51
#define LOCATE_TAG_FILE_KEY     "file"
#define LOCATE_TAG_FILE_KEY_OLD "filename"
52 53
#define LOCATE_TAG_ANY_KEY      "any"

54 55 56 57 58 59 60
enum {
	/**
	 * Limit the search to files within the given directory.
	 */
	LOCATE_TAG_BASE_TYPE = TAG_NUM_OF_ITEM_TYPES + 1,

	LOCATE_TAG_MODIFIED_SINCE,
61
	LOCATE_TAG_AUDIO_FORMAT,
62 63 64
	LOCATE_TAG_FILE_TYPE,
	LOCATE_TAG_ANY_TYPE,
};
65

66 67 68 69 70
/**
 * @return #TAG_NUM_OF_ITEM_TYPES on error
 */
gcc_pure
static unsigned
71
locate_parse_type(const char *str) noexcept
72
{
73 74
	if (StringEqualsCaseASCII(str, LOCATE_TAG_FILE_KEY) ||
	    StringEqualsCaseASCII(str, LOCATE_TAG_FILE_KEY_OLD))
75 76
		return LOCATE_TAG_FILE_TYPE;

77
	if (StringEqualsCaseASCII(str, LOCATE_TAG_ANY_KEY))
78 79
		return LOCATE_TAG_ANY_TYPE;

80 81 82
	if (strcmp(str, "base") == 0)
		return LOCATE_TAG_BASE_TYPE;

83 84 85
	if (strcmp(str, "modified-since") == 0)
		return LOCATE_TAG_MODIFIED_SINCE;

86 87 88
	if (StringEqualsCaseASCII(str, "AudioFormat"))
		return LOCATE_TAG_AUDIO_FORMAT;

89
	return tag_name_parse_i(str);
90 91
}

92 93
SongFilter::SongFilter(TagType tag, const char *value, bool fold_case)
{
94 95
	/* for compatibility with MPD 0.20 and older, "fold_case" also
	   switches on "substring" */
96 97
	and_filter.AddItem(std::make_unique<TagSongFilter>(tag,
							   StringFilter(value, fold_case, fold_case, false)));
98 99 100 101 102 103 104 105 106 107 108 109 110
}

SongFilter::~SongFilter()
{
	/* this destructor exists here just so it won't get inlined */
}

std::string
SongFilter::ToExpression() const noexcept
{
	return and_filter.ToExpression();
}

111 112
static std::chrono::system_clock::time_point
ParseTimeStamp(const char *s)
113 114 115 116 117 118 119
{
	assert(s != nullptr);

	char *endptr;
	unsigned long long value = strtoull(s, &endptr, 10);
	if (*endptr == 0 && endptr > s)
		/* it's an integral UNIX time stamp */
120
		return std::chrono::system_clock::from_time_t((time_t)value);
121

122
	/* try ISO 8601 */
123
	return ParseISO8601(s);
124 125
}

126 127 128
static constexpr bool
IsTagNameChar(char ch) noexcept
{
129
	return IsAlphaASCII(ch) || ch == '_' || ch == '-';
130 131 132 133 134 135 136 137 138 139 140
}

static const char *
FirstNonTagNameChar(const char *s) noexcept
{
	while (IsTagNameChar(*s))
		++s;
	return s;
}

static auto
141
ExpectWord(const char *&s)
142
{
143
	const char *begin = s;
144 145
	const char *end = FirstNonTagNameChar(s);
	if (end == s)
146
		throw std::runtime_error("Word expected");
147 148

	s = StripLeft(end);
149 150 151 152 153 154 155
	return std::string(begin, end);
}

static auto
ExpectFilterType(const char *&s)
{
	const auto name = ExpectWord(s);
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177

	const auto type = locate_parse_type(name.c_str());
	if (type == TAG_NUM_OF_ITEM_TYPES)
		throw FormatRuntimeError("Unknown filter type: %s",
					 name.c_str());

	return type;
}

static constexpr bool
IsQuote(char ch) noexcept
{
	return ch == '"' || ch == '\'';
}

static std::string
ExpectQuoted(const char *&s)
{
	const char quote = *s++;
	if (!IsQuote(quote))
		throw std::runtime_error("Quoted string expected");

178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
	char buffer[4096];
	size_t length = 0;

	while (*s != quote) {
		if (*s == '\\')
			/* backslash escapes the following character */
			++s;

		if (*s == 0)
			throw std::runtime_error("Closing quote not found");

		buffer[length++] = *s++;

		if (length >= sizeof(buffer))
			throw std::runtime_error("Quoted value is too long");
	}

	s = StripLeft(s + 1);
196

197
	return {buffer, length};
198 199
}

200 201 202 203 204 205 206 207 208
/**
 * Parse a string operator and its second operand and convert it to a
 * #StringFilter.
 *
 * Throws on error.
 */
static StringFilter
ParseStringFilter(const char *&s, bool fold_case)
{
209 210 211 212 213 214 215 216 217 218 219 220 221 222
	if (auto after_contains = StringAfterPrefixIgnoreCase(s, "contains ")) {
		s = StripLeft(after_contains);
		auto value = ExpectQuoted(s);
		return StringFilter(std::move(value),
				    fold_case, true, false);
	}

	if (auto after_not_contains = StringAfterPrefixIgnoreCase(s, "!contains ")) {
		s = StripLeft(after_not_contains);
		auto value = ExpectQuoted(s);
		return StringFilter(std::move(value),
				    fold_case, true, true);
	}

223
	bool negated = false;
224 225 226 227 228 229 230 231 232 233 234 235 236 237

#ifdef HAVE_PCRE
	if ((s[0] == '!' || s[0] == '=') && s[1] == '~') {
		negated = s[0] == '!';
		s = StripLeft(s + 2);
		auto value = ExpectQuoted(s);
		StringFilter f(std::move(value), fold_case, false, negated);
		f.SetRegex(std::make_shared<UniqueRegex>(f.GetValue().c_str(),
							 false, false,
							 fold_case));
		return f;
	}
#endif

238 239 240 241 242 243 244 245 246 247 248 249
	if (s[0] == '!' && s[1] == '=')
		negated = true;
	else if (s[0] != '=' || s[1] != '=')
		throw std::runtime_error("'==' or '!=' expected");

	s = StripLeft(s + 2);
	auto value = ExpectQuoted(s);

	return StringFilter(std::move(value),
			    fold_case, false, negated);
}

250 251
ISongFilterPtr
SongFilter::ParseExpression(const char *&s, bool fold_case)
252 253 254 255 256
{
	assert(*s == '(');

	s = StripLeft(s + 1);

257 258 259 260 261 262 263
	if (*s == '(') {
		auto first = ParseExpression(s, fold_case);
		if (*s == ')') {
			++s;
			return first;
		}

264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280
		if (ExpectWord(s) != "AND")
			throw std::runtime_error("'AND' expected");

		auto and_filter = std::make_unique<AndSongFilter>();
		and_filter->AddItem(std::move(first));

		while (true) {
			and_filter->AddItem(ParseExpression(s, fold_case));

			if (*s == ')') {
				++s;
				return and_filter;
			}

			if (ExpectWord(s) != "AND")
				throw std::runtime_error("'AND' expected");
		}
281
	}
282

283 284 285 286 287 288 289 290 291 292 293 294 295 296
	if (*s == '!') {
		s = StripLeft(s + 1);

		if (*s != '(')
			throw std::runtime_error("'(' expected");

		auto inner = ParseExpression(s, fold_case);
		if (*s != ')')
			throw std::runtime_error("')' expected");
		s = StripLeft(s + 1);

		return std::make_unique<NotSongFilter>(std::move(inner));
	}

297
	auto type = ExpectFilterType(s);
298 299 300 301 302

	if (type == LOCATE_TAG_MODIFIED_SINCE) {
		const auto value_s = ExpectQuoted(s);
		if (*s != ')')
			throw std::runtime_error("')' expected");
303 304
		s = StripLeft(s + 1);
		return std::make_unique<ModifiedSinceSongFilter>(ParseTimeStamp(value_s.c_str()));
305 306 307 308
	} else if (type == LOCATE_TAG_BASE_TYPE) {
		auto value = ExpectQuoted(s);
		if (*s != ')')
			throw std::runtime_error("')' expected");
309
		s = StripLeft(s + 1);
310

311
		return std::make_unique<BaseSongFilter>(std::move(value));
312
	} else if (type == LOCATE_TAG_AUDIO_FORMAT) {
313 314 315 316 317 318 319
		bool mask;
		if (s[0] == '=' && s[1] == '=')
			mask = false;
		else if (s[0] == '=' && s[1] == '~')
			mask = true;
		else
			throw std::runtime_error("'==' or '=~' expected");
320 321 322 323

		s = StripLeft(s + 2);

		const auto value = ParseAudioFormat(ExpectQuoted(s).c_str(),
324
						    mask);
325 326 327 328 329 330

		if (*s != ')')
			throw std::runtime_error("')' expected");
		s = StripLeft(s + 1);

		return std::make_unique<AudioFormatSongFilter>(value);
331
	} else {
332
		auto string_filter = ParseStringFilter(s, fold_case);
333 334 335
		if (*s != ')')
			throw std::runtime_error("')' expected");

336 337 338 339 340 341
		s = StripLeft(s + 1);

		if (type == LOCATE_TAG_ANY_TYPE)
			type = TAG_NUM_OF_ITEM_TYPES;

		if (type == LOCATE_TAG_FILE_TYPE)
342
			return std::make_unique<UriSongFilter>(std::move(string_filter));
343 344

		return std::make_unique<TagSongFilter>(TagType(type),
345
						       std::move(string_filter));
346 347 348
	}
}

349
void
350 351 352
SongFilter::Parse(const char *tag_string, const char *value, bool fold_case)
{
	unsigned tag = locate_parse_type(tag_string);
353 354 355

	switch (tag) {
	case TAG_NUM_OF_ITEM_TYPES:
356
		throw std::runtime_error("Unknown filter type");
357

358
	case LOCATE_TAG_BASE_TYPE:
359
		if (!uri_safe_local(value))
360
			throw std::runtime_error("Bad URI");
361

362
		and_filter.AddItem(std::make_unique<BaseSongFilter>(value));
363 364 365
		break;

	case LOCATE_TAG_MODIFIED_SINCE:
366
		and_filter.AddItem(std::make_unique<ModifiedSinceSongFilter>(ParseTimeStamp(value)));
367
		break;
368

369
	case LOCATE_TAG_FILE_TYPE:
370 371
		/* for compatibility with MPD 0.20 and older,
		   "fold_case" also switches on "substring" */
372 373 374 375
		and_filter.AddItem(std::make_unique<UriSongFilter>(StringFilter(value,
										fold_case,
										fold_case,
										false)));
376 377 378 379 380 381
		break;

	default:
		if (tag == LOCATE_TAG_ANY_TYPE)
			tag = TAG_NUM_OF_ITEM_TYPES;

382 383
		/* for compatibility with MPD 0.20 and older,
		   "fold_case" also switches on "substring" */
384
		and_filter.AddItem(std::make_unique<TagSongFilter>(TagType(tag),
385 386 387 388
								   StringFilter(value,
										fold_case,
										fold_case,
										false)));
389 390
		break;
	}
391 392
}

393
void
394
SongFilter::Parse(ConstBuffer<const char *> args, bool fold_case)
395
{
396
	if (args.empty())
397
		throw std::runtime_error("Incorrect number of filter arguments");
398

399
	do {
400 401
		if (*args.front() == '(') {
			const char *s = args.shift();
402 403
			const char *end = s;
			auto f = ParseExpression(end, fold_case);
404 405 406
			if (*end != 0)
				throw std::runtime_error("Unparsed garbage after expression");

407
			and_filter.AddItem(std::move(f));
408 409 410
			continue;
		}

411 412 413 414 415 416 417
		if (args.size < 2)
			throw std::runtime_error("Incorrect number of filter arguments");

		const char *tag = args.shift();
		const char *value = args.shift();
		Parse(tag, value, fold_case);
	} while (!args.empty());
418 419
}

420 421 422 423 424 425
void
SongFilter::Optimize() noexcept
{
	OptimizeSongFilter(and_filter);
}

426
bool
427
SongFilter::Match(const LightSong &song) const noexcept
428
{
429
	return and_filter.Match(song);
430 431
}

432 433 434
bool
SongFilter::HasFoldCase() const noexcept
{
435
	for (const auto &i : and_filter.GetItems()) {
436 437 438 439 440 441 442 443 444 445 446 447
		if (auto t = dynamic_cast<const TagSongFilter *>(i.get())) {
			if (t->GetFoldCase())
				return true;
		} else if (auto u = dynamic_cast<const UriSongFilter *>(i.get())) {
			if (u->GetFoldCase())
				return true;
		}
	}

	return false;
}

448
bool
449
SongFilter::HasOtherThanBase() const noexcept
450
{
451
	for (const auto &i : and_filter.GetItems()) {
452 453
		const auto *f = dynamic_cast<const BaseSongFilter *>(i.get());
		if (f == nullptr)
454
			return true;
455
	}
456 457 458 459

	return false;
}

460
const char *
461
SongFilter::GetBase() const noexcept
462
{
463
	for (const auto &i : and_filter.GetItems()) {
464 465 466 467
		const auto *f = dynamic_cast<const BaseSongFilter *>(i.get());
		if (f != nullptr)
			return f->GetValue();
	}
468

469
	return nullptr;
470
}
471 472 473 474 475 476 477

SongFilter
SongFilter::WithoutBasePrefix(const char *_prefix) const noexcept
{
	const StringView prefix(_prefix);
	SongFilter result;

478
	for (const auto &i : and_filter.GetItems()) {
479 480 481
		const auto *f = dynamic_cast<const BaseSongFilter *>(i.get());
		if (f != nullptr) {
			const char *s = StringAfterPrefix(f->GetValue(), prefix);
482 483 484 485 486 487 488 489
			if (s != nullptr) {
				if (*s == 0)
					continue;

				if (*s == '/') {
					++s;

					if (*s != 0)
490
						result.and_filter.AddItem(std::make_unique<BaseSongFilter>(s));
491 492 493 494 495 496

					continue;
				}
			}
		}

497
		result.and_filter.AddItem(i->Clone());
498 499 500 501
	}

	return result;
}