tags |
---|
Reverse Links, Schema Cleanup |
Van Helsing hypnotises Mina, who is now half vampire and can feel Dracula. He asks her questions:
“Where are you now?”
“I do not know. It is all strange to me!”
“What do you see?”
“I can see nothing; it is all dark.”
“What do you hear?”
“The water... and little waves.”
“Then you are on a ship?”
“Oh, yes!”
Now they know that Dracula has escaped on a ship with his last box and is going back to Transylvania. Van Helsing and Mina go to Castle Dracula, while the others go to Varna to try to catch the ship when it arrives. Jonathan Harker just sharpens his knife, and looks like a different person now. All he wants to do is kill Dracula and save his wife. But where is the ship? Every day they wait...and then one day, they get a message: the ship arrived at Galatz up the river, not Varna! Are they too late? They rush off up the river to try to find Dracula.
We have another ship moving around the map in this chapter. Last time, we made a Ship
type that looks like this:
type Ship extending HasNameAndCoffins {
multi link sailors -> Sailor;
multi link crew -> Crewman;
}
That's not bad, but we can build on it a bit more in this chapter. Right now, it doesn't have any information about visits made by which ship to where. Let's make a quick type that contains all the information on ship visits. Each visit will have a link to a Ship
and a Place
, and a cal::local_date
. It looks like this:
type Visit {
required link ship -> Ship;
required link place -> Place;
required property date -> cal::local_date;
}
This new ship that Dracula is on is called the Czarina Catherine
(a Czarina is a Russian queen). Let's use that to insert a few visits from the ships we know. You'll remember that the other ship was called The Demeter and left from Varna towards London.
But first we'll insert a new Ship
and two new places (City
s) so we can link them. We know the name of the ship and that there is one coffin in it: Dracula's last coffin. But we don't know about the crew, so we'll just insert this information:
INSERT Ship {
name := 'Czarina Catherine',
coffins := 1,
};
After that we have the two cities of Varna and Galatz. We'll put them in at the same time:
FOR city IN {'Varna', 'Galatz'}
UNION (
INSERT City {
name := city
}
);
The captain's book from the Demeter has a lot of other places too, so let's look at a few of them. The Demeter also passed through the Bosphorus. That area is the small bit of ocean in Turkey that divides Europe from Asia, so it's not a city. We can use the OtherPlace
type for it, which is also the type we added some annotations to in Chapter 14. Remember how to call them up? It looks like this:
SELECT (INTROSPECT OtherPlace) {
name,
annotations: {
@value
}
};
Let's look at the output to see what we wrote before to make sure that we should use it:
{
schema::ObjectType {
name: 'default::OtherPlace',
annotations: {
schema::Annotation {
@value: 'A place with under 50 buildings - hamlets, small villages, etc.',
},
schema::Annotation {
@value: 'Castles and castle towns do not count! Use the Castle type for that',
},
},
},
}
Well, it's not a castle and it isn't actually a place with buildings, so it should work. Soon we will create a Region
type so that we can have a Country
-> Region
-> City
layout. In that case, OtherPlace
might be linked to from Country
or Region
. But in the meantime we'll add it without being linked to anything:
INSERT OtherPlace {
name := 'Bosphorus'
};
That was easy. Now we can put the ship visits in.
FOR visit IN {
('The Demeter', 'Varna', '1887-07-06'),
('The Demeter', 'Bosphorus', '1887-07-11'),
('The Demeter', 'Whitby', '1887-08-08'),
('Czarina Catherine', 'London', '1887-10-05'),
('Czarina Catherine', 'Galatz', '1887-10-28')
}
UNION (
INSERT Visit {
ship := (SELECT Ship FILTER .name = visit.0),
place := (SELECT Place FILTER .name = visit.1),
date := <cal::local_date>visit.2
}
);
With this data, now our game can have certain ships in cities at certain dates. For example, imagine that a character has entered the city of Galatz. If the date is 28 October 1887, we can see if there are any ships in town:
SELECT Visit {
ship: {
name
},
place: {
name
},
date
} FILTER .place.name = 'Galatz' AND .date = <cal::local_date>'1887-10-28';
And it looks like there is a ship in town! It's the Czarina Catherine.
{
default::Visit {
ship: default::Ship {name: 'Czarina Catherine'},
place: default::City {name: 'Galatz'},
date: <cal::local_date>'1887-10-28',
},
}
While we're doing this, let's practice reverse lookup again on our visits. Here's one:
SELECT Ship.<ship[IS Visit] {
place: {
name
},
ship: {
name
},
date
} FILTER .place.name = 'Galatz';
Ship.<ship[IS Visit]
refers to all the Visits
with link ship
to type Ship
. Because we are selecting Visit
and not Ship
, our filter is now on Visit
's .place.name
instead of the properties inside Ship
.
Here is the output:
{
default::Visit {
place: default::City {name: 'Galatz'},
ship: default::Ship {name: 'Czarina Catherine'},
date: <cal::local_date>'1887-10-28',
},
}
By the way, the heroes of the story found out about the Czarina Catherine thanks to a telegram by a company in the city of Varna that told them. Here's what it said:
28 October.—Telegram. Rufus Smith, London, to Lord Godalming, care H. B. M. Vice Consul, Varna.
“Czarina Catherine reported entering Galatz at one o’clock to-day.”
Remember our Time
type? We made it so that we could enter a string and get some helpful information in return. You can see now that it's almost a function:
type Time {
required property date -> str;
property local_time := <cal::local_time>.date;
property hour := .date[0:2];
property awake := 'asleep' IF <int16>.hour > 7 AND <int16>.hour < 19 ELSE 'awake';
}
Now that we know that the time was one o'clock, let's put that into the query too - including the awake
property. Now it looks like this:
WITH time := (
INSERT Time {
date := '13:00:00'
}
)
SELECT Ship.<ship[IS Visit] {
place: {
name
},
ship: {
name
},
date,
time_ := time {
date,
local_time,
hour,
awake
},
} FILTER .place.name = 'Galatz';
Here's the output, including whether vampires are awake or asleep.
{
default::Visit {
place: default::City {name: 'Galatz'},
ship: default::Ship {name: 'Czarina Catherine'},
date: <cal::local_date>'1887-10-28',
time_: default::Time {
date: '13:00:00',
local_time: <cal::local_time>'13:00:00',
hour: '13',
awake: 'asleep',
},
},
}
It is of course cool that we can do a quick insert in a query like this, but it's a bit weird. The problem is that we now have a random Time
type floating around that is not linked to anything. Instead of that, let's just steal all the properties from Time
to improve the Visit
type instead.
type Visit {
link ship -> Ship;
link place -> Place;
required property date -> cal::local_date;
property time -> str;
property local_time := <cal::local_time>.time;
property hour := .time[0:2];
property awake := 'asleep' IF <int16>.hour > 7 AND <int16>.hour < 19 ELSE 'awake';
}
Then update the visit to Galatz to give it a time
:
UPDATE Visit FILTER .place.name = 'Galatz'
SET {
time := '13:00:00'
};
Then we'll use the reverse query again. Let's add a computed property for fun, assuming that it took two hours, five minutes and ten seconds for Arthur to get the telegram. We'll cast the string to a cal::local_time
and then add a duration
to it.
SELECT Ship.<ship[IS Visit] {
place: {
name
},
ship: {
name
},
date,
time,
hour,
awake,
when_arthur_got_the_telegram := (<cal::local_time>.time) + <duration>'2 hours, 5 minutes, 10 seconds'
} FILTER .place.name = 'Galatz';
And now we get all the output that the Time
type gave us before, plus our extra info about when Arthur got the telegram:
{
default::Visit {
place: default::City {name: 'Galatz'},
ship: default::Ship {name: 'Czarina Catherine'},
date: <cal::local_date>'1887-10-28',
time: '13:00:00',
hour: '13',
awake: 'asleep',
when_arthur_got_the_telegram: <cal::local_time>'15:05:10',
},
}
Since we are looking at Place
again, now we can finish up the chapter by filling out the map with the Region
type that we discussed. That's easy:
type Country extending Place {
multi link regions -> Region;
}
type Region extending Place {
multi link cities -> City;
multi link other_places -> OtherPlace;
multi link castles -> Castle;
}
That connects our types based on Place
quite well.
Now let's do a medium-sized entry that has Country
, Region
, and City
all at the same time. We'll choose Germany in 1887 because Jonathan went through there first. It will have:
- One country: Germany,
- Three regions: Prussia, Hesse, Saxony
- Two cities for each region, for six in total: Berlin and Königsberg, Darmstadt and Mainz, Dresden and Leipzig.
Here is the insert:
INSERT Country {
name := 'Germany',
regions := {
(INSERT Region {
name := 'Prussia',
cities := {
(INSERT City {
name := 'Berlin'
}),
(INSERT City {
name := 'Königsberg'
}),
}
}),
(INSERT Region {
name := 'Hesse',
cities := {
(INSERT City {
name := 'Darmstadt'
}),
(INSERT City {
name := 'Mainz'
}),
}
}),
(INSERT Region {
name := 'Saxony',
cities := {
(INSERT City {
name := 'Dresden'
}),
(INSERT City {
name := 'Leipzig'
}),
}
})
}
};
With this nice structure set up, we can do things like select a Region
and see the cities inside it, plus the country it belongs to. To get Country
from Region
we need a reverse lookup:
SELECT Region {
name,
cities: {
name
},
country := .<regions[IS Country] {
name
}
};
With the reverse lookup at the end we have another link between Country
and its property regions
, but now we get the Country
as the output instead. Here's the output:
{
default::Region {
name: 'Prussia',
cities: {default::City {name: 'Berlin'}, default::City {name: 'Königsberg'}},
country: {default::Country {name: 'Germany'}},
},
default::Region {
name: 'Hesse',
cities: {default::City {name: 'Darmstadt'}, default::City {name: 'Mainz'}},
country: {default::Country {name: 'Germany'}},
},
default::Region {
name: 'Saxony',
cities: {default::City {name: 'Dresden'}, default::City {name: 'Leipzig'}},
country: {default::Country {name: 'Germany'}},
},
}
Here is all our code so far up to Chapter 19.
-
How would you display all the
City
names and the names of theRegion
they are in? -
How about the
City
names plus the names of theRegion
and the name of theCountry
they are in?
Up next: The race against time.