GithubHelp home page GithubHelp logo

thelady / solid.js Goto Github PK

View Code? Open in Web Editor NEW

This project forked from solid-contrib/solid-client

0.0 1.0 0.0 822 KB

Javascript client library for Solid applications

License: MIT License

Shell 2.61% JavaScript 87.75% CSS 0.09% HTML 9.55%

solid.js's Introduction

solid.js

NPM Version Build Status

Javascript library for writing Solid applications. (See Changelog for version history.)

Usage

Solid.js is currently intended for client-side use only (inside a web browser):

  1. Load dependencies (currently, rdflib.js).
  2. Load solid.js from a local copy (or directly from Github Pages).
  3. Use the require('solid') function provided by Browserify to import.

Example index.html:

<script src="https://solid.github.io/releases/rdflib.js/rdflib-0.6.0.min.js"></script>
<script src="https://solid.github.io/releases/solid.js/solid-0.14.1.min.js"></script>
<script>
  // $rdf is exported as a global when you load RDFLib, above
  var solid = require('solid')
  // Use Solid client here ...
  console.log('solid.js version: ' + solid.meta.version())
</script>

Take a look at the Solid.js Demo Page (source located in demo/index.html) for more usage examples.

Tutorials

If you would like to learn how to build Solid apps using Solid.js, please see:

Developing Solid.js

Install dev dependencies:

npm install

Building (uses Browserify, builds to dist/solid.js and dist/solid.min.js):

npm run build

Testing

To run the unit tests:

npm test

This runs the Tape unit test suite.

Releases

The following steps specify how to release solid.js:

Make sure you're at the HEAD of master.

$ git checkout master && git pull

Run npm version to bump the package version via git commit and git tags.

# refer to http://semver.org/ for which of (major, minor, patch) to use
$ npm version [major|minor|patch]

Next, push the commit and tags:

$ git push --follow-tags

Finally release the package to npmjs.

$ npm publish

Logging In and User Profiles

Before doing any sort of reading or writing of Solid resources, your app will likely need to authenticate a user and load their profile, so let's start with those sections.

Authentication

Solid currently uses WebID-TLS for authentication, which relies on a web browser's built-in key store to manage certificates and prompt the user to select the correct certificate when accessing a server.

Solid servers must always return a Solid-specific HTTP header called User, which contains the WebID that the user used to access this particular server. An empty header usually means that the user is not authenticated.

Detecting the Current Logged-in User

Most of the WebID-TLS authentication process takes place before a web page gets fully loaded and the javascript code has had a chance to run. Since client-side Javascript code does not have access to most HTTP headers (including the User header) of the page on which it runs, how does an app discover if there is an already authenticated user that is accessing it?

The current best practice answer is -- the app should do an Ajax/XHR HEAD request to the relevant resource:

  1. either to the current page if it's a standalone app, or
  2. to the requested resource (if it's an app that's acting as a viewer or editor, and requires a resource URI as a parameter)

For the first case (standalone apps), Solid.js provides a convenience solid.currentUser() method (which does a HEAD request to the current page in the background). Usage:

solid.currentUser()
  .then(function (currentWebId) {
    if (currentWebId) {
      console.log('Current WebID: %s', currentWebId)
    } else {
      console.log('You are not logged in')
    }
  })

For the second case (apps that are wrapping a resource as viewers or editors), client apps can just use a solid.login(targetUrl) function to return the current user's WebID. And if users are unable to log in, prompt the user to create an account with solid.signup().

Login example

Here is a typical example of authenticating a user and returning their WebID. The following login function, specific to your application, wraps the solid.login() function. If the promise is resolved, then an application will do something with the webId value. Otherwise, if the promise is rejected, the application may choose to display an error message.

HTML:

<a href="#" onclick="login()">Login</a>

Javascript:

var solid = require('solid')
var login = function() {
  // Get the current user
  solid.login().then(function(webId){
    // authentication succeeded; do something with the WebID string
    console.log(webId)
  }).catch(function(err) {
    // authentication failed; display some error message
    console.log(err)
  })
}

Signup example

The signup function is very similar to the login function, wrapping the solid.signup() function. It results in either a WebID or an error message being returned.

HTML:

<a href="#" onclick="signup()">Sign up</a>

Javascript:

var solid = require('solid')
// Signup for a WebID
var signup = function() {
  solid.signup().then(function(webId) {
    // authentication succeeded; do something with the WebID string
    console.log(webId)
  }).catch(function(err) {
    // authentication failed; display some error message
    console.log(err)
  })
}

User Profiles

Once you have a user's WebID (say, from a login() call), it's often useful to load the user profile:

var profile = solid.login()
  .then(function(webId){
    // have the webId, now load the profile
    return solid.getProfile(webId)
  })

The call to getProfile(url) loads the full extended profile: the profile document itself, any sameAs and seeAlso links it finds there, as well as the Preferences file.

User Type Registry Index

If your application needs to do data discovery, it can also call loadTypeRegistry() after loading the profile:

var profile = solid.login()
  .then(function (webId) {
    return solid.getProfile(webId)
  })
  .then(function (profile) {
    return profile.loadTypeRegistry()
  })

Now, both listed and unlisted type indexes are loaded, and you can look up where the user keeps various types.

var vocab = solid.vocab
// .. load profile and load type registry

var addressBookRegistrations = solid.getProfile(webId)
  .then(function (profile) {
    return profile.loadTypeRegistry()
  })
  .then(function (profile) {
    return profile.typeRegistryForClass(vocab.vcard('AddressBook'))
  })
/*
-->
[
  an IndexRegistration(
    locationUri: 'https://localhost:8443/public-contacts/AddressBook.ttl',
    locationType: 'instance',
    isListed: true
  ),
  an IndexRegistration(
    locationUri: 'https://localhost:8443/personal-address-books/',
    locationType: 'container',
    isListed: false
  )  
]
*/

You can then load the resources from the returned locations, as usual.

addressBookRegistrations.forEach(function (registration) {
  if (registration.isInstance()) {
    // load the instance via web.get()
  } else if (registration.isContainer()) {
    // this is a container with many address books, use web.list()
  }
})

Registering (and un-registering) Types in the Type Registry

To register an RDF Class with a user's Type Registry (listed or unlisted), use profile.registerType():

var vocab = solid.vocab
// .. load profile

var classToRegister = vocab.sioc('Post')
var locationToRegister = 'https://localhost:8443/new-posts-container/'
var isListed = true
profile.registerType(classToRegister, locationToRegister, 'container', isListed)
  .then(function (profile) {
    // Now the type is registered, and the profile's type registry is refreshed
    // querying the registry now will include the new container
    profile.typeRegistryForClass(vocab.sioc('Post'))
  })

// To remove the same class from registry:
var classToRemove = vocab.sioc('Post')
profile.unregisterType(classToRemove, isListed)
  .then(function (profile) {
    // Type is removed
    profile.typeRegistryForClass(vocab.sioc('Post'))   // --> []
  })

Web operations

Solid.js uses a mix of LDP and Solid-specific functions to manipulate Web resources. Please see the Solid spec for more details.

Getting information about a resource

Sometimes an application may need to get some useful meta data about a resource. For instance, it may want to find out where the ACL resource is. Clients should take notice to the fact that the solid.web.head() function will always successfully complete, even for resources that don't exists, since that is considered useful information. For instance, clients can use the solidResponse.xhr.status value will indicate whether the resource exists or not.

Here, for example, we can find out where the corresponding ACL resource is for our new blog post hello-world.

var solid = require('solid')
var url = 'https://example.org/blog/hello-world'
solid.web.head(url).then(
  function(solidResponse) {
    console.log(solidResponse.acl) // the ACL uri
    if (!solidResponse.exists()) {
      console.log("This resource doesn't exist")
    } else if (solidResponse.xhr.status === 403) {
      if (solidResponse.isLoggedIn()) {
        console.log("You don't have access to the resource")
      } else {
        console.log("Please authenticate")
      }
    }
  }
)

The SolidResponse object returned by most solid.web calls, including head(), contains the following properties:

  • url - the URL of the resource // https://example.org/blog/hello-world
  • acl - the URL of the corresponding .acl resource // https://example.org/blog/hello-world.acl
  • meta - the URL of the corresponding .meta resource // https://example.org/blog/hello-world.meta
  • types - An array of LDP types for the resource, if applicable. For example: [ 'http://www.w3.org/ns/ldp#LDPResource', 'http://www.w3.org/ns/ldp#Resource' ]
  • user - the WebID of the authenticated user (if authenticated) // https://user.example.org/profile#me
  • websocket - the URI of the corresponding websocket instance // wss://example.org/blog/hello-world
  • method - the HTTP verb (get, put, etc) of the original request that resulted in this response.
  • xhr - the raw XMLHttpRequest object (e.g. xhr.status)

The response object also has some convenience methods:

  • contentType() - returns the MIME type of the resource
  • isContainer() - determines whether the resource is a Container or a regular resource

Fetching a Resource

Assuming that a resource or a container exists (see creating resources and creating containers below), you can retrieve it using web.get():

solid.web.get(url)
  .then(function(response) {
    if (response.isContainer()) {
      // This is an instance of SolidContainer, see Listing Containers below
      for (resourceUrl in response.resources) {
        // iterate over resources
      }
      for (subcontainerUrl in response.containers) {
        // iterate over sub-containers
      }
    } else {
      // Regular resource
      console.log('Raw resource: %s', response.raw())
      
      // You can parse it using RDFLib.js, etc:
      var parsedGraph = response.parsedGraph()
    }
  })
  .catch(function(err) {
      console.log(err) // error object
    // ...
   })

Fetching a Parsed Graph

Once a resource is retrieved, we can access it as a parsed graph (here, parsed by rdflib.js). This graph can then be queried.

var solid = require('solid')
solid.config.parser = 'rdflib'  // 'rdflib' is the default parser
var vocab = solid.vocab

var url = 'https://example.org/blog/hello-world'

solid.web.get(url)
  .then(function(response) {
    var graph = response.parsedGraph()
    // Print all statements matching resources of type foaf:Post
    console.log(graph.statementsMatching(undefined, vocab.rdf('type'),
      vocab.sioc('Post')))
  })
  .catch(function(err) {
    console.log(err) // error object
  })

Creating a Solid Container

The Solid client offers a function called solid.web.createContainer(), which is used to create containers. The function accepts the following parameters:

  • parentDir (string) - the URL of the parent container in which the new resource/container will be created.
  • containerName (string) (optional) - the value for the Slug header -- i.e. the name of the new resource to be created; this value is optional.
  • options (object) - Optional hashmap of request options
  • data (string) - Optional RDF data serialized as text/turtle; can also be an empty string if no data will be sent.

In the example below we are also sending some meta data (semantics) about the container, setting its type to sioc:Blog.

// Assumes you've loaded rdflib.js and solid.js, see Dependences above
var solid = require('solid')
var parentUrl = 'https://example.org/'
var containerName = 'blog'
var options = {}
var data = '<#this> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://rdfs.org/sioc/ns#Blog> .'

solid.web.createContainer(parentUrl, containerName, options, data).then(
  function(solidResponse) {
    console.log(solidResponse)
    // The resulting object has several useful properties.
    // See lib/solid/response.js for details
    // solidResponse.url - value of the Location header
    // solidResponse.acl - url of acl resource
    // solidResponse.meta - url of meta resource
  }
).catch(function(err){
  console.log(err) // error object
})

Note that the options and data parameters are optional, and you can simply do solid.web.createContainer(url, name).

Listing a Solid Container

To list the contents of a Solid container, just use solid.web.get(). This returns a promise that resolves to a SolidContainer instance, which will contain various useful properties:

  • A short name (.name) and absolute URI (.uri)
  • A .parsedGraph property for further RDF queries
  • A parsed list of links to all the contents (both containers and resources) (.contentsUris)
  • A list of RDF types to which the container belongs (.types)
  • A hashmap of all sub-Containers within this container, keyed by absolute uri (.containers)
  • A hashmap of all non-container Resources within this container, also keyed by absolute uri. (.resources)

Containers also have several convenience methods:

  • container.isEmpty() will return true when there are no sub-containers or resources inside it
  • container.findByType(rdfClass) will return an array of resources or containers that have the given rdfClass in their .types array

For example:

var container = solid.web.list('/settings/')
                  .then(function (container) {
                    console.log(container)
                    // See below
                  })

// container is an instance of SolidContainer (see lib/solid/container.js)
container.uri   // -> 'https://localhost:8443/settings/'
container.name  // -> 'settings'
container.isEmpty()  // -> false
container.types // ->
[
  'http://www.w3.org/ns/ldp#BasicContainer',
  'http://www.w3.org/ns/ldp#Container'
]
container.contentsUris // ->
[
  'https://localhost:8443/settings/prefs.ttl',
  'https://localhost:8443/settings/privateTypeIndex.ttl',
  'https://localhost:8443/settings/testcontainer/'
]

var subContainer =
    container.containers['https://localhost:8443/settings/testcontainer/']
subContainer.name // -> 'testcontainer'
subContainer.types // ->
[
  'http://www.w3.org/ns/ldp#BasicContainer',
  'http://www.w3.org/ns/ldp#Container',
  'http://www.w3.org/ns/ldp#Resource'
]

var resource =
  container.resources['https://localhost:8443/settings/privateTypeIndex.ttl']
// resource - SolidResource instance
resource.name // -> 'privateTypeIndex.ttl'
resource.types // ->
[
  'http://www.w3.org/ns/ldp#Resource',
  'http://www.w3.org/ns/solid/terms#TypeIndex',
  'http://www.w3.org/ns/solid/terms#UnlistedDocument'
]
resource.isType('http://www.w3.org/ns/solid/terms#TypeIndex')  // -> true

container.findByType('http://www.w3.org/ns/ldp#Resource')  // ->
[
  // a SolidContainer('testcontainer'),
  // a SolidResource('privateTypeIndex.ttl'),
  // a SolidResource('prefs.ttl')
]

Creating a resource

Creating a regular LDP resource is done using the web.post() method.

In this example we will create the resource hello-world under the newly created blog/ container.

var solid = require('solid')
var parentDir = 'https://example.org/blog/'
var slug = 'hello-world'
var data = `
<> a <http://rdfs.org/sioc/ns#Post> ;
    <http://purl.org/dc/terms/title> "First post" ;
    <http://rdfs.org/sioc/ns#content> "Hello world! This is my first post" .
`

solid.web.post(parentDir, data, slug).then(
  function(meta) {
    console.log(meta.url) // URL of the newly created resource
  }
).catch(function(err){
  console.log(err) // error object
  // ...
})

Updating a resource

Sometimes we need to update a resource after making a small change. For instance, we sometimes need to delete a triple, or update the value of an object (technically by replacing the triple with a new one). Luckily, Solid allows us to use the HTTP PATCH operation to do very small changes.

Let's try to change the value of the title in our first post. To do so, we need to indicate which triple we want to replace, and then the triple that will replace it.

Let's create the statements and serialize them to Turtle before patching the blog post resource:

// $rdf is a global exposed by loading 'rdflib.js'
var url = 'https://example.org/blog/hello-world'
var vocab = solid.vocab

var oldTitleTriple = $rdf.st($rdf.sym(url), vocab.dct('title'),
  $rdf.lit("First post")).toNT()

var newTitleTriple = $rdf.st($rdf.sym(url), vocab.dct('title'),
  $rdf.lit("Hello")).toNT()

Now we can actually patch the resource. The solid.web.patch() function (also aliased to solid.web.update()) takes three arguments:

  • url (string) - the URL of the resource to be overwritten.
  • toDel (array) - an array of statements to be deleted, serialized as Turtle.
  • toIns (array) - an array of statements to be inserted, serialized as Turtle.
var solid = require('solid')
var toDel = [ oldTitleTriple ]
var toIns = [ newTitleTriple ]
solid.web.patch(url, toDel, toIns).then(function (meta){
  console.log(meta.xhr.status) // HTTP 200 (OK)
}).catch(function(err) {
  console.log(err) // error object
  // ...
})

Replacing a resource

We can also completely replace (overwrite) existing resources with new content, using the client's solid.web.put() function (also aliased to replace()). The function accepts the following parameters:

  • url (string) - the URL of the resource to be overwritten.
  • data (string) - RDF data serialized as text/turtle; can also be an empty string if no data will be sent.
  • mime (string) (optional) - the mime type for this resource; this value is optional and defaults to text/turtle.

Here is an example where we try to overwrite the existing resource hello-world, giving it a bogus type - http://example.org/#Post.

var solid = require('solid')
var url = 'https://example.org/blog/hello-world'
var data = '<> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://example.org/#Post> .'

solid.web.put(url, data).then(
  function (meta) {
    console.log(meta.xhr.status) // HTTP 200 (OK)
  }
).catch(function(err){
  console.log(err) // error object
  // ...
})

Deleting a resource

Delete an RDF resource from the Web. For example, we can delete the blog post hello-world we created earlier, using the solid.web.del() function.

NOTE: while this function can also be used to delete containers, it will only work for empty containers. For now, app developers should make sure to empty a container by recursively calling calling this function on its contents.

var solid = require('solid')
var url = 'https://example.org/blog/hello-world'

solid.web.del(url).then(
  function(success) {
    console.log(success) // true/false
  }
).catch(
  function(err) {
    console.log(err) // error object
    // ...
  }
)

solid.js's People

Contributors

dmitrizagidulin avatar dan-f avatar nicola avatar

Watchers

James Cloos avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.