tags | leadImage |
---|---|
Constraints, Deleting |
illustration_03.jpg |
In this chapter we are going to start to think about time, as you can see from what Jonathan Harker is doing:
Jonathan Harker has just arrived at Castle Dracula after a ride in the carriage through the mountains. The ride was terrible: there was snow, strange blue fires and wolves everywhere. It was night when he arrived. He meets with Count Dracula, goes inside, and they talk all night. Dracula leaves before the sun rises though, because vampires are hurt by sunlight. Days go by, and Jonathan still doesn't know that he's a vampire. But he does notice something strange: the castle seems completely empty. If Dracula is so rich, where are his servants? Who is making his meals that he finds on the table? But Jonathan finds Dracula's stories of history very interesting, and so far is enjoying his trip.
Now we are completely inside Dracula's castle, so this is a good time to create a Vampire
type. We can extend it from abstract type Person
because that type has name
and places_visited
, which are good for Vampire
too. But vampires are different from humans because they can live forever. One possibility is adding age
to Person
so that all the other types can use it too. Then `Person' would look like this:
abstract type Person {
required property name -> str;
multi link places_visited -> City;
property age -> int16;
}
int16
means a 16 bit (2 byte) integer, which has enough space for -32768 to +32767. That's enough for age, so we don't need the bigger int32
or int64
types which are much larger. We also don't want it to be a required property
, because we don't care about everybody's age.
But we don't want PC
s and NPC
s to live up to 32767 years, so let's remove age
from the abstract Person
type and give it only to Vampire
now. We will think about the other types later. We'll make Vampire
a type that extends Person
, and adds age:
type Vampire extending Person {
property age -> int16;
}
Now we can create Count Dracula. We know that he lives in Romania, but that isn't a city. This is a good time to change the City
type. We'll change the name to Place
and make it an abstract type
, and then City
can extend from it. We'll also add a Country
type that does the same thing. Now they look like this:
abstract type Place {
required property name -> str;
property modern_name -> str;
property important_places -> array<str>;
}
type City extending Place;
type Country extending Place;
We will need to change places_visited
in the Person
type now to be a Place
instead of a City
. After all, characters can visit more places than just cities:
abstract type Person {
required property name -> str;
multi link places_visited -> Place;
}
Now it's easy to make a Country
, just do an insert and give it a name. We'll quickly insert Country
objects for Hungary and Romania:
INSERT Country {
name := 'Hungary'
};
INSERT Country {
name := 'Romania'
};
(By the way, you might have noticed that important_places
is still an array<str>
and would probably be better as a multi link
. That's true, though in this tutorial we never end up using it and it just stays in the schema as an array. If this were a schema for a real game, it would probably either end up turned to a multi link
or removed if we decide we don't need it.)
With these countries added, we are now ready to make Dracula. First we will change places_visited
in Person
from City
to Place
so that it can include many things: London, Bistritz, Hungary, etc. We only know that Dracula has been in Romania, so we can do a quick FILTER
when we select it. When doing this, we put the SELECT
inside ()
brackets. The brackets (parentheses) are necessary to capture the result of the query using SELECT
, that we then use to do something. In other words, the brackets delimit (set the boundaries for) the SELECT
query so that its result can be used as a whole. EdgeDB will do the operation inside the brackets, and then that completed result is given to places_visited
.
INSERT Vampire {
name := 'Count Dracula',
places_visited := (SELECT Place FILTER .name = 'Romania'),
# .places_visited is the result of this SELECT query.
};
The result is {default::Vampire {id: 7f5b25ac-ff43-11eb-af59-3f8e155c6686}}
.
The uuid
there is the reply from the server showing which object was just created and that we were successful.
Let's check if places_visited
worked. We only have one Vampire
object now, so let's SELECT
it:
SELECT Vampire {
places_visited: {
name
}
};
This gives us: {default::Vampire {places_visited: {default::Country {name: 'Romania'}}}}
Perfect.
Now let's think about age
again. It was easy for the Vampire
type, because they can live forever. But now we want to give age
to PC
and NPC
too, who are humans who don't live forever (we don't want them living up to 32767 years). For this we can add a "constraint" (a limit). Instead of age
, we'll give them a new type called HumanAge
. Then we can write constraint
on it and use {ref}one of the functions <docs:ref_datamodel_constraints>
that it can take. We will use max_value()
.
Here's the signature for max_value()
:
std::max_value(max: anytype)
The anytype
part is interesting, because that means it can work on types like strings too. With a constraint max_value('B')
for example you couldn't use 'C', 'D', etc.
Now let's go back to our constraint for HumanAge
, which is 120. The HumanAge
type looks like this:
scalar type HumanAge extending int16 {
constraint max_value(120);
}
Remember, it's a scalar type because it can only have one value. Then we'll add it to the NPC
type.
type NPC extending Person {
property age -> HumanAge;
}
It's our own type with its own name, but underneath it's an int16
that can't be greater than 120. So if we write this, it won't work:
INSERT NPC {
name := 'The innkeeper',
age := 130
};
Here is the error: ERROR: ConstraintViolationError: Maximum allowed value for HumanAge is 120.
Perfect.
Now if we change age
to 30, we get a message showing that it worked: {Object {id: 72884afc-f2b1-11ea-9f40-97b378dbf5f8}}
. Now no NPCs can be over 120 years old.
Deleting in EdgeDB is very easy: just use the DELETE
keyword. It's similar to SELECT
in that you write DELETE
and then the type, which will by default delete them all. And in the same way as SELECT
, if you FILTER
then it will only delete the ones that match the filter.
This similarity to SELECT
might make you nervous, because if you type something like SELECT City
then it will select all of them. DELETE
is the same: DELETE City
deletes every object for the City
type. That's why you should think carefully before deleting anything.
So let's give it a try. Remember our two Country
objects for Hungary and Romania? Let's delete them:
DELETE Country;
We got an error telling us that deleting a Country
is not possible because it is still referenced by places_visited
of a Person
.
ERROR: ConstraintViolationError: deletion of default::Country (7f3c611c-ff43-11eb-af59-dfe5a152a5cb) is prohibited by link target policy
Detail: Object is still referenced in link places_visited of default::Person (7f5b25ac-ff43-11eb-af59-3f8e155c6686).
That's Count Dracula who visited Romania getting in the way. Let's delete him first then:
DELETE Vampire;
Just like an insert, it gives us the id numbers of the objects that are now deleted: {default::Vampire {id: 7f5b25ac-ff43-11eb-af59-3f8e155c6686}}
. Now we can try deleting the Country
objects:
DELETE Country;
We got confirmation that two Country
objects have been deleted:
{
default::Country {id: 7f2da5e6-ff43-11eb-af59-33db995c2682}, default::Country {id: 7f3c611c-ff43-11eb-af59-dfe5a152a5cb},
}
Okay, insert them again. Now let's delete with a filter:
DELETE Country FILTER .name ILIKE '%States%';
Nothing matches, so the output is {}
- we deleted nothing. Let's try again:
DELETE Country FILTER .name ILIKE '%ania%';
We got a {default::Country {id: 7f3c611c-ff43-11eb-af59-dfe5a152a5cb}}
, which is certainly Romania. Only Hungary is left. What if we want to see what we deleted? No problem - just put the DELETE
inside brackets and SELECT
it. Let's delete all the Country
objects again but this time we'll select it:
SELECT (DELETE Country) {
name
};
The output is {default::Country {name: 'Hungary'}}
, showing us that we deleted Hungary. And now if we do SELECT Country
we get a {}
, which confirms that we did delete them all.
(Fun fact: DELETE
statements in EdgeDB are actually {ref}syntactic sugar <docs:ref_eql_statements_delete>
for DELETE (SELECT ...)
. You'll be learning something called LIMIT
in the next chapter with SELECT
and as you do so, keep in mind that you can apply the same to DELETE
too.)
Finally, let's insert Hungary and Romania again to finish the chapter. We'll leave them alone now.
Here is all our code so far up to Chapter 3.
-
This query is trying to display every
NPC
along with thename
plus everyCity
type for eachNPC
, but it's giving an error. What is it missing?SELECT NPC { name, cities := SELECT City.name };
-
If the
City
type needed a required property calledpopulation
, what would it look like? What type would 'population' be? -
This query wants to display
name
twice for some reason but is giving an error. Can you think of a way to do it?SELECT Person { name, name };
(Hint: the problem is that the name
name
is being used twice) -
People keep trying to make characters with negative ages. Can you think of a constraint that can stop this?
Hint: the current constraint is
max_value(120);
-
Can you insert a HumanAge type?
Up next: Jonathan: "This Count Dracula knows so much about history! I'm glad I came."