:(scenarios load)
:(scenario first_recipe)
recipe main [
1:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(code)
vector<recipe_ordinal> load(string form) {
istringstream in(form);
in >> std::noskipws;
return load(in);
}
vector<recipe_ordinal> load(istream& in) {
in >> std::noskipws;
vector<recipe_ordinal> result;
while (!in.eof()) {
skip_whitespace_and_comments(in);
if (in.eof()) break;
string command = next_word(in);
if (command == "recipe") {
result.push_back(slurp_recipe(in));
}
else if (command == "recipe!") {
Disable_redefine_warnings = true;
result.push_back(slurp_recipe(in));
Disable_redefine_warnings = false;
}
else {
raise << "unknown top-level command: " << command << '\n' << end();
}
}
return result;
}
long long int slurp_recipe(istream& in) {
string recipe_name = next_word(in);
if (recipe_name.empty())
raise << "empty recipe name\n" << end();
if (Recipe_ordinal.find(recipe_name) == Recipe_ordinal.end()) {
Recipe_ordinal[recipe_name] = Next_recipe_ordinal++;
}
if (warn_on_redefine(recipe_name)
&& Recipe.find(Recipe_ordinal[recipe_name]) != Recipe.end()) {
raise << "redefining recipe " << Recipe[Recipe_ordinal[recipe_name]].name << "\n" << end();
}
Recipe[Recipe_ordinal[recipe_name]] = slurp_body(in);
Recipe[Recipe_ordinal[recipe_name]].name = recipe_name;
recently_added_recipes.push_back(Recipe_ordinal[recipe_name]);
return Recipe_ordinal[recipe_name];
}
recipe slurp_body(istream& in) {
recipe result;
skip_whitespace(in);
if (in.get() != '[')
raise << "recipe body must begin with '['\n" << end();
skip_whitespace_and_comments(in);
instruction curr;
while (next_instruction(in, &curr)) {
result.steps.push_back(curr);
}
return result;
}
bool next_instruction(istream& in, instruction* curr) {
in >> std::noskipws;
curr->clear();
if (in.eof()) {
raise << "0: unbalanced '[' for recipe\n" << end();
return false;
}
skip_whitespace(in);
if (in.eof()) {
raise << "1: unbalanced '[' for recipe\n" << end();
return false;
}
skip_whitespace_and_comments(in);
if (in.eof()) {
raise << "2: unbalanced '[' for recipe\n" << end();
return false;
}
vector<string> words;
while (in.peek() != '\n' && !in.eof()) {
skip_whitespace(in);
if (in.eof()) {
raise << "3: unbalanced '[' for recipe\n" << end();
return false;
}
string word = next_word(in);
words.push_back(word);
skip_whitespace(in);
}
skip_whitespace_and_comments(in);
if (SIZE(words) == 1 && words.at(0) == "]") {
return false;
}
if (SIZE(words) == 1 && !isalnum(words.at(0).at(0)) && words.at(0).at(0) != '$') {
curr->is_label = true;
curr->label = words.at(0);
trace("parse") << "label: " << curr->label << end();
if (in.eof()) {
raise << "7: unbalanced '[' for recipe\n" << end();
return false;
}
return true;
}
vector<string>::iterator p = words.begin();
if (find(words.begin(), words.end(), "<-") != words.end()) {
for (; *p != "<-"; ++p) {
if (*p == ",") continue;
curr->products.push_back(reagent(*p));
}
++p;
}
if (p == words.end()) {
raise << "instruction prematurely ended with '<-'\n" << end() << end();
return false;
}
curr->name = *p;
if (Recipe_ordinal.find(*p) == Recipe_ordinal.end()) {
Recipe_ordinal[*p] = Next_recipe_ordinal++;
}
if (Recipe_ordinal[*p] == 0) {
raise << "Recipe " << *p << " has number 0, which is reserved for IDLE.\n" << end() << end();
return false;
}
curr->operation = Recipe_ordinal[*p]; ++p;
for (; p != words.end(); ++p) {
if (*p == ",") continue;
curr->ingredients.push_back(reagent(*p));
}
trace("parse") << "instruction: " << curr->name << end();
for (vector<reagent>::iterator p = curr->ingredients.begin(); p != curr->ingredients.end(); ++p) {
trace("parse") << " ingredient: " << p->to_string() << end();
}
for (vector<reagent>::iterator p = curr->products.begin(); p != curr->products.end(); ++p) {
trace("parse") << " product: " << p->to_string() << end();
}
if (in.eof()) {
raise << "9: unbalanced '[' for recipe\n" << end();
return false;
}
return true;
}
string next_word(istream& in) {
ostringstream out;
skip_whitespace(in);
slurp_word(in, out);
skip_whitespace(in);
skip_comment(in);
return out.str();
}
void slurp_word(istream& in, ostream& out) {
char c;
if (!in.eof() && in.peek() == ',') {
in >> c;
out << c;
return;
}
while (in >> c) {
if (isspace(c) || c == ',') {
in.putback(c);
break;
}
out << c;
}
}
void skip_whitespace(istream& in) {
while (!in.eof() && isspace(in.peek()) && in.peek() != '\n') {
in.get();
}
}
void skip_whitespace_and_comments(istream& in) {
while (true) {
if (in.eof()) break;
if (isspace(in.peek())) in.get();
else if (in.peek() == '#') skip_comment(in);
else break;
}
}
void skip_comment(istream& in) {
if (!in.eof() && in.peek() == '#') {
in.get();
while (!in.eof() && in.peek() != '\n') in.get();
}
}
void skip_comma(istream& in) {
skip_whitespace(in);
if (!in.eof() && in.peek() == ',') in.get();
skip_whitespace(in);
}
:(before "End Globals")
bool Disable_redefine_warnings = false;
:(before "End Setup")
Disable_redefine_warnings = false;
:(code)
bool warn_on_redefine(const string& recipe_name) {
if (Disable_redefine_warnings) return false;
return true;
}
:(before "End Globals")
bool Show_rest_of_stream = false;
:(code)
void show_rest_of_stream(istream& in) {
if (!Show_rest_of_stream) return;
cerr << '^';
char c;
while (in >> c) {
cerr << c;
}
cerr << "$\n";
exit(0);
}
:(before "End Globals")
vector<recipe_ordinal> recently_added_recipes;
long long int Reserved_for_tests = 1000;
:(before "End Setup")
for (long long int i = 0; i < SIZE(recently_added_recipes); ++i) {
if (recently_added_recipes.at(i) >= Reserved_for_tests)
Recipe_ordinal.erase(Recipe[recently_added_recipes.at(i)].name);
Recipe.erase(recently_added_recipes.at(i));
}
recently_added_recipes.clear();
:(scenario parse_comment_outside_recipe)
recipe f1 [ ]
recipe main [
1:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(scenario parse_comment_amongst_instruction)
recipe main [
1:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(scenario parse_comment_amongst_instruction_2)
recipe main [
1:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(scenario parse_comment_amongst_instruction_3)
recipe main [
1:number <- copy 23
2:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "2", properties: ["2": "number"]}
:(scenario parse_comment_after_instruction)
recipe main [
1:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(scenario parse_label)
recipe main [
+foo
]
+parse: label: +foo
:(scenario parse_dollar_as_recipe_name)
recipe main [
$foo
]
+parse: instruction: $foo
:(scenario parse_multiple_properties)
recipe main [
1:number <- copy 23/foo:bar:baz
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal", "foo": "bar":"baz"]}
+parse: product: {name: "1", properties: ["1": "number"]}
:(scenario parse_multiple_products)
recipe main [
1:number, 2:number <- copy 23
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: product: {name: "1", properties: ["1": "number"]}
+parse: product: {name: "2", properties: ["2": "number"]}
:(scenario parse_multiple_ingredients)
recipe main [
1:number, 2:number <- copy 23, 4:number
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: ingredient: {name: "4", properties: ["4": "number"]}
+parse: product: {name: "1", properties: ["1": "number"]}
+parse: product: {name: "2", properties: ["2": "number"]}
:(scenario parse_multiple_types)
recipe main [
1:number, 2:address:number <- copy 23, 4:number
]
+parse: instruction: copy
+parse: ingredient: {name: "23", properties: ["23": "literal"]}
+parse: ingredient: {name: "4", properties: ["4": "number"]}
+parse: product: {name: "1", properties: ["1": "number"]}
+parse: product: {name: "2", properties: ["2": "address":"number"]}
:(scenario parse_properties)
recipe main [
1:number:address/lookup <- copy 23
]
+parse: product: {name: "1", properties: ["1": "number":"address", "lookup": ]}
:(code)
void test_parse_comment_terminated_by_eof() {
Trace_file = "parse_comment_terminated_by_eof";
load("recipe main [\n"
" a:number <- copy 34\n"
"]\n"
"# abc");
cerr << ".";
}
:(scenario warn_on_redefine)
% Hide_warnings = true;
recipe main [
1:number <- copy 23
]
recipe main [
1:number <- copy 24
]
+warn: redefining recipe main
:(scenario redefine_without_warning)
% Hide_warnings = true;
recipe main [
1:number <- copy 23
]
recipe! main [
1:number <- copy 24
]
-warn: redefining recipe main
$warn: 0