patrick
/
plasp
Archived
1
0
Fork 0
This repository has been archived on 2023-07-19. You can view files and clone it, but cannot push or open issues or pull requests.
plasp/src/plasp/pddl/Expression.cpp

242 lines
7.8 KiB
C++
Raw Normal View History

#include <plasp/pddl/Expression.h>
#include <plasp/pddl/Context.h>
#include <plasp/pddl/Domain.h>
#include <plasp/pddl/ExpressionContext.h>
#include <plasp/pddl/Identifier.h>
#include <plasp/pddl/IO.h>
2016-06-03 13:33:05 +02:00
#include <plasp/pddl/expressions/And.h>
2016-06-04 17:20:25 +02:00
#include <plasp/pddl/expressions/Imply.h>
2016-06-03 13:33:05 +02:00
#include <plasp/pddl/expressions/Not.h>
#include <plasp/pddl/expressions/Or.h>
#include <plasp/pddl/expressions/Predicate.h>
2016-06-03 17:12:39 +02:00
#include <plasp/pddl/expressions/PredicateDeclaration.h>
#include <plasp/pddl/expressions/Reference.h>
#include <plasp/utils/IO.h>
#include <plasp/utils/ParserException.h>
namespace plasp
{
namespace pddl
{
////////////////////////////////////////////////////////////////////////////////////////////////////
//
// Expression
//
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseExpressionContent(const std::string &expressionIdentifier, Context &context,
ExpressionContext &expressionContext);
ExpressionPointer parseEffectBodyExpressionContent(const std::string &expressionIdentifier,
Context &context, ExpressionContext &expressionContext);
ExpressionPointer parsePredicate(Context &context, ExpressionContext &expressionContext);
2016-06-02 21:53:10 +02:00
////////////////////////////////////////////////////////////////////////////////////////////////////
inline void warnUnsupported(Context &context, const std::string &expressionIdentifier)
2016-06-02 21:53:10 +02:00
{
context.logger.parserWarning(context.parser, "Expression type \"" + expressionIdentifier + "\" currently unsupported in this context");
skipSection(context.parser);
2016-06-02 21:53:10 +02:00
}
2016-06-02 18:31:47 +02:00
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parsePreconditionExpression(Context &context,
ExpressionContext &expressionContext)
{
context.parser.expect<std::string>("(");
const auto expressionIdentifier = context.parser.parseIdentifier(isIdentifier);
ExpressionPointer expression;
2016-06-02 17:35:01 +02:00
if (expressionIdentifier == "and")
{
expression = expressions::And::parse(context, expressionContext,
parsePreconditionExpression);
}
2016-06-02 21:53:10 +02:00
else if (expressionIdentifier == "forall"
|| expressionIdentifier == "preference")
{
warnUnsupported(context, expressionIdentifier);
return nullptr;
2016-06-02 21:53:10 +02:00
}
2016-06-02 17:35:01 +02:00
else
expression = parseExpressionContent(expressionIdentifier, context, expressionContext);
2016-06-02 18:31:47 +02:00
context.parser.expect<std::string>(")");
2016-06-02 18:31:47 +02:00
return expression;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseExpression(Context &context, ExpressionContext &expressionContext)
2016-06-02 18:31:47 +02:00
{
context.parser.expect<std::string>("(");
2016-06-02 18:31:47 +02:00
const auto expressionIdentifier = context.parser.parseIdentifier(isIdentifier);
2016-06-02 18:31:47 +02:00
auto expression = parseExpressionContent(expressionIdentifier, context, expressionContext);
context.parser.expect<std::string>(")");
2016-06-02 17:35:01 +02:00
return expression;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseExpressionContent(const std::string &expressionIdentifier, Context &context,
ExpressionContext &expressionContext)
2016-06-02 18:31:47 +02:00
{
context.parser.skipWhiteSpace();
2016-06-02 18:31:47 +02:00
if (expressionIdentifier == "and")
return expressions::And::parse(context, expressionContext, parseExpression);
2016-06-04 18:12:41 +02:00
if (expressionIdentifier == "or")
return expressions::Or::parse(context, expressionContext, parseExpression);
2016-06-04 18:12:41 +02:00
if (expressionIdentifier == "not")
return expressions::Not::parse(context, expressionContext, parseExpression);
2016-06-04 18:12:41 +02:00
if (expressionIdentifier == "imply")
return expressions::Imply::parse(context, expressionContext, parseExpression);
2016-06-04 18:12:41 +02:00
if (expressionIdentifier == "exists"
2016-06-02 18:31:47 +02:00
|| expressionIdentifier == "forall"
|| expressionIdentifier == "-"
|| expressionIdentifier == "="
|| expressionIdentifier == "*"
|| expressionIdentifier == "+"
|| expressionIdentifier == "-"
|| expressionIdentifier == "/"
|| expressionIdentifier == ">"
|| expressionIdentifier == "<"
|| expressionIdentifier == "="
|| expressionIdentifier == ">="
|| expressionIdentifier == "<=")
{
warnUnsupported(context, expressionIdentifier);
return nullptr;
2016-06-02 21:53:10 +02:00
}
auto &predicateDeclarations = expressionContext.domain.predicates();
2016-06-04 18:12:41 +02:00
// Check if predicate with that name exists
const auto match = std::find_if(predicateDeclarations.cbegin(), predicateDeclarations.cend(),
2016-06-04 18:12:41 +02:00
[&](const auto &predicate)
{
return predicate->name() == expressionIdentifier;
});
// If predicate exists, parse it
if (match != predicateDeclarations.cend())
return expressions::Predicate::parse(expressionIdentifier, context, expressionContext);
2016-06-04 18:12:41 +02:00
throw utils::ParserException(context.parser, "Unknown expression \"" + expressionIdentifier + "\"");
2016-06-02 21:53:10 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseEffectExpression(Context &context, ExpressionContext &expressionContext)
2016-06-02 21:53:10 +02:00
{
context.parser.expect<std::string>("(");
2016-06-02 21:53:10 +02:00
const auto expressionIdentifier = context.parser.parseIdentifier(isIdentifier);
2016-06-02 21:53:10 +02:00
ExpressionPointer expression;
2016-06-02 21:53:10 +02:00
if (expressionIdentifier == "and")
expression = expressions::And::parse(context, expressionContext, parseEffectExpression);
2016-06-02 21:53:10 +02:00
else if (expressionIdentifier == "forall"
|| expressionIdentifier == "when")
{
warnUnsupported(context, expressionIdentifier);
return nullptr;
2016-06-02 21:53:10 +02:00
}
else
expression = parseEffectBodyExpressionContent(expressionIdentifier, context, expressionContext);
2016-06-02 21:53:10 +02:00
context.parser.expect<std::string>(")");
2016-06-02 21:53:10 +02:00
return expression;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parseEffectBodyExpressionContent(const std::string &expressionIdentifier,
Context &context, ExpressionContext &expressionContext)
2016-06-02 21:53:10 +02:00
{
ExpressionPointer expression;
2016-06-02 21:53:10 +02:00
if (expressionIdentifier == "not")
return expressions::Not::parse(context, expressionContext, parsePredicate);
2016-06-04 18:12:41 +02:00
if (expressionIdentifier == "="
2016-06-02 21:53:10 +02:00
|| expressionIdentifier == "assign"
|| expressionIdentifier == "scale-up"
|| expressionIdentifier == "scale-down"
|| expressionIdentifier == "increase"
|| expressionIdentifier == "decrease")
{
warnUnsupported(context, expressionIdentifier);
return nullptr;
2016-06-02 18:31:47 +02:00
}
const auto &predicateDeclarations = expressionContext.domain.predicates();
2016-06-04 18:12:41 +02:00
// Check if predicate with that name exists
const auto match = std::find_if(predicateDeclarations.cbegin(), predicateDeclarations.cend(),
2016-06-04 18:12:41 +02:00
[&](const auto &predicate)
{
return predicate->name() == expressionIdentifier;
});
// If predicate exists, parse it
if (match != predicateDeclarations.cend())
return expressions::Predicate::parse(expressionIdentifier, context, expressionContext);
2016-06-04 18:12:41 +02:00
throw utils::ParserException(context.parser, "Expression \"" + expressionIdentifier + "\" not allowed in this context");
2016-06-02 18:31:47 +02:00
}
////////////////////////////////////////////////////////////////////////////////////////////////////
ExpressionPointer parsePredicate(Context &context, ExpressionContext &expressionContext)
{
context.parser.expect<std::string>("(");
const auto predicateName = context.parser.parseIdentifier(isIdentifier);
ExpressionPointer expression;
const auto &predicateDeclarations = expressionContext.domain.predicates();
// Check if predicate with that name exists
const auto match = std::find_if(predicateDeclarations.cbegin(), predicateDeclarations.cend(),
[&](const auto &predicate)
{
return predicate->name() == predicateName;
});
// If predicate exists, parse it
if (match == predicateDeclarations.cend())
throw utils::ParserException(context.parser, "Unknown predicate \"" + predicateName + "\"");
expression = expressions::Predicate::parse(predicateName, context, expressionContext);
context.parser.expect<std::string>(")");
return expression;
}
////////////////////////////////////////////////////////////////////////////////////////////////////
}
}