npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@educandu/educandu

v3.35.4

Published

The educandu framework

Downloads

1,000

Readme

educandu

codecov

The educandu framework

Prerequisites

  • node.js ^18.0.0
  • Docker
  • optional: globally installed gulp: npm i -g gulp-cli

The output of this repository is an npm package (@educandu/educandu).

Configuration options

| Option | Description | Type | Required | | --- | --- | --- | --- | | appName | The name of the application using educandu | string | yes | | port | Port on which the project is run | number, mininum 1 | no, defaults to 80 | | trustProxy | Whether to trust proxies at all or how many maximum consecutive proxy hubs to trust | boolean or number (mininum 0) | no, defaults to false | | mongoConnectionString | The URI for the project's MongoDB | string | yes | | skipMaintenance | Whether or not to run MongoDB migrations and checks on startup | boolean | no, defaults to false | | cdnEndpoint | The URL of the AWS-hosted CDN | string | yes | | cdnRegion | The region of the AWS-hosted CDN | string | yes | | cdnAccessKey | The access key of the AWS-hosted CDN | string | yes | | cdnSecretKey | The secret key of the AWS-hosted CDN | string | yes | | cdnBucketName | The name of the AWS S3 bucket storing the CDN data | string | yes | | cdnRootUrl | The root url of the CDN | string | yes | | customResolvers | The same object that is also used to hydrate the app on the client side | { resolveCustomPageTemplate, resolveCustomHomePageTemplate, resolveCustomSiteLogo, resolveCustomPluginInfos } | yes, accepts null for either property and it will default to the internal setup | publicFolders | The project-specific public folders that need to be accesible on the project domain | array of { publicPath: <string>, destination: <string>, setHeaders: <function(res, path){}> } | no, also setHeaders is optional | | resources | URLs to additional resource bundles, e.g. extra translations | array of string | no | | themeFile | URL to overrides of educandu LESS variables, through which the AntDesign theme is set | string | no | | additionalControllers | Custom controllers | arrayOfControllers: [] | no, defaults to [] | | additionalHeadHtml | Custom HTML to inject in the <head> of the document | string | no | | sessionSecret | The unique ID of the user session | string | no, defaults to a generated unique id | | sessionCookieDomain | The domain attribute to be set on the session cookie | string | no, defaults to the request's host header domain | | sessionCookieName | The name to be used for the session cookie | string | yes | | sessionCookieSecure | The value of the cookie's "secure" flag. Note: When set to true, the cookie only gets set over an HTTPS connection | boolean, defaults to false | no | | sessionDurationInMinutes | The validity of the user session in minutes | number, minumum 1 | no, defaults to 60 | | consentCookieNamePrefix | Prefix for the consent cookie name | string | yes | | uploadLiabilityCookieName | Name for the public storage upload liability cookie | string | yes | | announcementCookieNamePrefix | Prefix for the admin-set announcement cookie | string | yes | | xFrameOptions | Value for the x-frame-options header set on pages response | string ('DENY' or 'SAMEORIGIN') | no, by default the header is not set | | xRoomsAuthSecret | Value for the x-rooms-auth-secret header expected from rooms-auth-lambda authorization requests upon accessing rooms CDN resources | string | no, when not provided the effect of the request is an automatic session cookie invalidation/regeneration | | smtpOptions | The SMTP setup for sending emails to users upon registration or password reset | anything | yes | | emailSenderAddress | The email address from which emails are sent | string | yes | | adminEmailAddress | The email address to show to users as the admin email address | string | no | | emailAddressIgnorePattern | A RegExp pattern to match receiver addresses of emails that should not be sent, defaults to ^.+@test\.com$ | string | no | | initialUser | The first user account, with admin role | { email, password, displayName } or null | no | | basicAuthUsers | When provided, the web pages become protected by a basic auth layer through which the provided users can authenticate. This way non-production environments can be protected. | object with usernames as keys and passwords as values | no | | plugins | List of plugins available to platform users when they create website content | array of string | no, defaults to ['markdown', 'image'] | | allowedLicenses | A list of SPDX license names that should be usable inside the app (defaults to the whole currently "known" list) | string[] | no | | disabledFeatures | A list of names of disabled features | string[] | no | | exposeErrorDetails | Whether or not to expose details of thrown errors (e.g. stack trace) | boolean | no, defaults to false | | disableScheduling | Whether or not to run job schedulers | boolean | no, defaults to false | | ambConfig | Configuration for the AMB endpoint (https://dini-ag-kim.github.io/amb/) | { apiKey: <string>, image: <string>, publisher: [{ type: <'Organization'/'Person'>, name: <string> }], about: [{ id: <category URL from https://skohub.io/dini-ag-kim/hochschulfaechersystematik/heads/master/w3id.org/kim/hochschulfaechersystematik/scheme.en.html>}] } | no, however if provided, apiKey is mandatory | | samlAuth | Configuration for SAML authentication | { decryption: { pvk: <string>, cert: <string> }, identityProviders: [{ key: <string>, displayName: <string>, entryPoint: <string>, cert: <string>, logoUrl: <string>, expiryTimeoutInDays: <number> }] } | no, however if provided, decryption.pvk, decryption.cert, identityProviders.key, identityProviders.displayName, identityProviders.entryPoint and identityProviders.cert are mandatory, while identityProviders.logoUrl defaults to null and identityProviders.expiryTimeoutInDays defaults to 180 days |

How to use

$ yarn add @educandu/educandu

Use it in code as follows:

import educandu from '@educandu/educandu';

educandu({
  appName: 'My educandu app',
  port: 3000,
  mongoConnectionString: 'mongodb://root:rootpw@localhost:27017/dev-educandu-db?replicaSet=educandurs&authSource=admin',
  skipMaintenance: false,
  cdnEndpoint: 'http://localhost:9000',
  cdnRegion: 'eu-central-1',
  cdnAccessKey: 'UVDXF41PYEAX0PXD8826',
  cdnSecretKey: 'SXtajmM3uahrQ1ALECh3Z3iKT76s2s5GBJlbQMZx',
  cdnBucketName: 'dev-educandu-cdn',
  cdnRootUrl: 'http://localhost:9000/dev-educandu-cdn',
  customResolvers: {
    resolveCustomPageTemplate: null,
    resolveCustomHomePageTemplate: null,
    resolveCustomSiteLogo: null,
    resolveCustomPluginInfos: null
  }).required(),
  publicFolders: [{ publicPath: '/', destination: path.resolve(thisDir, '../static') }],
  resources: ['./test-app/resource-overrides.json'].map(x => path.resolve(x)),
  themeFile: path.resolve('./test-app/theme.less'),
  additionalControllers: [MyCustomPageController],
  additionalHeadHtml: '<link rel=\"apple-touch-icon\" sizes=\"180x180\" href=\"/apple-touch-icon.png?v=cakfaagbe\">\n<link rel=\"icon\" type=\"image/png\" sizes=\"32x32\" href=\"/favicon-32x32.png?v=cakfaagbe\">',
  sessionSecret: 'd4340515fa834498b3ab1aba1e4d9013',
  sessionCookieDomain: 'localhost',
  sessionCookieName: 'APP_SESSION_ID',
  sessionCookieSecure: false,
  sessionDurationInMinutes: 60,
  consentCookieNamePrefix: 'APP_CONSENT_COOKIE_NAME',
  uploadLiabilityCookieName: 'APP_UPLOAD_LIABILITY_COOKIE_NAME',
  announcementCookieNamePrefix: 'APP_ANNOUNCEMENT_COOKIE_NAME',
  xFrameOptions: 'SAMEORIGIN',
  smtpOptions: 'smtp://localhost:8025/?ignoreTLS=true',
  emailSenderAddress: '[email protected]',
  initialUser: {
    email: '[email protected]',
    password: 'test',
    displayName: 'Testibus'
  },
  basicAuthUsers: {
    gatekeeper: 'gatekeeperPassword'
  },
  plugins: ['markdown', 'image', 'table', 'audio', 'video'],
  allowedLicenses: ['CC0-1.0', 'CC-BY-4.0', 'MIT'],
  exposeErrorDetails: true,
  disableScheduling: false,
  ambConfig {
    apiKey: 'C36CAD3A805A11EDA1EB0242AC120002',
    image: './test-app/images/app-logo.png',
    publisher: [
      {
        type: 'Organization',
        name: 'My educandu app'
      }
    ],
    about: [
      {
        id: 'https://w3id.org/kim/hochschulfaechersystematik/n78'
      }
    ]
  },
  samlAuth: {
    decryption: {
      pvk: '<private_key>',
      cert: '<certificate>',
    },
    identityProviders: [
      {
        key: 'panda',
        displayName: 'The Panda University',
        entryPoint: 'https://en.wikipedia.org/wiki/Giant_panda',
        cert: 'nonsense',
        logoUrl: '/images/panda-logo.svg',
        expiryTimeoutInDays: 4 * 30
      }
    ]
  }
});

This should run the entire application and provide you with an admin user as defined on the initialUser setting.

How to override the styles

In order to override the styles provided by educandu you need to import the main.less of the educandu project and override the less variables provided. An example override implementation is provided by the test-app in the main.less file. We will keep this file updated so any user will know what can be overriden. Alternatively you can go to the global-variables.less in educandu and consult the list there.

How to run and develop locally

The gulpfile has a number of useful tasks for local development which can be run with gulp {taskName}, most commonly used:

  • (default): build and start up the test app (in watch mode), which is set up to use educandu, CLI args:
    • --instances 3 (number of app instances to run, optional, default 1)
    • --tunnel (flag to run using the tunnel proxy, optional, default: no tunneling)
  • test: runs all tests (with coverage)
  • testWatch: runs tests in watch mode
  • lint: runs eslint
  • fix: runs eslint in fixing mode
  • up: starts all the containers (if not already running)
  • down: stops all the containers and deletes them
  • maildev(Up|Down), mongo(Up|Down), minio(Up|Down): starts/stops individual containers
  • createSamlCertificate: creates a self-signed certificate that can be used for SAML de/encryption, CLI args:
    • --domain my-domain.com (common name value, mandatory)
    • --days 365 (expiration time, optional, default: 100 * 365)
    • --dir ./output (output directory, optional, default: ./certificates)

By default the test application requires that the following ports are available to be taken:

  • 3000: the test application or the load balancerin case of load balancing (instances > 1)
  • 400x: the individual test application instances, in case of load balancing (instances > 1)
  • 8000: maildev UI, can be used for debugging emails that would be sent to the users (http://localhost:8000)
  • 8025: maildev SMTP server
  • 9000: minio server and UI (http://localhost:9000)
  • 21017: mongodb server

The ports can be changed in gulpfile.js and need to be reflected in test-app/index.js.

When tunneling is enabled, the following environment variables are required to be set:

  • TUNNEL_TOKEN token used to verify the tunnel connection
  • TUNNEL_WEBSITE_DOMAIN domain of the website served over the tunnel
  • TUNNEL_WEBSITE_CDN_DOMAIN domain of the cdn belonging to the website served over the tunnel
  • TUNNEL_WEBSITE_SAML_AUTH_DECRYPTION certificate pems for SAML en-/decryption in JSON format ({ "pvk": <private_key>, "cert": <cert> })

OER learning platform for music

Funded by 'Stiftung Innovation in der Hochschullehre'

A Project of the 'Hochschule für Musik und Theater München' (University for Music and Performing Arts)

Project owner: Hochschule für Musik und Theater München
Project management: Ulrich Kaiser