If you are around software development you've heard of IDEs (integrated development environment). You probably use one or know someone who does.
I'd like to contrast IDEs (integrated) with iDEs (integrating).
"Integrated" connotes finality. As in "it has already been integrated for you -- you just use it now."
Of course IDEs are extensible but I only know one person that has made an extension/plug-in for an IDE. On the other hand, I know many vim and emacs users that have made extensions for their iDE. No, they don't call their thing an iDE but they do use it like an iDE.
The spirit of an iDE is: using escape hatches to weave together tools that do their thing well. That probably explains why you might not hear about iDEs... maybe they don't deserve a name but I needed to give this blog post a title.
It might just be that the people that select an IDE do so because it is already integrated (and they just want to work within it) while vim and emacs users are inclined to shave the yak a little.
The first IDE I used was Eclipse. I didn't choose it though. That was the IDE my team was using to develop and maintain some software and all the setup guides for our codebase were Eclipse oriented. So I used it as I wasn't prepared to roll my own approach yet.
Long story short, I never enjoyed using it. I did what I had to in it and used the command line (custom scripts, pipelines, etc.) for everything else.
Fast forward to today -- I now adapt my iDE to whatever project/codebase I need to work on. For my day job one of things things I get to do is develop and maintain knowledge graphs. That requires that I modify RDF and SPARQL files.
I want to show you my iDE for the semantic web stack (SPARQL/RDF).
Since my iDE is mostly a patchwork of tools that work easily and well together I've found that it is easier to get tools to work together if they can communicate using text. That can be in the form of files on the filesystem, command line options, keyboard input, or text-based network protocols (like REST).
In order to demonstrate this workflow I've set up some RDF files in the example/
directory.
Here are the elements of my semantic web iDE:
-
Named terminal sessions
- terminal multiplexer (tmux)
- I typically have a session for editing RDF files, editing and running SPARQL queries, and I usually keep a session for each serivice I am running (Apache Jena Fuseki, SPARQL Anything, Protege, etc.)
-
Editing, viewing, syntax highlighting/checking, execution
- vim
- Note that I get asynchronous SPARQL query execution in vim so I can keep editing queries while running a big query
- Apache Jena's riot command
- vim
-
RDF (e.g. turtle file) formatting
- rdf-toolkit
- This has the benefit that everyone on your team can use their favorite RDF editor and you can still have minimal diffs (when using version control)
-
SPARQL query formatting and execution
- Apache Jena
- Note: I've done the same thing with other triple stores (RDFox, GraphDB, Stardog, etc.)
- Even if I am not using Jena as the triplestore I still always use the Jena java library and the command line utilities.
- Apache Jena
-
RDF navigation (jumping to RDF resources)
- ctags
- It's been around a long time and is supported by many tools.
- ctags
-
Analyzing/manipulating query results
- VisiData
- I highly recommend learning VisiData -- I used this tutorial.
- VisiData
First the demonstration then I'll describe the setup below.
I thought it would be easier show the workflow with a video and narration.
When I want to see a tree view of a class hierarchy or taxonomy I often use Protege but for skos based taxonomies I first have to transform to it to a rdfs:subClassOf
style hierarchy.
I started on a visidata based tree viewer but I need to finish it.
When I need to produce a pictorial graph I often use my former colleague's Turtle Editor Viewer. It might be nice to incorporate that or something like it into my workflow.
I use LSP for some langauges. It would be interesting to think about adding LSP support for RDF and SPARQL files.
Also, the plugin I use for RDF file syntax highlight has support for completion (using prefix.cc) but it didn't work out of the box so I need to tinker with it. In the meantime I just use vim's built-in tab completion.
At the center of it all: the command line, of course.
I assume you are using bash
as the interpreter but I think the commands will still work if zsh
was selected for you.
Also, although I use vim I have a colleague who uses emacs in a similar manner. I bet you could get a similar iDE with several other text editors provided they have sufficient escape hatch capability.
First we need to install some packages (I am assuming a Debian based distro):
sudo apt install tmux git vim-common default-jre universal-ctags python3
Also, I recommend that you install Docker. This is how I package the SPARQL query pretty printer and it is useful for running things like instances of Apache Jena Fuseki, SPARQL Anything, etc. Without Docker, you have to think "do I have the runtime it requires? all the dependencies? the config files? what is this thing's path?"
Set up the vim plugin manager: Vundle There are instructions there in its readme.
And put this into your ~/.vimrc
in the section designated for Vundle:
" for syntax highlighting
Plugin 'niklasl/vim-rdf'
" for running commands asynchronously in vim
Plugin 'https://github.com/skywind3000/asyncrun.vim'
" for quikly commenting on/off blocks of code (in this case in RDF and SPARQL)
Plugin 'tpope/vim-commentary'
In vim, run :PluginInstall
as Vundle requires for new packages.
Download an rdf-toolkit release jarfile.
The current version is here.
In order to make navigation around RDF files possible we need to format (serialize) turtle files in a standard way. rdf-toolkit works well as a serializer.
Here is how I sometimes run it over all the turtle files in my current directory:
ls -1 *ttl | while read IN ; do java -jar ~/Downloads/rdf-toolkit.jar -sdt explicit -dtd -ibn -s $IN > ${IN}_formatted.ttl ; done
Later I'll show you a way to do directly from vim if you prefer.
Since we are serializing using rdf-toolkit we can the widely supported ctags
to jump to URIs (in the same and different files).
cd
to the example/
directory in this repository and run the following to build the index file:
ctags --langdef=turtle --langmap=turtle:.ttl --regex-turtle='/^([<][^>]+[>])/\1/T,urinoprefix/' --regex-turtle='/^([a-z]+:[0-9a-zA-Z_.]+).*/\1/T,uri,a uri/' -V -R .
Then you should see a tags
file in your current directory.
As long as you start vim from that directory the tags index will be used.
Also, you need to tell vim about the lexical syntax of keywords in turtle files.
e.g. URI Qnames have the :
character, full URIs are surrounded by <
and >
, etc.
While we are at the let's set up a few other useful settings for RDF and SPARQL files.
Put this into your ~/.vimrc
:
(NOTE: you'll need to look at the raw markdown file to copy this vim configuration from.)
" set the filetype to 'sparql' for .sparql and .rq
au BufRead,BufNewFile *.{sparql,rq} setfiletype sparql
au BufRead,BufNewFile *.{turtle,ttl} setfiletype turtle
" set up comment characters
au FileType turtle set commentstring=#%s
au FileType sparql set commentstring=#%s
" allow URIs (full and qnames) be to keywords in vim
au FileType turtle set iskeyword=@,48-57,_,192-255,:,-,<,>,/,.
" allow RDF file formatting/syntax checking in vim
" NOTE you'll need the path to where you downloaded and unzipped the Jena utilities
au BufRead *.ttl let &l:equalprg='~/Downloads/apache-jena-5.0.0/bin/riot --formatted=turtle --syntax=turtle -
'
" define a macro for running the SPARQL query pretty printer
au FileType sparql let @p = '?query=
j0v/''/-1/
!docker run --rm -i justin2004/sparql_pretty
:noh
'
" define a macro for executing a SPARQL query and putting the results into visidata
au FileType sparql let @q = 'mm1Gvap"cy''mvap"Cy:split +put\ c buf0
:set buftype=nofile
:file! `mktemp`
:w! /tmp/lala1.sh
:q
''mzz
:AsyncRun tmux split-window bash -l -c "source /tmp/lala1.sh > /tmp/lala1.csv ; vd /tmp/lala1.csv"
'
Now start vim and edit one of the .ttl files in example/
.
Put your cursor on a URI and press ctrl
]
to jump to the definition.
You can get more help on jumping to keywords in vim by running:
:help ^]
or
:help tags
After you jump to several keywords (URIs) you can jump back through the URIs you came through with ctrl
t
.
In vim, you can format (and syntax check) a RDF (turtle) file by going to the first line GG
and then pressing =G
.
That will invoke the equalprg
we defined in .vimrc
on the text in the file.
e.g.
If I do that on a file called some.ttl
with this content:
PREFIX skos: <http://www.w3.org/2004/02/skos/core#>
PREFIX ex: <http://example.com/>
ex:thing55 skos:broader ex:thing34 .
ex:thing57 skos:broader [ skos:prefLabel "apple" .
You'll get:
13:45:02 ERROR riot :: [line: 4, col: 51] Triples not terminated properly in []-list
Since we forgot the closing ]
at the end of the blank node on line 4.
Maybe first find a quick tutorial on tmux. I started with GNU Screen long ago and then upgraded to tmux so I don't have one to recommend but there are several out there.
Create a tmux session called "ontology".
In that session, using vim, open one of the .ttl files in the example/
directory.
Now create a tmux session called "query"
Using vim, create a file (e.g. sparql.sh
) for your SPARQL queries.
At the top of the file put this:
# vim: filetype=sparql
#
# optionally you can include variables like this if you need them:
USER=justin
PASSWORD=your_password_here
#
# the execution macro will prepend these variables definitions to your SPARQL query so you can do variable substitution if you need.
# and here is a query to get you started
#
# NOTE that the macro that executes that depends on seeing `query=` and the ending `'` on a line of its own
curl --silent 'https://query.wikidata.org/sparql' \
--header "Accept: text/csv" \
--data-urlencode 'query=
select *
where {?s ?p ?o}
limit 5
'
Before we use the execution macro let's get VisiData setup.
VisiData has installation instructions here.
You can test to see if it is working by running vd .
to open VisiData on your current directory.
After you do that put this into ~/.visidatarc
:
def get_qname(uri):
# TODO perhaps build this from a .ttl file
PREFIXES={
'http://www.bbc.co.uk/ontologies/bbc/':'bbc',
'http://www.bbc.co.uk/ontologies/coreconcepts/':'core',
'http://www.bbc.co.uk/ontologies/creativework/':'cwork',
'http://purl.org/dc/elements/1.1/':'dc',
'http://purl.org/dc/terms/':'dcterms',
'http://www.w3.org/2002/07/owl#':'owl',
'http://www.bbc.co.uk/ontologies/provenance/':'provenance',
'http://www.w3.org/1999/02/22-rdf-syntax-ns#':'rdf',
'http://www.w3.org/2000/01/rdf-schema#':'rdfs',
'http://www.bbc.co.uk/ontologies/tagging/':'tagging',
'http://www.w3.org/2001/XMLSchema#':'xsd'
}
for prefix in PREFIXES.keys():
if uri.startswith(prefix):
return uri.replace(prefix, PREFIXES[prefix] + ":")
return uri
def to_uri():
TMUX_RDF_SESSION='ontology'
subprocess.run(['tmux','switch-client','-t',TMUX_RDF_SESSION])
uri = get_qname(vd.sheet.cursorCell.value)
subprocess.run(['tmux','send-keys','-t',TMUX_RDF_SESSION,':tsel ' + uri , 'Enter', '1','Enter'])
BaseSheet.addCommand('3', 'go-to-uri', 'to_uri()')
That defines a python function and makes a VisiData command to invoke it and binds the key 3
to it.
You can change the key of course.
Now, put your cursor anywhere on the body of the SPARQL query we put into query.sh
.
To pretty print the query press @p
.
Note that the pretty printing adds a empty line between the prefixes and the select
line.
Unfortunately you'll have to delete that empty line that before you execute the query since the macro depends on the query being wholly within the vim paragraph text object.
That also means you can't have other empty lines in your SPARQL queries.
If I want whitespace I just use a leading #
on those lines.
In vim run this if you want more details on text objects:
:help text-objects
Now to execute the SPARQL query.
With your cursor anywhere in the SPARQL query body, press @q
and the query should get executed and VisiData will open in another tmux pane with the results of the query.
In VisiData when your cursor is on a cell with a URI in it you can press 3
to jump to the tmux "ontology" session and find that keyword.
I hope you find this useful. Perhaps it can inspire you to build an iDE for yourself.
These tools (the command line, tmux, and vim/emacs) and techniques (escape hatchery) can serve you well even if you don't work on the semantic web stack.
Feel free to open an issue or start a discussion on this repository if you need help getting any of this to work or if you have any ideas on how to extend it!
- https://www.bbc.co.uk/ontologies/documents/creativework.ttl
- https://www.bbc.co.uk/ontologies/documents/bbc.ttl
- https://www.dublincore.org/specifications/dublin-core/dcmi-terms/dublin_core_terms.ttl
- http://www.w3.org/2002/07/owl#
- http://www.w3.org/1999/02/22-rdf-syntax-ns#
- http://www.w3.org/2000/01/rdf-schema#