Cypress Plugin to test applications using the Solid Protocol.
Install the package using npm:
npm install cypress-solid --save-dev
Once it is installed, you'll need to call it in your support file. For example, in cypress/support/e2e.ts
:
import 'cypress-solid/support';
You'll also need to set up the node events in the configuration. For example, in cypress.config.ts
:
import { defineConfig } from 'cypress';
import { setupSolidNodeEvents } from 'cypress-solid/config';
export default defineConfig({
e2e: {
setupNodeEvents: setupSolidNodeEvents,
},
});
This plugin assumes that you have an instance of the Community Solid Server running (with version 7.0.0 or newer).
To install it, you can run the following command:
npm install @solid/community-server@^7 --save-dev
The server must be running whilst Cypress tests your application, so you may want to use something like the start-server-and-test
and concurrently
packages to launch both your application and the server. For example, you can configure the following scripts in your package.json
:
"scripts": {
"cy:dev": "concurrently --kill-others \"npm run test:serve-app\" \"npm run test:serve-pod\" \"npm run cy:open\"",
"cy:open": "cypress open --e2e --browser chromium",
"cy:run": "cypress run",
"cy:test": "start-server-and-test test:serve-app http-get://localhost:5001 test:serve-pod http-get://localhost:3000 cy:run",
"test:serve-app": "vite --port 5001",
"test:serve-pod": "community-solid-server -p 3000 -l warn"
}
With these, you should be able to run your Cypress tests with npm run cy:test
, or open the Cypress UI for local development with npm run cy:dev
.
In this example, the application is served using Vite, but you should be able to use the same approach for any other tools. You can learn more about this in the official Cypress documentation: Boot your server.
The default configuration assumes that the server is running on http://localhost:3000
, but you can customize this behaviour changing the serverUrl
config option.
In tests where you want to interact with the Solid server, you should call cy.solidReset()
before starting. It is important to do it before, and not after, given that if the test fails or is closed unexpectedly, the clean up won't be executed.
This command will prepare the account in the server, and you can complete the log in form during the authentication process using the cy.solidLogin()
command.
Finally, there are some helper functions you can use in order to get dynamic data at runtime, such as webId()
. This is ideal to avoid hard-coding values that depend on the configuration.
Here's a full example illustrating the basic usage:
import { webId } from 'cypress-solid';
it('Logs in', () => {
// Prepare the POD.
cy.solidReset();
// Open the application and trigger the log in process.
cy.visit('/');
cy.get('[aria-label="Login url"]').type(webId()).type('{enter}');
// Log in to the POD.
cy.solidLogin();
// Continue with your test...
});
For a complete example with more advanced functionality, you can check out the Cypress Solid Sandbox sample application.
In order to customize the plugin's behaviour, you can pass some configuration options to the setupSolidNodeEvents
method:
import { defineConfig } from 'cypress';
import { setupSolidNodeEvents } from 'cypress-solid/config';
export default defineConfig({
e2e: {
setupNodeEvents(on, config) {
setupSolidNodeEvents(on, config, {
serverUrl: 'http://localhost:1234',
account: 'bob',
// ...
});
// IMPORTANT Return the updated config object.
return config;
},
},
});
Here's the complete list of available options:
Option | Default | Description |
---|---|---|
serverUrl | 'http://localhost:3000' |
Url where the Solid server is running. |
account | 'alice' |
Account name, this will be used to create a new account in the Solid server. For example, http://localhost:3000/alice/ . |
name | 'Alice Cooper' |
Display name, this will be declared as foaf:name in the user's profile. |
'alice@example.com' |
Account email, this can be used to log in. | |
password | 'secret' |
Account password, this can be used to log in. |
Loads a fixture file with replacements applied.
Name | Default | Description |
---|---|---|
path | - | Fixture file path, relative to your fixtures folder. |
replacements | {} |
Object with replacements to apply on the fixture. Learn more about replacements. |
The contents of the fixture with the replacements applied.
Creates a container in the POD.
Name | Default | Description |
---|---|---|
path | - | Relative container path. For example, to create a container at http://localhost:3000/alice/tasks/ , you would use '/tasks/' . |
name | 'Container' |
Container name, declared as rdfs:label in the document's metadata. |
Nothing.
Creates a document in the POD.
Name | Default | Description |
---|---|---|
path | - | Relative document path. For example, to create a document at http://localhost:3000/alice/tasks/1 , you would use '/tasks/1' . |
turtleOrFixture | - | Contents of the document in Turtle format, or the path to a Cypress fixture including the contents. For example, for a fixture at cypress/fixtures/task.ttl you would use 'task.ttl' . |
replacements | {} |
Object with replacements to apply on the contents before creating the document. Learn more about replacements. |
Nothing.
Deletes a document in the POD.
Name | Default | Description |
---|---|---|
path | - | Relative document path. For example, to delete the document at http://localhost:3000/alice/tasks/1 , you would use '/tasks/1' . |
Nothing.
Accepts the authorization form. This command is usually necessary when you have already logged in to your POD and you're reconnecting after a page reload.
None.
Nothing.
Authenticates using the test account indicated in the configuration. This command should be called once the authentication process has been started and the application has been redirected to the identity provider.
None.
Nothing.
Reads a document in the POD.
Name | Default | Description |
---|---|---|
path | - | Relative document path. For example, to read the document at http://localhost:3000/alice/tasks/1 , you would use '/tasks/1' . |
The contents of the Solid document in Turtle format.
Makes an authenticated request to the Solid server.
Takes the same arguments as Node's built-in fetch().
An object with the following fields:
Name | Type | Description |
---|---|---|
headers | Headers | Response headers. |
status | number |
Response status code. |
statusText | string |
Response status message. |
body | string |
Response body. |
Creates the account in the server if it didn't exist, and resets the content in the POD to its initial state.
None.
Nothing.
Updates a document in the POD.
Name | Default | Description |
---|---|---|
path | - | Relative document path. For example, to update a document at http://localhost:3000/alice/tasks/1 , you would use '/tasks/1' . |
sparqlOrFixture | - | SPARQL updates to apply in the document, or the path to a Cypress fixture including the query. For example, for a fixture at cypress/fixtures/update-task.sparql you would use 'update-task.sparql' . |
replacements | {} |
Object with replacements to apply on the query before creating the document. Learn more about replacements. |
Nothing.
You can also use some of these helper functions in your tests. These should be imported directly from cypress-solid
, like this:
import { webId } from 'cypress-solid';
it('See tasks container url', () => {
cy.contains(podUrl('/tasks/'));
});
Get configuration values.
Name | Default | Description |
---|---|---|
key | - | Name of the config value to (serverUrl , account , etc.). If this argument is omitted, the whole configuration object will be returned. |
Value of the configuration option if key
was provided, or the entire configuration object otherwise.
Resolve relative paths in the test account to get the full url.
Name | Default | Description |
---|---|---|
path | '' |
Relative path of the url to resolve. |
Url for the given path. For example, for the '/tasks/'
path with the default configuration you would get 'http://localhost:3000/alice/tasks/'
.
Resolve relative paths in the server to get the full url.
Name | Default | Description |
---|---|---|
path | '' |
Relative path of the url to resolve. |
Url for the given path. For example, for the '/.account/'
path with the default configuration you would get 'http://localhost:3000/.account/'
.
Get test account webId. You can use this to log in in your application without hard-coding any urls.
None.
WebId for the test account. For example, with the default configuration you would get 'https://localhost:3000/alice/profile/card#me'
.
This plugin also adds the following Chai Assertions. All of them can also use wildcards.
Use it to compare SPARQL strings. For example:
it('Compares SPARQL', () => {
cy.intercept('PATCH', podUrl('/tasks/1')).as('updateTask');
// Test your application...
cy.get('@updateTask')
.its('request.body')
.should(
'be.sparql',
`
DELETE DATA {
<#it> <https://schema.org/name> "Previous task name" .
} ;
INSERT DATA {
<#it> <https://schema.org/name> "New task name" .
} .
`
);
});
Use it to compare Turtle strings. For example:
it('Compares Turtle', () => {
cy.intercept('PUT', podUrl('/tasks/*')).as('createTask');
// Test your application...
cy.get('@createTask')
.its('request.body')
.should(
'be.turtle',
`
@prefix schema: <https://schema.org/> .
<#it>
a schema:Action ;
schema:name "Learn to use cypress-solid" .
`
);
});
There are some situations when seeding data or making assertions where it may be impossible to know some values before runtime. For example, seeding some data that depends on ids generated dynamically, or making assertions in a document using dates and timestamps.
For these scenarios, some of the methods in this plugin accept replacements and wildcards.
Replacements are placeholders in fixture files that can be replaced once they are loaded in your tests. They use the {{placeholder}}
syntax.
For example, imagine that you have the following fixture to represent an update to a Movie document fixtures/watch-movie.sparql
:
INSERT DATA {
@prefix schema: <https://schema.org/>.
@prefix xsd: <http://www.w3.org/2001/XMLSchema#>.
<#watch-action>
a schema:WatchAction ;
schema:object <#it> ;
schema:startTime "{{date}}"^^xsd:dateTime .
}
This fixture has a {{date}}
placeholder that can be replaced at runtime. For example, this can be used with the solidUpdateDocument command:
it('Watch a movie', () => {
const now = new Date();
cy.solidCreateDocument('/movies/movie', 'movie.ttl');
cy.solidUpdateDocument('/movies/movie', 'watch-movie.sparql', {
date: now.toISOString(),
});
});
The text within {{}}
is actually treated as JavaScript and evaluated before rendering, so you can do things like {{ name || 'Fallback' }}
in your fixture files.
Additionally, there are some helper functions you can use:
now()
returns the current date in ISO string format.date('2024-03-03')
returns the parsed date in ISO string format.uuid()
returns a random UUID.
Here's an example of a complex fixture file used to test a SPARQL query:
INSERT DATA {
@prefix schema: <https://schema.org/>.
@prefix xsd: <http://www.w3.org/2001/XMLSchema#>.
<#{{ resourceHash || uuid() }}>
a schema:Movie ;
schema:name {{ name || 'Spirited Away' }} ;
schema:datePublished "{{ datePublished || date('2001-07-20') }}"^^xsd:dateTime .
}
Similarly, asserting some data that has been creating at runtime may be challenging. Wildcards can be used to run regular expression matches in assertions. They use the [[regex]]
syntax.
For example, imagine that you want to assert that a Movie document has been updated properly:
it('Watch a movie', () => {
cy.intercept('PATCH', podUrl('/movies/*')).as('watchMovie');
// Interact with your application to watch a Movie...
cy.get('@watchMovie')
.its('request.body')
.should(
'be.sparql',
`
INSERT DATA {
@prefix schema: <https://schema.org/>.
@prefix xsd: <http://www.w3.org/2001/XMLSchema#>.
<#[[.*]]>
a schema:WatchAction ;
schema:object <#it> ;
schema:startTime "[[.*]]"^^xsd:dateTime .
} .
`
);
});
In this case, we're using two wildcards. One for the value of schema:startTime
, which is a date generated at runtime; and one for the fragment of the schema:WatchAction
resource id, which could be a string generated dynamically.