Skip to content

A config loader that searches for and loads configuration files for your program, with support for `type:module`, `esm`, and `cjs` module formats

License

Notifications You must be signed in to change notification settings

hyperse-io/config-loader

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

26 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

@hyperse/config-loader

A config loader that searches for and loads configuration files for your program, with support for type:module, esm, and cjs module formats

build stable version GitHub top language Licence

The config loader is a powerful utility that searches for and loads configuration files for your JavaScript/TypeScript applications. It automatically handles multiple file formats and module systems while providing type safety.

Built on top of cosmiconfig, it implements smart defaults aligned with JavaScript ecosystem conventions, while remaining highly configurable. You can customize search paths, file formats, and loading behavior to match your specific needs.

Features

  • 🔍 Automatic config file resolution with support for multiple formats (JSON, YAML, JS, TS)
  • 🎯 Flexible configuration search across multiple file locations
  • 🚀 ESM and CommonJS module support
  • đź”§ TypeScript configuration with full type inference
  • ⚡️ Built on top of cosmiconfig for reliable config loading
  • đź§Ş Well-tested and production-ready
  • 🔄 Automatic resolution of TypeScript path aliases from tsconfig.json
  • đź”— Smart handling of external modules and dependencies
  • 📦 Automatically marks package.json dependencies and devDependencies as external
  • 🏗️ Built-in support for monorepo setups and workspace packages

Installation

npm install --save @hyperse/config-loader

API Reference

searchConfig<T>(moduleName: string, searchFrom?: string, options?: LoaderOptions)

Searches up the directory tree for configuration files, checking multiple locations in each directory until it finds a valid configuration or reaches the home directory.

Parameters:

  • moduleName (string): Your module name. Used to create default search places and package properties
  • searchFrom (string, optional): Directory to start searching from. Defaults to process.cwd()
  • options (LoaderOptions, optional): Configuration options for the loader

Returns: Promise<ConfigLoadResult | null>

Example:

import { searchConfig } from '@hyperse/config-loader';

// Search for a config file named 'myapp'
const result = await searchConfig<{
  port: number;
  database: {
    url: string;
  };
}>('myapp');

if (result) {
  console.log(result.config.port); // Access the loaded config
}

loadConfig<T>(configFile: string, options?: LoaderOptions)

Loads a configuration file from a specific path.

Parameters:

  • configFile (string): Path to the configuration file
  • options (LoaderOptions, optional): Configuration options for the loader

Returns: Promise<ConfigLoadResult | null>

Example:

import { loadConfig } from '@hyperse/config-loader';

// Load a specific config file
const result = await loadConfig<{
  port: number;
  database: {
    url: string;
  };
}>('./config/myapp.config.ts');

if (result) {
  console.log(result.config.port);
}

mergeOptions<T>(target: T, source: DeepPartial<T>, mergeUndefined?: boolean)

Performs a deep merge of two configuration objects. Unlike Object.assign(), the target object is not mutated.

Parameters:

  • target (T): The target object to merge into
  • source (DeepPartial): The source object to merge from
  • mergeUndefined (boolean, optional): Whether to merge undefined values. Defaults to false

Returns: T (new merged object)

Example:

import { mergeOptions } from '@hyperse/config-loader';

const defaultConfig = {
  server: {
    port: 3000,
    host: 'localhost',
  },
  database: {
    url: 'postgres://localhost:5432/mydb',
    pool: {
      min: 0,
      max: 10,
    },
  },
};

const userConfig = {
  server: {
    port: 4000,
  },
  database: {
    pool: {
      max: 20,
    },
  },
};

const mergedConfig = mergeOptions(defaultConfig, userConfig);
// Result:
// {
//   server: {
//     port: 4000,
//     host: 'localhost'
//   },
//   database: {
//     url: 'postgres://localhost:5432/mydb',
//     pool: {
//       min: 0,
//       max: 20
//     }
//   }
// }

defineConfig<T, ConfigEnv extends ConfigEnvBase = ConfigEnvBase>

A type helper for defining configuration objects with proper typing and environment support.

Example:

import { defineConfig } from '@hyperse/config-loader';

interface MyConfig {
  port: number;
  database: {
    url: string;
  };
}

interface MyConfigEnv {
  mode: 'development' | 'production';
}

export default defineConfig<MyConfig, MyConfigEnv>((env) => {
  return {
    port: env.mode === 'production' ? 80 : 3000,
    database: {
      url:
        env.mode === 'production'
          ? 'postgres://prod:5432/mydb'
          : 'postgres://localhost:5432/mydb',
    },
  };
});

LoaderOptions

Configuration options for the loader:

interface LoaderOptions {
  // Project root directory
  projectCwd?: string;
  // Path to tsconfig.json
  tsconfig?: string;
  // Additional plugins
  plugins?: Plugin[];
  // External dependencies to exclude
  externals?: Array<RegExp | string>;
  // The function to exclude aggregated externals, if the function returns true, the module will be excluded.
  externalExclude?: (moduleId: RegExp | string) => boolean;
  // The function to create your custom loaders.
  createLoaders?: (
    options?: LoaderOptions,
    searchFrom?: string
  ) => ConfigLoaders;
}

Custom Loaders Options

externalExclude - A function to exclude specific modules from being marked as external. Takes a module ID (string or RegExp) and returns true if the module should be excluded from externalization. This is useful for keeping certain dependencies bundled even if they match the external patterns.

import { searchConfig } from '@hyperse/config-loader';

// Example 1: Basic usage with string patterns
const result1 = await searchConfig('myapp', undefined, {
  // Default external pattern
  externals: ['@hyperse/ts-node'],
  externalExclude: (moduleId) => {
    // Don't exclude modules that match these patterns
    if (typeof moduleId === 'string') {
      return !(
        moduleId.includes('@myorg/') || moduleId.includes('my-local-package')
      );
    }
    return true;
  },
});

// Example 2: Advanced usage with RegExp patterns
const result2 = await searchConfig('myapp', undefined, {
  // Multiple external patterns
  externals: ['@hyperse/ts-node', /^@vendor\//],
  externalExclude: (moduleId) => {
    if (typeof moduleId === 'string') {
      // Exclude specific packages from being marked as external
      return (
        moduleId.includes('@myorg/shared-utils') ||
        moduleId.includes('internal-tools')
      );
    }
    // For RegExp patterns, check specific module names
    return (
      moduleId.test('@myorg/shared-utils') || moduleId.test('internal-tools')
    );
  },
});

createLoaders - You can create custom loaders to handle specific file formats or add custom loading logic. Here's an example of how to create a custom loader for .toml files:

import { readFile } from 'fs/promises';
import {
  type ConfigLoaders,
  type LoaderOptions,
  searchConfig,
} from '@hyperse/config-loader';
import { parse } from '@iarna/toml';

// Create a custom loader for TOML files
const createTomlLoader = (
  options?: LoaderOptions,
  searchFrom?: string
): ConfigLoaders => {
  return {
    '.toml': async (filepath: string) => {
      try {
        const content = await readFile(filepath, 'utf-8');
        const config = parse(content);
        return { config, filepath };
      } catch (error) {
        throw new Error(
          `Failed to load TOML config from ${filepath}: ${error.message}`
        );
      }
    },
  };
};

// Use the custom loader
const result = await searchConfig('myapp', undefined, {
  createLoaders: createTomlLoader,
});

You can also reuse existing loaders from @hyperse/config-loader. The package exports several built-in loaders like tsLoader, jsonLoader, and yamlLoader that you can compose together to create custom loading behavior.

import type { ConfigLoaders, LoaderOptions } from '@hyperse/config-loader';
import { searchConfig, tsLoader } from '@hyperse/config-loader';

// Create a custom loader for TOML files
function createJsLoader(
  options?: LoaderOptions,
  searchFrom?: string
): ConfigLoaders {
  return {
    // Override JavaScript file resolver using tsLoader for better TypeScript support
    '.js': async (path: string, content: string) => {
      const { projectCwd, ...restLoaderOptions } = options || {};
      return tsLoader({
        plugins: [],
        externals: [],
        ...restLoaderOptions,
        projectCwd: projectCwd || searchFrom,
      })(path, content);
    },
  };
}

// Use the custom loader
const result = await searchConfig('myapp', undefined, {
  createLoaders: createJsLoader,
});

Supported File Formats

The config loader supports the following file formats:

  • .js (ESM and CommonJS)
  • .mjs (ESM)
  • .cjs (CommonJS)
  • .ts (TypeScript)
  • .mts (TypeScript ESM)
  • .json
  • .yaml/.yml

The loader will search for files in the following order:

  1. package.json (in the [moduleName] field)
  2. .[moduleName]rc
  3. .[moduleName]rc.json
  4. .[moduleName]rc.yaml
  5. .[moduleName]rc.yml
  6. .[moduleName]rc.js
  7. .[moduleName]rc.mjs
  8. .[moduleName]rc.cjs
  9. .[moduleName]rc.ts
  10. [moduleName].config.js
  11. [moduleName].config.mjs
  12. [moduleName].config.cjs
  13. [moduleName].config.ts
  14. [moduleName].config.mts

About

A config loader that searches for and loads configuration files for your program, with support for `type:module`, `esm`, and `cjs` module formats

Topics

Resources

License

Code of conduct

Stars

Watchers

Forks

Sponsor this project

Packages

No packages published

Contributors 2

  •  
  •