Research
Security News
Quasar RAT Disguised as an npm Package for Detecting Vulnerabilities in Ethereum Smart Contracts
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
ts-configurable
Advanced tools
Make all properties of a class configurable using only one decorator!
Let's be real, configuration is no fun. We all just want to set up our project's configuration as easily as possible and then move on to build whatever we actually wanted to build. But what if I told you that you can meet all your configuration needs with a single line of code and then never have to worry about it again?
Simply create a class defining all configuration properties, tack on the @Configurable()
decorator and off you go! Take advantage of type-safety, environment variable loading, command line argument parsing and much more right out-of-the-box.
Install via npm i ts-configurable
and create a config class with default values for each property:
// server-config.ts
import { Configurable } from 'ts-configurable';
@Configurable()
class ServerConfig {
host = 'localhost';
port = 3000;
}
console.log(new ServerConfig());
Due to the @Configurable()
decorator, the values for all properties can now be set via environment variables or command line arguments:
# Default config instance
$ ts-node server-config.ts
ServerConfig { host: 'localhost', port: 3000 }
# Change port via command line argument
$ ts-node server-config.ts --port=4200
ServerConfig { host: 'localhost', port: 4200 }
# Change host via environment variable
$ host=0.0.0.0 ts-node server-config.ts
ServerConfig { host: '0.0.0.0', port: 3000 }
# Throw an error if a value with a different type was assigned
$ port=random ts-node server-config.ts
TypeError: Property 'ServerConfig.port' is of type number but a value of type string ('"random"') was assigned!
strictTypeChecking
option)url
from host
and port
)get debugPort() { return this.port + 1000; }
)enforceReadonly
option)Configurable(options?: IDecoratorOptions
)
Class decorator for marking a class configurable: The values of all class properties can be set using the following sources, listed by priority (1 = highest):
The final values for the config instance's properties are calculated upon instantiation.
boolean
Enforce that all properties are read-only by using Object.freeze()
(default: true)
false
| DotenvConfigOptionsApply environment variables from a file to the current process.env
false
| IArgvOptions
Whether to parse command line arguments (default: true)
string
Prefix for command line arguments (default: no prefix)
false
| IEnvOptions
Whether to parse environment variables (default: true)
boolean
Whether to lower-case environment variables (default: false)
string
Prefix for environment variables (default: no prefix)
string
Seperator for environment variables (default: '__')
boolean
Attempt to parse well-known values (e.g. 'false', 'null', 'undefined' and JSON values) into their proper types (default: true)
boolean
Throw an error if a config entry is set to a value of a different type than the default value (e.g. assigning a number to a string property) (default: true)
boolean
Throw an error if a config entry is set to a value of a different structure than the default value (e.g., assigning an object to a primitive property) (default: true)
false
| IDecryptionOptions
Whether to attempt decryption of encrypted configuration values (default: false)
By extending the BaseConfig
class, both the options passed to the @Configurable()
decorator as well as the default values assigned to the instance properties can be overriden via the constructor during instantiation:
// server-config.ts
import { Configurable, BaseConfig } from 'ts-configurable';
@Configurable({ parseEnv: { prefix: '' } })
class ServerConfig extends BaseConfig<ServerConfig> {
host = 'localhost';
port = 3000;
}
const config = new ServerConfig({
options: {
parseEnv: false,
},
config: {
host: '0.0.0.0',
},
});
console.log(config);
While activating the parsing of environment variables inside the decorator options, we override this setting again in the constructor provided options and set it to false
, therefore disabling any environment variable parsing. Additionally, we override the default value for the host
property from localhost
to 0.0.0.0
inside the constructor as well:
# Ignore env variable "port", override host variable via constructor
$ port=4200 ts-node server-config.ts
ServerConfig { host: '0.0.0.0', port: 3000 }
For nested configuration properties, it is recommended to define a new type for keeping type safety:
// order-pizza.ts
import { Configurable, BaseConfig } from 'ts-configurable';
type TOrder = Partial<{
recipient: string;
priceTotal: number;
delivered: boolean;
billing: Partial<{ address: string }>;
}>;
class BasePizzaConfig extends BaseConfig<BasePizzaConfig> {
id = 5;
topping = 'cheese';
rating = null;
ingredients = ['tomato', 'bread'];
order: TOrder = {
recipient: 'John Doe',
priceTotal: 6.2,
delivered: true,
billing: {
address: 'Jerkstreet 53a, 1234 Whatevertown',
},
};
}
const pizzaConfig = new PizzaConfig({ topping: 'bacon' });
console.log(JSON.stringify(pizzaConfig, null, 2));
Accessing nested properties is done using the __
separator for environment variables (can be configured via parseEnv.separator
) and the dot notation (.
) for command line arguments:
export pizza_order__delivered=false
$ ts-node order-pizza.ts --order.recipient=Jonny"
{
"id": 5,
"topping": "bacon",
"rating": null,
"ingredients": [
"tomato",
"bread"
],
"order": {
"recipient": "Jonny",
"priceTotal": 6.2,
"delivered": false,
"billing": {
"address": "Jerkstreet 53a, 1234 Whatevertown"
}
}
}
It can be useful during development to keep a local .env
file containing configuration values that the developer wants to set for himself. The file is typically excluded from version control (e.g., by adding it to the .gitignore
of the repository). When using the @Configurable()
decorator, environment variables specified in a .env
file located in the current working directory are loaded into process.env
before the configuration object is created. However, another path can be specified using the loadEnvFromFile.path
option.
This functionality is implemented using the dotenv package. You can find out more by reading the package's documentation:
If you want to explicitly set a field to false
instead of just leaving it undefined
or to override a default you can add a no-
before the key: --no-key
.
// pay-pizza.ts
import { Configurable } from 'ts-configurable';
@Configurable()
class PizzaConfig {
cash = false;
debit = true;
paypal = true;
}
console.log(new PizzaConfig());
$ ts-node pay-pizza.ts --cash --no-paypal
{ cash: true, debit: true, paypal: false }
The ts-configurable package encourages developers to keep all configuration values static throughout the lifetime of the application. All sources (environment variables, command line arguments, ...) are parsed and merged during the instantiation of the configuration class object. After an object has been created, the object should be considered read-only, which can be enforced using the enforceReadonly
option. This ensures that the configuration object is not modified at runtime (using Object.freeze). Additionally, all class properties can be marked with the TypeScript readonly
modifier to prevent assignments at compile time. This way, the developer gets instant feedback from his IDE if he accidentially tries to set a read-only configuration value.
Keeping all configuration values read-only after initialization has several benefits:
Only a restart of the application triggers a reloading of the configuration.
The ability to specify configuration values using multiple sources (env vars, cmd args) necessitates the specification of a configuration source hierarchy. Each property assignment specified via one of the sources is always overriden by the same property assignment of a source higher in the loading hierarchy. The values specified via a source take precedence over all values specified in sources that are below in the loading hierarchy.
The configuration loading hierarchy is listed below (1 = highest, 4 = lowest):
This behavior is implemented using the nconf package. In short, each configuration source is converted into a partially filled configuration object. All those configuration objects are then merged with the precedence order listed above. This means that individual values can be set, even for nested properties!
It is possible to provide encrypted configuration values. This is useful for secrets that should not be checked into source control but should be available as soon as the application is in possession of a single (or multiple) decryption secrets instead of having to provide each secret configuration value via environment variables. The decryption secrets are provided via the decryption
option. The following decryption secret types can be specified:
raw
: the secret is directly provided as a stringenv
: the secret is read from the environment variable with the specified name (via: environmentVariable
)file
: the secret is read from the file with the specified filepath (via: filepath
)If multiple decryption secrets are provided, ts-configurable will attempt to decrypt each encrypted configuration value with all of the available keys. If a key could not be loaded, it is ignored. If an encrypted configuration value could not be successfully decrypted, it is left in its original, encrypted form.
You can encrypt configuration values via the encrypt(secret: string, plaintext: string)
method that is being exported by the ts-configurable
package. The typical workflow would be to encrypt the secret configuration value e.g., via a separate node invocation and then using the resulting ciphertext for configuring the application. Here is a simple ts-node
example on how to encrypt a secret configuration value:
import { encrypt } from 'ts-configurable';
const secret = 'secret_key';
const value = 'sensitive_api_key';
const cipher = encrypt(secret, value);
console.log(`CIPHER: ${cipher}`);
// output: CIPHER: $ENC$.6aaa302f81c67ce6fe2da026ba7a2d0b.91b5db5ffea63e0d2c89c350760ca54152e6d4b5526ef06f8d6952407120891c
This error occurs when a custom class extends the BaseConfig
class and the TypeScript compiler target is below ES6. The root cause for this issue is that the ts-configurable package is compiled with the ES6 target and an ES5 class cannot extend an ES6 class (classes are compiled to functions in ES5 while ES6 natively supports classes). Therefore, this error can be fixed by setting the compiler target to ES6 or higher:
// tsconfig.json
...
"compilerOptions": {
"target": "es6"
Here are some example applications showcasing how to use the ts-configurable package:
You are welcome to contribute to the ts-configurable GitHub repository! All infos can be found here: How to contribute
FAQs
Make all properties of a class configurable using only one decorator!
We found that ts-configurable demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Research
Security News
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
Security News
Research
A supply chain attack on Rspack's npm packages injected cryptomining malware, potentially impacting thousands of developers.
Research
Security News
Socket researchers discovered a malware campaign on npm delivering the Skuld infostealer via typosquatted packages, exposing sensitive data.