ruby-rdf / spira

Spira is a framework for viewing RDF data as model objects
http://rubygems.org/gems/spira
The Unlicense
58 stars 36 forks source link

How to define type later? (on instance level instead of class level) #34

Closed tmaier closed 10 years ago

tmaier commented 10 years ago

You have the following sample in your readme:

class Man < Spira::Base
  type RDF::URI.new('http://example.org/people/father')
  type RDF::URI.new('http://example.org/people/cop')
end

So when I understand this correctly, all men instantiated with m = Man.new are cops and fathers.

But what when m is a fire fighter? Do I need to set up another class ManFireFighter? Or when he gets retired. Do I need to get his attributes and instantiate a new class of ManRetiredFireFighterAndGrandfather?

As this does not seem feasible, I ask myself how to set the type on instance level?

abrisse commented 10 years ago

Hi @tmaier.

Indeed in that example (which should be changed btw since not very clear) each Man would be a cop and a father.

You just need to create a new class FireFighter actually with the attributes. Then:

uri = 'http://example.org/4563'

person = Man.for(uri)
person.firstName = 'John'
person.save!

person = FireFighter.for(uri)
person.level = 5
person.save!

Each save will add the RDF.type statements into the repository.

cordawyn commented 10 years ago

Think of Spira resources as being "reflections" of your storage - if you have some type declared in the RDF storage, you are expected to have a corresponding Ruby class in Spira, and vice versa. So adding/replacing types is about actually adding/replacing individual Spira types (Ruby classes).

As a side note, I've been pondering the implementation of "automagically" declaring Ruby classes based on the data in the RDF storage, to keep things "synchronized". However, there are certain issues with that, so we've been leaving that to be decided by Spira users' for their projects individually, so to say.

Robsteranium commented 10 years ago

(I've deleted a previous commented about manipulating the resource's @types set as I've since realised this operates on the singleton_class and therefore affects all instances)

I think there are two issues here: setting type(s) at runtime and removing types from instances.

You can set type at runtime with @abrisse's answer and a bit of metaprogramming:

klass = Class.new(Spira::Base) do
  type RDF::URI("http://example.org/people/firefighter")
end

klass.for(m.uri).save

Removing types doesn't appear to be possible without modifying the repository graph directly:

Spira.repository.delete([m.uri, RDF::URI.new("http://www.w3.org/1999/02/22-rdf-syntax-ns#type"), RDF::URI.new("http://example.org/people/cop")])

Which will remove the offending statement from the repository, but m.types will still report it (even after m.reload), and m.save will add the statement back into the repository.

If the models have unique types/ predicates then it's possible to remove types with model.destroy:

class Man < Spira::Base
  type RDF::URI.new('http://example.org/people/man')
end

class FireFighter < Spira::Base
  type RDF::URI.new('http://example.org/people/fire_fighter')
end

class Cop < Spira::Base
  type RDF::URI.new('http://example.org/people/cop')
end

uri = 'http://example.org/4563'
m = Man.for(uri).save           # uri a people:man
c = Cop.for(uri).save           # uri a people:man, people:cop

# retires
c.destroy                       # uri a people:man

# new career             
f = FireFighter.for(uri).save   # uri a people:man, people:fire_fighter

But if the types of predicates overlap then they'll be lost:

class PoliceMan < Spira::Base
  type RDF::URI.new('http://example.org/people/man')
  type RDF::URI.new('http://example.org/people/cop')
end

another_uri = 'http://example.org/999'
p = PoliceMan.for(uri).save     # another_uri a people:man, people:cop
p.destroy                       # another_uri has no type

Although it's possible to reinstate them, of course

Man.for(another_uri).save       # another_uri a people:man

Of course the fact that this sort of manipulation is awkward is probably inevitable given that object and graph models are so very different in the first place!

abrisse commented 10 years ago

@Robsteranium: You really shouldn't use more than one type per class. You will never find an ontology with an owl:Class which is just a union of 2 others.

I like to compare owl:Classes and Ruby Mixins : they both add new capacities/attributes on an instance.