Layers

Shared Code

2025-01-15


Description

Layers is a Nuxt feature allowing common code to be shared among projects.

Shared code can be co-located with apps in monorepos, accessed from installed npm pkgs, and from external GitHub repos. The production host will clone an entire monorepo and build using shared assets as configured. If using NPM pkg or GitHub repo for base assets, the host server installs or clones the asset base during the build.

Monorepo

A monorepo contains common assets, plus similarly focused apps that require a common, centralized code base. This approach would have a structure like:

kevin/
├── base/
│   ├── nuxt.config.ts
│   ├── tailwind.config.ts
│   ├── assets/
│   │   └── css/
│   │       └── fleet.css
│   └── content/
│
├── apps/
│   ├── app1/
│   │    ├── nuxt.config.ts
│   │    └── assets/
│   │           └── css/
│   │                 └── ship.css
│   ├── app2/
│   └── ...
│
└── vercel.json

Base Config

Config @@/base/nuxt.config.ts to specify assets to be shared amongst monorepo apps.

Ex, Make @/base/assets/css/fleet.css accessable to all monorepo apps:

//<--------@@/base/nuxt.config.ts---------------------------------------------->
import { fileURLToPath } from 'url'
import { dirname, join } from 'path'

// Relative paths in Layer are resolved relative to the app, not the Layer
const currentDir = dirname(fileURLToPath(import.meta.url))

// Make @/base/assets/css/fleet.css available for reuse
export default defineNuxtConfig({
  css: [
    join(currentDir, './assets/css/fleet.css')
  ]
})
//<--------@@/base/nuxt.config.ts---------------------------------------------->

Apps Config

Extends Property

The extends property in a project's nuxt.config.ts allows the project's code to be extended to include common monorepo assets, as well as code from installed npm pkgs and GitHub repos:

//<--------@@/apps/app1/nuxt.config.ts----------------------------------------->
export default defineNuxtConfig({

    ...

    extends: [

        // Extend all shared assets in co-located monorepo
        '../../base',

        // Extend assets located in a public GitHub repo
        ['github:annebrown/base-pub', { auth: process.env.GITHUB_TOKEN }]

        // GitHub private repo, using token, env var, authn
        ['github:annebrown/base-priv', { auth: process.env.GITHUB_TOKEN }]

    ],

    ...

})

...

//<--------@@/apps/app1/nuxt.config.ts----------------------------------------->

Prod Host Config

Ex, @@/vercel.json:

{
  "buildCommand": "npm run build",
  "outputDirectory": ".output",
  "ignoreCommand": "git diff --quiet HEAD^ HEAD ./",
  "projects": [
    { "name": "app1", "path": "apps/app1" },
    { "name": "app2", "path": "apps/app2" }
  ]
}

Reusing Components

Nuxt component syntax rqd to reuse component @/base/components/fleet/navi/Toc.vue from the shared base:

<FleetNaviToc />

Shared Assets

The following list of assets in the common code base are automatically scanned by Nuxt for use in apps extending this base layer:

  • nuxt.config.ts
  • app.config.ts
  • components
  • composables
  • layouts
  • pages
  • plugins
  • server
  • utils

If using @Nuxt/content module, it is scanned automatically as well.

Deployments

When deploying, the host automatically clones the entire monorepo, then builds using the extended the base layer, according to base and app config.

Copyright @ 2025 Anne Brown