Bizarre trouble with taking items from a held container

My players have expressed frustration with a pizza box and a money envelope I have. The rules of Inform prohibit taking multiple copies of a ‘kind’ when the container they are in is held by the player.

This is the ‘exclude indirect possessions’ rule. Removing it causes bizarre behavior.

So, here’s a simple example:

"Sandbox" by Brian Rushton.

Parlor is a room.

The closet door is a door. It is north from parlor and south from Elsewhere.

The delivery box is a closed openable container. It is held by the player.

A pizza slice is a kind of thing. There are five pizza slices in the delivery box. 

The exclude indirect possessions from take all rule is not listed in the for deciding whether all includes rulebook.

The parser nothing error internal rule response (D) is "[The noun] [can't] contain things."

This gives the following gameplay:

>open box
You open the delivery box, revealing five pizza slices.
 
>take slices
The closet door can't contain things.
 
>get all slices
There are none at all available!
 
>get all from box
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
 
> 

Edit: for some reason I can’t get the above transcript to happen again with the same code, but it can work (kind of) if typing TAKE SLICES twice.

Any clue what’s going on? This is in Inform 7, 6M62. If it’s a bug and already fixed in Inform 10, then I’ll leave it alone. But if there’s a way to fix it in Inform 7, 6M62, I’d love to know!

1 Like

Okay, it’s even weirder. Even if I don’t get rid of that rule, it still messes up:

"Sandbox" by Brian Rushton.

Parlor is a room.

The closet door is a door. It is north from parlor and south from Elsewhere.

The delivery box is a closed openable container. It is held by the player.

A pizza slice is a kind of thing. There are five pizza slices in the delivery box. 

The parser nothing error internal rule response (D) is "[The noun] [can't] contain things."

This gives similar results:

>take slices
The closet door can't contain things.

>get all from box
There are none at all available!

>get all slices
There are none at all available!

Edit:

The closet door doesn’t have to be a door, apparently it happens for any scenery object. So I tried making the closet an open container:

"Sandbox" by Brian Rushton.

Parlor is a room.

The closet is an open container in parlor.

The delivery box is a closed openable container. It is held by the player.

A pizza slice is a kind of thing. There are five pizza slices in the delivery box.

and got this result:

>open box

You open the delivery box, revealing five pizza slices.

>take slices

The closet is empty.

>showme closet

closet - open container

location: in Parlor

singular-named, improper-named; unlit, inedible, portable; opaque, open, unopenable, unlocked

list grouping key: none

printed name: "closet"

printed plural name: "containers"

indefinite article: none

description: none

initial appearance: none

carrying capacity: 100

>

It says the closet is empty but SHOWME CLOSET shows it’s open!

Editedit:

Looks like it’s connected to behavior described here:

My guess is it’s defaulting to the ‘remove’ action but interpreting the target of the ‘remove’ action as the held pizza box, since ‘GET SLICES FROM BOX’ works just fine.

1 Like

When using the first code you posted, I get this:

Sandbox
An Interactive Fiction by Brian Rushton
Release 1 / Serial number 231124 / Inform 7 build 6M62 (I6/v6.41 lib 6/12N) SD

Parlor
You can see a closet door here.

>open box
You open the delivery box, revealing five pizza slices.

>take slices
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>undo
Parlor
[Previous turn undone.]

>get all slices
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

I get the same result in 6M62 or the current version.

2 Likes

I just noticed that. Try TAKE SLICES two times in a row to get behavior more like what I’m looking at, something messed up with my transcript.

1 Like

I’m having Baker of Shireton flashbacks: GET ALL DOUGH FROM OVEN.

Could you do a swap? Have a SLICED PIZZA in the box then “instead of taking a slice/pizza” put an off-stage slice in the player’s hand? That’s the kind of kludge I end up doing!

4 Likes

Hmm, that’s not a bad idea!

2 Likes

How about:

[note: doesn't work - see below] 
The exclude indirect possessions from take all rule does nothing when the particular possession is a pizza slice.

You can modify as appropriate with a definition to allow it to apply to multiple kinds.

EDIT: FYI for those coming across this later, there is an unintended side effect of this in that the command >TAKE ALL will have a tendency to remove the pizza slices from the delivery box, which may not be desirable.

EDIT 2: Note that the particular possession is not set correctly for this to work in the generic case, so the approach of trying to suppress the exclude indirect possessions... rule is not workable. However, a new decide whether all includes... rule that takes priority can be specified:

First rule for deciding whether all includes a pizza slice while taking or removing:
	it does.
3 Likes

I think this avoids most of the problems. It just doesn’t cover this one case:

"Sandbox" by Brian Rushton.

Parlor is a room.

The closet is scenery in parlor.

A pizza slice is a kind of thing. There are five pizza slices carried by the player.

The exclude indirect possessions from take all rule does nothing when the particular possession is a pizza slice.
>take slices

That can't contain things.

And what’s worse is the code parsing this is at an Inform 6 level, so doesn’t show up in rule tracing. I can only assume it’s something going wrong here:

if (etype == NOTHING_PE) {
	if (parser_results-->ACTION_PRES == ##Remove &&
    	parser_results-->INP2_PRES ofclass Object) {
    	noun = parser_results-->INP2_PRES; ! ensure valid for messages
        if (noun has animate) { PARSER_N_ERROR_INTERNAL_RM('C', noun); new_line; }
        else if (noun hasnt container or supporter) { PARSER_N_ERROR_INTERNAL_RM('D', noun); new_line; }
        else if (noun has container && noun hasnt open)  { PARSER_N_ERROR_INTERNAL_RM('E', noun); new_line; }
        else if (children(noun)==0) { PARSER_N_ERROR_INTERNAL_RM('F', noun); new_line; } ! <---  HERE
        else parser_results-->ACTION_PRES = 0;
    }
    if (parser_results-->ACTION_PRES ~= ##Remove) {
        if (multi_wanted==100) { PARSER_N_ERROR_INTERNAL_RM('A'); new_line; }
        else                  {  PARSER_N_ERROR_INTERNAL_RM('B'); new_line; }
    }
}

This is reminding me of Jim Aikin’s hamster problem. I think it has a similar cause; it has to do with the “most interesting” error that develops as parsing fails (before the part you posted).

I think it’s fixable, but I won’t have time to try until later. If someone else doesn’t come along before then, I’ll see what I can do.

2 Likes

Thansk! If not, I’ll probably use Hanon’s suggestion. For now I’ve patched it like so:

The parser nothing error internal rule response (D) is "[bracket]Inform has difficulty understanding what you want when grabbing multiple similar items. You may need to specify where you are taking them from, like GET SLICES FROM BOX or GET BILLS FROM ENVELOPE.[close bracket]"

1 Like

One pizza with extra kludge:

Understand "pizza/-- slices" and "pizza" as a pizza slice.

check taking a pizza slice (this is the you can't have all the pizza rule):
    if delivery box does not contain a pizza slice:
        say "The pizza's all gone, dude." instead;
    otherwise:
        let P be a random pizza slice in delivery box;
        try taking P instead;
2 Likes

Just to make sure that I understand what you’re after…

When the player has the box, you want output like:

holding box
>[1] open box
You open the delivery box, revealing five pizza slices.

>[2] take slices
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[3] take slices
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.

>[4] put slices in box
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.

>[5] take slices from box
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[6] take slices from box
You can't see any such thing.

>[7] put slices in box
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.

>[8] take all
There are none at all available!

>[9] take all
There are none at all available!

>[10] take all from box
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

and when the player does not have the box, you want output like:

not holding box
>[1] drop box
Dropped.

>[2] open box
You open the delivery box, revealing five pizza slices.

>[3] take slices
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[4] take slices
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.
pizza slice: You already have that.

>[5] put slices in box
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.

>[6] take slices from box
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[7] take slices from box
You can't see any such thing.

>[8] put slices in box
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.

>[9] take all
delivery box: Taken.

>[10] take all
There are none at all available!

>[11] take all from box
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
2 Likes

Yes, that covers even more use cases than I had anticipated.

OK. There are a lot of things that contribute to this odd behavior, and it took a while to figure out how to thread the needle.

Pizza Delivery II
"Pizza Delivery II"

Parlor is a room.

The closet door is a door. It is north from parlor and south from Elsewhere.

The delivery box is a closed openable container. It is held by the player.

A pizza slice is a kind of thing. There are five pizza slices in the delivery box. 

The plural of pizza slice is slices of pizza.

To decide which object is token-constraining item:
	(- advance_warning -).

To decide whether the noun text names (O - object):
	(- (Refers({O}, match_from)) -).

Rule for deciding whether all includes something (called item) enclosed by the person reaching while taking or removing (this is the revised exclude indirect possessions from take all rule):
	if taking:
		if the noun text names item:
			it does;
		otherwise:
			it does not;
	if removing:
		if the the token-constraining item contains the item:
			it does;
		otherwise:
			it does not.

The revised exclude indirect possessions from take all rule is listed instead of the exclude indirect possessions from take all rule in the for deciding whether all includes rules.

Rule for deciding whether all includes something (called item) contained by a portable container (called encloser) not enclosed by the player:
	if taking or removing:
		unless the noun text names item:
			it does not.

Test me with "open box / take slices / take slices / put slices in box / take slices from box / take slices from box / put slices in box / take all / take all / take all from box".

Test dropped with "drop box / open box / take slices / take slices / put slices in box / take slices from box / take slices from box / put slices in box / take all / take all / take all from box".

[some additional items to test that the above is working generically]

A rock is a kind of thing.

A bucket is here. Five rocks are in the bucket.

Anything more refined (at least, anything more refined that I could come up with) requires I6 hacking on a bigger scale.

If this doesn’t work for you or you find bugs, let me know.

3 Likes

I tried popping it in my code for my long game. I ran through a TEST ME of the entire game, and then tried specifically messing with the pizza, and it worked. Thanks! I’ll include it, and I’ll put you in the credits. I have you in there now as ‘otistdog’; would you prefer that to be changed?

1 Like

Since variations of this question have popped up many times over the years, I will try to explain step-by-step what’s happening when you get the unexpected behavior, i.e. when trying to take the slices even though the player is already carrying them indirectly as the contents of the delivery box.

THE OUT-OF-THE-BOX EXPERIENCE

The grammar lines for the word “take” can be seen using the debugging command >SHOWVERB TAKE, which produces:

Verb 'carry' 'hold' 'take' 
	 * 'inventory' -> Inv
	 * multi -> Take
	 * 'off' noun -> Disrobe
	 * noun 'off' -> Disrobe
	 * multiinside 'from' noun -> Remove
	 * multiinside 'off' noun -> Remove

These all come from the Standard Rules. The output is in I6 style, so you need to do some translating to make sense of it.

When all of the pizza is enclosed by the player and the entered command is >TAKE SLICES, the only grammar lines that matter here are the second, fifth and sixth. The second uses a “multi” token, which comes from the I7 grammar line:

Understand "take [things]" as taking.

The fifth and sixth use a “multiinside” token. They come from the I7 grammar lines:

Understand "take [things inside] from [something]" as removing it from.
Understand "take [things inside] off [something]" as removing it from.

When evaluating the second grammar line, the parser sees the word “slices” in the place where it will be looking for words that match objects. It will correctly identify all pizza slices in scope as matching that word. However, it will also check whether anything being put on the multiple object list belongs there according to the deciding whether all includes rules.

As you’ve seen, the exclude indirect possessions from take all rule is a rule in that rulebook, defined in the Standard Rules as:

Rule for deciding whether all includes things enclosed by the person reaching
	while taking or taking off or removing (this is the exclude indirect
	possessions from take all rule): it does not.

Since every pizza slice is enclosed by the person reaching (where person reaching is another way of specifying actor, used here because this rule is checked outside of action processing), and since the action being parsed is a taking action, every pizza slice is then kept off the multiple object list, leaving it empty.

Coming back from the deciding whether all includes rules, the parser sees that there is nothing left to be a noun and fails the grammar line. However, it’s not done trying grammar lines for the word “take” and moves on. The next two fail, and it continues to the fifth line.

Even though the command that was entered is only >TAKE SLICES, the parser is trying to match a line of the form >TAKE [THINGS] FROM [SOMETHING CONTAINING THOSE THINGS]. Part of what it does for this type of line is a “second noun lookahead” to try figure out what the second noun will be, so as to be able to know whether the nouns are inside of it. Another part of what it does is “inference,” i.e. to try to guess what the player meant when it thinks part of the command is missing.

Here, those two bits of logic interact to make the parser temporarily assume that the player really meant >TAKE SLICES FROM [SOMETHING TO BE DETERMINED]. It then attempts to make a best guess about what that undetermined thing is according to some fairly generic rules. These rules make the parser assume that, because the verb is removing it from, something which is in the location (i.e. “on the floor” of or “directly in” the current room) is more likely to be the correct object.

In your scenario, the only such object is the closet door, so it selects this as the best guess. The parser’s working assumption at this point is that the “real” command intended by the player is >TAKE SLICES FROM CLOSET DOOR, and it helpfully informs you that That can't contain things., where “that” means the closet door.

If the player carries every pizza slice and the command is >TAKE SLICES FROM BOX, then the parser will look at the second grammar line first, as before, and reject every slice due to the exclude indirect possessions... rule, as before. (Arguably these are not “indirect” possessions, but the actual logic uses the enclosure relation, so directly carried things still count.)

When it moves on to the fifth grammar line, however, the second noun lookahead will see that the word “from” appears later in the command input and try to resolve the words following it as an object. It has no trouble matching the word “box” to the delivery box, so it takes note of that object and stores it in an I6 global variable called advance_warning.

It then rewinds to look at the words between “take” and “from” to see if they can be matched, but, because advance_warning is defined as the delivery box, the parser is now checking to make sure that any matching item is in the delivery box. All of the pizza meets this condition, but it is also enclosed by the player, so it is once again excluded by the exclude indirect possessions... rule and the multiple object list ends up empty.

This is a slightly different error state, in that the delivery box was correctly identified as the second noun, and it is a container. The resulting default message is There are none at all available! because the parser didn’t see any of the contents of the box as part of what is “included by all” (i.e. allowed on the multiple object list).

That’s a lot of complexity emerging from what looks like a simple situation. To review, this is the default behavior:

Parlor
You can see a closet door here.

>test me
(Testing.)

>[1] open box
You open the delivery box, revealing five pizza slices.

>[2] take slices
That can't contain things. [parser error]

>[3] take slices
That can't contain things. [same state, same parser error]

>[4] put slices in box
There are none at all available!

>[5] take slices from box
There are none at all available! [parser error]

>[6] take slices from box
There are none at all available! [same state, same parser error]

Compare that with what happens when the player does not hold the pizza box.

Parlor
You can see a closet door here.

>test me
(Testing.)

>[1] drop box
Dropped.

>[2] open box
You open the delivery box, revealing five pizza slices.

>[3] take slices [exclude indirect possessions rule doesn't apply]
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[4] take slices [parser can't infer a second noun]
What do you want to take those things from?

>[5] put slices in box
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.
pizza slice: Done.

>[6] take slices from box [exclude indirect possessions rule doesn't apply]
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.
pizza slice: Taken.

>[7] take slices from box
You can't see any such thing. [parser error]

In the second situation, the first >TAKE SLICES or >TAKE SLICES FROM BOX works as expected because the exclude indirect possessions... rule does not apply; the slices are not enclosed by the player.

The second >TAKE SLICES (command 4) has a new message. That’s because when the parser gets to the inference stage while processing the fifth grammar line there are two non-carried things in the area this time (delivery box and closet door), and the parser can’t decide on a best guess based on default scoring. Accordingly, it assumes the word “from” and just asks for clarification about which item the slices should be taken from. If the player supplies that input as >BOX, then the command becomes >TAKE SLICES FROM BOX, and it works.

The second >TAKE SLICES FROM BOX (command 7) has a default can't see any such thing parser error because, while processing the fifth grammar line, it identifies the delivery box as the advance_warning, but there is nothing that matches the word “slices” that is inside the box. (I think but am not certain that in this case the parser never gets as far as considering the words “slices” to have matched any object, so it reports the error as though the command had been >TAKE LSDJFLSJD or other gibberish.)

5 Likes