human theft now reads from xml rules file
This commit is contained in:
parent
248592dbff
commit
8f7f173108
@ -21,6 +21,10 @@ desktop_mode = False
|
|||||||
app = flask.Flask("seagull-game", root_path=path_appdir)
|
app = flask.Flask("seagull-game", root_path=path_appdir)
|
||||||
orig_url_for = app.url_for
|
orig_url_for = app.url_for
|
||||||
|
|
||||||
|
xml_namespaces = {
|
||||||
|
"items": "seagull:rules/items"
|
||||||
|
}
|
||||||
|
|
||||||
#REDIS_HOST="stub-implementation.example.net"
|
#REDIS_HOST="stub-implementation.example.net"
|
||||||
#REDIS_PORT=6379
|
#REDIS_PORT=6379
|
||||||
#REDIS_USER="seagull"
|
#REDIS_USER="seagull"
|
||||||
|
@ -1,9 +1,14 @@
|
|||||||
import os
|
import os
|
||||||
|
import random
|
||||||
import subprocess
|
import subprocess
|
||||||
import xml.etree.ElementTree as xmltree
|
|
||||||
|
import lxml.etree as xmltree
|
||||||
|
|
||||||
from . import core
|
from . import core
|
||||||
|
|
||||||
|
# NOTE: due to how XML libraries handle namespaces, you have to prepend "{seagull:rules/items}" to every tag value
|
||||||
|
# this is merely the price you pay for editor autocompletions
|
||||||
|
|
||||||
valid_resources = [
|
valid_resources = [
|
||||||
"food", "shinies", "psi" # early game
|
"food", "shinies", "psi" # early game
|
||||||
]
|
]
|
||||||
@ -11,7 +16,11 @@ valid_resources = [
|
|||||||
rant_env = os.environ.copy()
|
rant_env = os.environ.copy()
|
||||||
rant_env["RANT_MODULES_PATH"] = (core.path_appdir / "rant").as_posix()
|
rant_env["RANT_MODULES_PATH"] = (core.path_appdir / "rant").as_posix()
|
||||||
|
|
||||||
def generate_item(resource, target):
|
fd_item_schema = xmltree.parse(core.path_appdir / "rules/schemas/items.xsd")
|
||||||
|
item_schema = xmltree.XMLSchema(fd_item_schema)
|
||||||
|
item_schema_parser = xmltree.XMLParser(schema=item_schema)
|
||||||
|
|
||||||
|
def generate_item_description(resource, target):
|
||||||
if core.desktop_mode:
|
if core.desktop_mode:
|
||||||
rant_path = core.path_appdir / "opt/rant/bin/rant"
|
rant_path = core.path_appdir / "opt/rant/bin/rant"
|
||||||
else:
|
else:
|
||||||
@ -21,6 +30,26 @@ def generate_item(resource, target):
|
|||||||
core.log.warning("rant is throwing up:\n" + proc_rant.stderr.decode())
|
core.log.warning("rant is throwing up:\n" + proc_rant.stderr.decode())
|
||||||
return proc_rant.stdout.decode().strip()
|
return proc_rant.stdout.decode().strip()
|
||||||
|
|
||||||
|
def generate_item_list(resource, target, min, max, storybeat=0):
|
||||||
|
count = random.randint(min, max)
|
||||||
|
result = []
|
||||||
|
rulefile = xmltree.parse(core.path_appdir / f"rules/items/{target}.xml", item_schema_parser)
|
||||||
|
ruleset = rulefile.getroot()
|
||||||
|
resource_rules = []
|
||||||
|
for res_rule in ruleset.iter(f"{{seagull:rules/items}}{resource.title()}"):
|
||||||
|
if int(res_rule.get("StoryBeat", "0")) > storybeat:
|
||||||
|
continue
|
||||||
|
|
||||||
|
mindata = res_rule.xpath("./items:Min", namespaces=core.xml_namespaces)[0]
|
||||||
|
maxdata = res_rule.xpath("./items:Max", namespaces=core.xml_namespaces)[0]
|
||||||
|
resource_rules.append((res_rule, int(mindata.text), int(maxdata.text)))
|
||||||
|
for i in range(0, count):
|
||||||
|
core.log.warning("TODO: we don't know which rule this parses yet")
|
||||||
|
core.log.warning(f"{resource} vs humans: {resource_rules[0]}")
|
||||||
|
result.append(TickItem(resource, round(random.uniform(resource_rules[0][1], resource_rules[0][2]), 2), target))
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
class TickItem(object):
|
class TickItem(object):
|
||||||
def __init__(self, resource, amount, target):
|
def __init__(self, resource, amount, target):
|
||||||
if resource not in valid_resources:
|
if resource not in valid_resources:
|
||||||
@ -29,4 +58,4 @@ class TickItem(object):
|
|||||||
self.resource = resource
|
self.resource = resource
|
||||||
self.amount = amount
|
self.amount = amount
|
||||||
self.target = target
|
self.target = target
|
||||||
self.desc = generate_item(resource, target)
|
self.desc = generate_item_description(resource, target)
|
@ -47,8 +47,8 @@ def tick():
|
|||||||
case 10: # ENCHUMAN
|
case 10: # ENCHUMAN
|
||||||
result["items"] = {
|
result["items"] = {
|
||||||
# TODO: read ranges from XML rule files
|
# TODO: read ranges from XML rule files
|
||||||
"food": [items.TickItem("food", round(random.uniform(0.0, 20.0), 2), "humans") for i in range(random.randint(0, 3))],
|
"food": items.generate_item_list("food", "humans", 0, 2),
|
||||||
"shinies": [items.TickItem("shinies", round(random.uniform(0.0, 20.0), 2), "humans") for i in range(random.randint(0, 3))]
|
"shinies": items.generate_item_list("shinies", "humans", 0, 2)
|
||||||
}
|
}
|
||||||
case _:
|
case _:
|
||||||
core.log.warning("undefined tick: {0}".format(result["event_type"]))
|
core.log.warning("undefined tick: {0}".format(result["event_type"]))
|
||||||
|
@ -47,7 +47,7 @@
|
|||||||
##
|
##
|
||||||
|
|
||||||
{
|
{
|
||||||
a piece of `[pick: <wordlist/nouns/cheese>] |
|
a piece of `[pick: <wordlist/nouns/cheese>] cheese |
|
||||||
a `{
|
a `{
|
||||||
[if: [maybe]{[desc_food]}] [get_entree] |
|
[if: [maybe]{[desc_food]}] [get_entree] |
|
||||||
[pick: <wordlist/nouns/fruit>]
|
[pick: <wordlist/nouns/fruit>]
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
Flask==3.1.1
|
Flask==3.1.1
|
||||||
gevent==25.5.1
|
gevent==25.5.1
|
||||||
hiredis>=1.0.0
|
lxml>=6.0.0
|
||||||
redis==6.2.0
|
|
@ -1,16 +1,15 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<ItemRules xmlns="seagull:rules/items">
|
<ItemRules xmlns="seagull:rules/items">
|
||||||
<Event>ENCHUMAN</Event>
|
|
||||||
<Food>
|
<Food>
|
||||||
<Min>0</Min>
|
<Min>1</Min>
|
||||||
<Max>10</Max>
|
<Max>10</Max>
|
||||||
</Food>
|
</Food>
|
||||||
<Shinies>
|
<Shinies>
|
||||||
<Min>0</Min>
|
<Min>1</Min>
|
||||||
<Max>20</Max>
|
<Max>20</Max>
|
||||||
</Shinies>
|
</Shinies>
|
||||||
<Food StoryBeat="3">
|
<Food StoryBeat="3">
|
||||||
<Min>0</Min>
|
<Min>1</Min>
|
||||||
<Max>20</Max>
|
<Max>20</Max>
|
||||||
</Food>
|
</Food>
|
||||||
<Shinies StoryBeat="3">
|
<Shinies StoryBeat="3">
|
||||||
|
23
app/rules/items/seagulls.xml
Normal file
23
app/rules/items/seagulls.xml
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ItemRules xmlns="seagull:rules/items">
|
||||||
|
<Food>
|
||||||
|
<Min>1</Min>
|
||||||
|
<Max>5</Max>
|
||||||
|
</Food>
|
||||||
|
<Shinies>
|
||||||
|
<Min>1</Min>
|
||||||
|
<Max>10</Max>
|
||||||
|
</Shinies>
|
||||||
|
<Food StoryBeat="3">
|
||||||
|
<Min>5</Min>
|
||||||
|
<Max>20</Max>
|
||||||
|
</Food>
|
||||||
|
<Shinies StoryBeat="3">
|
||||||
|
<Min>5</Min>
|
||||||
|
<Max>50</Max>
|
||||||
|
</Shinies>
|
||||||
|
<Psi StoryBeat="3">
|
||||||
|
<Min>0</Min>
|
||||||
|
<Max>15</Max>
|
||||||
|
</Psi>
|
||||||
|
</ItemRules>
|
@ -3,7 +3,6 @@
|
|||||||
<xs:element name="ItemRules">
|
<xs:element name="ItemRules">
|
||||||
<xs:complexType>
|
<xs:complexType>
|
||||||
<xs:choice maxOccurs="unbounded" minOccurs="0">
|
<xs:choice maxOccurs="unbounded" minOccurs="0">
|
||||||
<xs:element type="xs:string" name="Event"/>
|
|
||||||
<xs:element name="Food">
|
<xs:element name="Food">
|
||||||
<xs:complexType>
|
<xs:complexType>
|
||||||
<xs:sequence>
|
<xs:sequence>
|
||||||
|
@ -7,6 +7,7 @@ a = Analysis(
|
|||||||
binaries=[],
|
binaries=[],
|
||||||
datas=[
|
datas=[
|
||||||
('app/templates', './templates'),
|
('app/templates', './templates'),
|
||||||
|
('app/rules', './rules'),
|
||||||
('static', './static'),
|
('static', './static'),
|
||||||
('app/rant', './rant'),
|
('app/rant', './rant'),
|
||||||
('opt', './opt')
|
('opt', './opt')
|
||||||
|
@ -88,8 +88,8 @@ function update_ui() {
|
|||||||
page_elements["lbl_name"].innerHTML = gamestate["name"];
|
page_elements["lbl_name"].innerHTML = gamestate["name"];
|
||||||
page_elements["lbl_tick"].innerHTML = gamestate["tick"];
|
page_elements["lbl_tick"].innerHTML = gamestate["tick"];
|
||||||
page_elements["lbl_colony"].innerHTML = gamestate["colony"];
|
page_elements["lbl_colony"].innerHTML = gamestate["colony"];
|
||||||
page_elements["lbl_shinies"].innerHTML = gamestate["shinies"];
|
page_elements["lbl_shinies"].innerHTML = gamestate["shinies"].toFixed(2);
|
||||||
page_elements["lbl_food"].innerHTML = gamestate["food"];
|
page_elements["lbl_food"].innerHTML = gamestate["food"].toFixed(2);
|
||||||
page_elements["lbl_class"].innerHTML = gamestate["class"];
|
page_elements["lbl_class"].innerHTML = gamestate["class"];
|
||||||
page_elements["lbl_xp"].innerHTML = gamestate["xp"];
|
page_elements["lbl_xp"].innerHTML = gamestate["xp"];
|
||||||
page_elements["lbl_xp_next"].innerHTML = gamestate["xp_next"];
|
page_elements["lbl_xp_next"].innerHTML = gamestate["xp_next"];
|
||||||
@ -118,13 +118,25 @@ function dev_toolbox(open) {
|
|||||||
dev_toolbox_open = open;
|
dev_toolbox_open = open;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function steal_resource(resource, amount, items) {
|
function reward_xp(amount) {
|
||||||
|
gamestate["xp"] += amount;
|
||||||
|
if (gamestate["xp"] >= gamestate["xp_next"]) {
|
||||||
|
old_xp_next = gamestate["xp_next"];
|
||||||
|
gamestate["xp"] -= old_xp_next;
|
||||||
|
gamestate["level"] += 1;
|
||||||
|
gamestate["xp_next"] = (old_xp_next * 1.5) + (gamestate["level"] * 5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function steal_resource(resource, amount, itemstr) {
|
||||||
|
var items = itemstr.split(",")
|
||||||
var stealdata = await fetch("/act/steal/" + resource, {method: "POST", body: JSON.stringify({gamestate: gamestate})})
|
var stealdata = await fetch("/act/steal/" + resource, {method: "POST", body: JSON.stringify({gamestate: gamestate})})
|
||||||
.then(res => { return res.json(); })
|
.then(res => { return res.json(); })
|
||||||
.catch(e => { throw e; });
|
.catch(e => { throw e; });
|
||||||
|
|
||||||
if (stealdata["success"] && amount > 0) {
|
if (stealdata["success"] && amount > 0) {
|
||||||
gamestate[resource] += amount;
|
gamestate[resource] += amount;
|
||||||
|
reward_xp(2);
|
||||||
record_log("Stole " + resource + " from a human: " + items.join(", "));
|
record_log("Stole " + resource + " from a human: " + items.join(", "));
|
||||||
}
|
}
|
||||||
else { record_log("Didn't steal " + resource + " from a human"); }
|
else { record_log("Didn't steal " + resource + " from a human"); }
|
||||||
@ -164,24 +176,25 @@ async function game_tick() {
|
|||||||
tickdata.items.food.forEach((item) => {
|
tickdata.items.food.forEach((item) => {
|
||||||
total_food += item["amount"];
|
total_food += item["amount"];
|
||||||
food_descs.push(item["desc"]);
|
food_descs.push(item["desc"]);
|
||||||
})
|
});
|
||||||
tickdata.items.shinies.forEach((item) => {
|
tickdata.items.shinies.forEach((item) => {
|
||||||
total_shinies += item["amount"];
|
total_shinies += item["amount"];
|
||||||
shinies_descs.push(item["desc"]);
|
shinies_descs.push(item["desc"]);
|
||||||
})
|
});
|
||||||
|
|
||||||
var logstring = "You have encountered a human. It is carrying these resources:\n\n"
|
var logstring = "You have encountered a human. It is carrying these resources:\n\n"
|
||||||
logstring += "<ol>\n"
|
logstring += "<ol>\n"
|
||||||
if (total_food > 0) {
|
if (total_food > 0) {
|
||||||
logstring += `<li><b>${total_food} food:</b> ${food_descs.join(", ")}</li>\n`;
|
logstring += `<li><b>${total_food.toFixed(2)} food:</b> ${food_descs.join(", ")}</li>\n`;
|
||||||
}
|
}
|
||||||
if (total_shinies > 0) {
|
if (total_shinies > 0) {
|
||||||
logstring += `<li><b>${total_shinies} shinies:</b> ${shinies_descs.join(", ")}</li>\n`;
|
logstring += `<li><b>${total_shinies.toFixed(2)} shinies:</b> ${shinies_descs.join(", ")}</li>\n`;
|
||||||
}
|
}
|
||||||
logstring += "</ol>\nWhat would you like to do?";
|
logstring += "</ol>\nWhat would you like to do?";
|
||||||
|
|
||||||
record_log_with_choices(logstring,
|
record_log_with_choices(logstring,
|
||||||
"Steal food", `steal_resource('food', ${total_food}, ${JSON.stringify(food_descs)})`,
|
"Steal food", `steal_resource('food', ${total_food}, "${food_descs.toString()}")`,
|
||||||
"Steal shinies", `steal_resource('shinies', ${total_shinies}, ${JSON.stringify(shinies_descs)})`
|
"Steal shinies", `steal_resource('shinies', ${total_shinies}, "${shinies_descs.toString()}")`
|
||||||
)
|
)
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -190,14 +203,14 @@ async function game_tick() {
|
|||||||
total_food += item["amount"];
|
total_food += item["amount"];
|
||||||
food_descs.push(item["desc"]);
|
food_descs.push(item["desc"]);
|
||||||
})
|
})
|
||||||
steal_resource("food", total_food, food_descs);
|
steal_resource("food", total_food, food_descs.toString());
|
||||||
break;
|
break;
|
||||||
case "steal-shinies":
|
case "steal-shinies":
|
||||||
tickdata.items.shinies.forEach((item) => {
|
tickdata.items.shinies.forEach((item) => {
|
||||||
total_shinies += item["amount"];
|
total_shinies += item["amount"];
|
||||||
shinies_descs.push(item["desc"]);
|
shinies_descs.push(item["desc"]);
|
||||||
})
|
})
|
||||||
steal_resource("shinies", total_shinies, shinies_descs);
|
steal_resource("shinies", total_shinies, shinies_descs.toString());
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
console.error("undefined action " + page_elements["menu_enc_human"]);
|
console.error("undefined action " + page_elements["menu_enc_human"]);
|
||||||
@ -249,13 +262,15 @@ target.addEventListener(start_event, function (ev) {
|
|||||||
page_elements["menu_enc_human"] = document.querySelector("#menu-enc-human");
|
page_elements["menu_enc_human"] = document.querySelector("#menu-enc-human");
|
||||||
page_elements["menu_enc_seagull"] = document.querySelector("#menu-enc-seagull");
|
page_elements["menu_enc_seagull"] = document.querySelector("#menu-enc-seagull");
|
||||||
|
|
||||||
prepare_gamestate().then(update_ui());
|
prepare_gamestate();
|
||||||
|
|
||||||
record_log("seagull game ver. " + ver_string);
|
record_log("seagull game ver. " + ver_string);
|
||||||
|
|
||||||
const interval = setInterval(() => {
|
const interval = setInterval(() => {
|
||||||
if (tick_meter_running) { game_tick(); }
|
if (tick_meter_running) { game_tick(); }
|
||||||
}, 1200);
|
}, 1200);
|
||||||
|
|
||||||
|
update_ui();
|
||||||
});
|
});
|
||||||
|
|
||||||
function change_seagull_name() {
|
function change_seagull_name() {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user