Our server-to-server API ensures the Q42 Puzzle Studio controls what puzzles and functionality are published per website, without the client’s technical team having to make any changes, and retaining great UX and SEO performance.

How it works

When a user visits a URL on the client’s website, the client’s server calls api.42puzzles.com to determine what HTML to render.

  • If there is no content on the URL, a 404 is returned and the client’s website can also return a 404 to the user.
  • If there is content, the server-to-server API returns exactly what content should be served between the website’s header and footer.

Getting started

Client’s URL 42 Puzzles API URL
/puzzles HOST/hub/ORGANISATION/server?path=/puzzles&brand=BRAND
/puzzles/sudoku HOST/hub/ORGANISATION/server?path=/puzzles/sudoku&brand=BRAND
/puzzles/sudoku/2025-02-01 HOST/hub/ORGANISATION/server?path=/puzzles/sudoku/2025-02-01&brand=BRAND

Try it out with:

HOST: https://api.42puzzles.com
ORGANISATION: nrc
BRAND: nrc

HOST: https://api.42puzzles.com
ORGANISATION: mediahuis
BRAND: tel

HOST: https://api.42puzzles.com
ORGANISATION: mediahuis
BRAND: indo

Response:

{
  "title": "Puzzels",
  "description": "Speel puzzels bij de Telegraaf",
  "html": "<script src=\"https://cdn.42puzzles.com/hub/v1/games-hub.js\" type=\"module\"></script>
           <games-hub-overview></games-hub-overview>"
}

The client’s server renders the resulting HTML in the page that is served to the user:

  • title goes in html.head.title
  • description goes in html.head.description and can also be used as description for socials.
  • html is rendered on the page where the puzzles should be placed. There can be content above or below it.

Additionally, the client’s website should trigger the initialize() call by adding this piece of code in the output HTML:

<script>
  window.gamesHubEvents = window.gamesHubEvents || []
  window.gamesHubEvents.push({ 
    name: 'initialize', 
    data: {
      // Required call - gameshub does not start without a userId
      // Pass null for anonymous users, they will receive a device-specific userid
      userId: 'USERID' | null

      // Required
      platform: 'ios' | 'android' | 'web'

      // Required. Name of the organisation
      organisation: 'nrc' | 'mediahuis'

      // Required. Brand code within an organisation
      brand: 'tel' | 'ds' | 'indo' | ...

      // Optional. In the future we want to be able to disable features based on the subscription of a user
      // for instance, digital-only users cannot access the printed puzzles
      // A user can have multiple subscriptions, for instance 'premium' and 'digital'
      userSubscription: 'SUBNAME' | ['SUBNAME1', 'SUBNAME2'] | null

      // Optional. A/B testing version with support for multiple tests simultaneously
      // for instance, { streak: 'with', history: 'without' }
      abVersion: { 'TESTNAME': 'VERSION', ... }

      // Optional. Defaults to `default` which follows the system theme
      theme: 'darkMode' | 'lightMode' | 'default'

      // Optional. Defaults to `false` which doesn't output logging to console
      debug: true | false
    }
  });
</script>

Test environment

A demo consumer of the server-to-server API can be found here:

https://demo.42puzzles.com/server.html

This endpoint calls the hub/ORGANISATION/server endpoint, and renders the response.

This connects to our test environment which does not have production-grade performance. The first page hit can take several seconds.

Brands

If you want to know what brands are supported, visit

HOST/hub/ORGANISATION/configs

HOST: api.42puzzles.com
ORGANISATION: mediahuis