Generic Data Viewer React Admin

This Web app allows users to easily execute queries over multiple data sources (including Solid pods) and inspect the corresponding results.

Table of contents:

Getting Started

To install the application:

npm install

To run the Web application in development mode:

npm run dev

Now you can browse the displayed URL.

To see the queries provided in the example configuration src/config.json at work, you also need to activate the supporting resources:

  1. In a new terminal window, prepare and start the local pods:

    npm run reset:pods && npm run start:pods
  2. In a new terminal window, start the http proxy:

    npm run start:proxy
  3. In a new terminal window, start a server which denies all CORS headers:

    npm run start:badCors

Some queries require a log in. Log in with the IDP http://localhost:8080 and the credentials for the user owning the pod named example in the file seeded-pod-config.json.

Static, production build

To make a standalone version of the result of this project, you can make a static build and serve it using any webserver. Execute:

npm run build

The static build appears in the dist folder.

Logging in

Some queries access data sources that are only readable by authenticated users. This requires you to log in. To log in, you need to provide an Identity Provider or a WebID. The application will detect which one you use and redirect you to the login page of your Identity Provider. If you provide a WebID, the first Identity Provider found in the given WebID is used.

Configuration file

The configuration file follows a simple structure.

{
  "title": "Title shown at the top of the app.",
  "logoLocation": "Image location of the logo shown at the top of the app (relative to public folder.).",
  "logoRedirectURL": "The URL the Web application redirects to when a user clicks on the logo.",
  "mainAppColor": "The main colors used in the app, can be any CSS color.",
  "backgroundColor": "Background color of the app, can be any CSS color.",
  "titleColor": "The color of the title, can be any CSS color",
  "textColor": "The color of all the text in teh app body, this means all text except header and footer.",
  "footer": "HTML components or text that will function as the footer (will be placed in the footer div.)",
  "defaultIDP": "The default value used for IDP when logging in, this IDP can be manually changed in the Web app as well. ",
  "queryFolder": "The base location of the queries, all query locations will start from this folder (relative to public folder.)",
  "httpProxy": "The http proxy through which the requests will be rerouted. When left empty, the Comunica query engine will handle it. This is useful when CORS headers are not set (correctly) on the queried source.",
  "introductionText": "The text that the app shows on the dashboard, which the app also shows when you first open it.",
  "queryGroups" : [
    {
      "id": "A unique ID for the query group",
      "name": "A name for the query group",
      "icon": "The key to the icon for the query group. This is optional and a default menu icon will be used when left empty."
    }
  ],
  "queries": [
    {
      "id": "A unique ID for the query. This ID appears in the URL of the displayed result. Queries are ordered in the menu according to ascending ID.",
      "queryGroupId": "ID of the query group too which this query belongs. If not given, the query is displayed outside existing groups.",
      "queryLocation": "Path to the query location, relative to 'queryFolder'",
      "name": "A name for the query",
      "description": "Description of the query",
      "icon": "The key to the icon for the query. This is optional and a default menu icon will be used when left empty.",
      "comunicaContext": {
        "sources": "Initial list of sources over which the query should be executed",
        "useProxy": "True or false, whether the query should be executed through the proxy or not. This field is optional and defaults to false.",
        ... any other field that can be used in the Comunica query engine https://comunica.dev/docs/query/advanced/context/
      },
      "sourcesIndex": {
        "url": "URL of the publicly available RDF resource acting as an index file for more sources over which the query should be executed",
        "queryLocation": "Path to the location, relative to 'queryFolder', of the (auxiliary) query that yields the sources from above RDF resource"
      },
      "variables": {
        "variableExampleString": ["\"String1\"", "\"String2\""],
        "variableExampleUri": ["<https://example.com/uri1>", "<https://example.com/uri2>"]
      },
      "askQuery": {
        "trueText": "The text that is to be shown when the query result is true (in ASK queries).",
        "falseText": "The text that is to be shown when the query result is false (in ASK queries)."
      }
    },
    ... etc
  ]
}

Specifying sources

The set of sources over which a query will be executed is derived from two optional inputs in a query entry:

  • comunicaContext.sources: an array of sources, known at the time of writing the config file;
  • sourceIndex: describes an external RDF resource, from which sources are derived at execution time.

If both inputs are present, the query will be executed over the superset of sources.

The (auxiliary) query provided in sourceIndex.queryLocation is executed on sourceIndex.url and must result in the list of source URLs.

If sourceIndex is used and there is no comunicaContext.lenient property found, one will be created with value true. This makes sure that the (main) query can succeed if not all obtained sources are accessible.

Adding variable type

When executing a query, it gives us either a URL, a literal value or a blank node. These URLs could reference to anything e.g. a picture, spreadsheet, resume, and so on. Also literals can be lots of things e.g. a float, integer, string, birthdate, price, and so on. By clarifying what the expected type is of the query result corresponding to a given variable we can fully interpret how we can display and represent the result.

You can specify the type of a variable by extending its name with the type in the query as such: variableName_variableType. The underscore _ here is crucial to make a clear distinction between name and type.

Templated queries

This application supports queries whose contents are not completely fixed upfront: they contain variables whose value can be set interactively.

To change a query into a templated query:

  • replace the fixed portion(s) of the query with (a) variable(s); a variable is an identifier preceded by a $ sign, e.g. $genre
  • add a variables object in the query's entry in the configuration file
  • in the variables object, for each variable, add a property with name equal to the variable's identifier
  • set each such property's value to an array of possible values for the corresponding variable

Note that variables' values are not restricted to strings: URIs for example are alo possible. As a consequence, for strings the surround double quotes " must be added to the values in the list. This is shown in the configuration structure above.

Query icons

In the selection menu the name of the query is proceeded by an icon. You configure this icon per query in the configuration file.
For this to work you need to add the icon to the exports in IconProvider.js. We advise to use the Material UI icons as this is what's used internally in react-admin and it is also included in the dependencies. Nevertheless, you can use any React component you want, just make sure it's a functional component.

Custom queries

Besides the prepared queries in the configuration file, a user can edit custom queries:

  • To create a custom query, open "Custom Query Editor" from the menu on the left.
  • Complete the custom query editor form and click the "CREATE QUERY" button when ready.
  • Your new query is added to the "Custom queries" group and you are redirected to the query's result view.
  • If not satisfied with the query result, you can click "EDIT QUERY" to further edit your query. When saving changes, the result is recalculated.
  • Because the custom query only lives as long as your browser remembers it, a "SAVE QUERY LINK" button is provided. Use it to generate a unique URL for this custom query. Copy that URL to your clipboard and save it. You can then visit that URL any time later, to recreate this query.
  • To clean up an unwanted custom query, there is always a button "DELETE QUERY"...

Representation Mapper

If you want to add your own type representations you can do this by adding your representation to the representationProvider.js file. This can be useful for example when querying images. The result of the query is a reference to the image. By mapping a representation we can show the actual image instead of the reference.

The mapper follows a structure:

{
    "typeName": mapperComponent,
    ...
}

With typeName being the name of the variable as defined in the query which is defined in the configuration file. The function mapperComponent takes the query result for the corresponding variable and returns either a React component (see below). Examples of how you can do this can already be found in the representationProvider components folder.

The components get the following props:

  • record (the query result), an object of RDF/JS objects.
  • variable the variable name and key of record, a string.

Hint use the Field components from react-admin to display the result. They've already got styling matching that of react-admin and are easy to use.

Warning if you change the record object, the changed will still be present in the next render.

Using the local pods

To support the provided example configuration src/config.json and the tests, this repo integrates some local pods. You can make use of these for your own tests. Follow these steps:

  • Add your data and .acl files in the initial-pod-data folder. These files will be available in the pod relative to http://localhost:8080/example/.
  • Prepare the pods by executing npm run reset:pods.

Testing

For testing we use Cypress.

It is important to test the production version at least at the end of a development cycle.

The development version might be tested repeatedly during development.

Testing the production version

  1. Build the production version of the Web application and serve it:

    # make really, really sure to build from scratch
    rm -rf node_modules/
    rm -rf dist/
    npm install
    # build
    npm run build
    # serve
    npx http-server -p 5173 ./dist
  2. In a new terminal window, prepare and start the local pods:

    npm run reset:pods && npm run start:pods
  3. In a new terminal window, start the http proxy:

    npm run start:proxy
  4. In a new terminal window, start a server which denies all CORS headers:

    npm run start:badCors
  5. Finally, in a new terminal window, you can execute the tests by running:

    ## for normal test execution:
    npm run test
    ## for interactive testing:
    npm run test:interactive

Testing the development version

The procedure is the same as for testing the production version, except for step 1, which is now:

  1. Start the Web application in development mode:

    npm run dev