Creating services
Learn how to define and register services with Aurelia's dependency injection container.
Overview
Creating injectable services is crucial for building maintainable applications in Aurelia. Services encapsulate shared functionalities such as business logic or data access, and can be easily injected where needed. This guide will demonstrate various methods for creating injectable services, including the use of DI.createInterface()
and directly exporting classes.
Using DI.createInterface()
to Create Injectable Services
DI.createInterface()
to Create Injectable ServicesCreating an Interface Token with a Default Implementation
DI.createInterface()
allows you to create an injection token that can be used with a default implementation:
import { DI } from 'aurelia';
export class LoggerService {
log(message: string) {
console.log(message);
}
}
export const ILoggerService = DI.createInterface<ILoggerService>('ILoggerService', x => x.singleton(LoggerService));
// Export type equal to the class to create an interface
export type ILoggerService = LoggerService;
Here, ILoggerService
is both an injection token and a type representing the LoggerService
class. This simplifies the process as you won't need to manually define an interface with all the methods of the LoggerService
.
Creating an Interface Token Without a Default Implementation
You can also create an interface token without a default implementation for more flexibility:
import { DI } from 'aurelia';
export class LoggerService {
log(message: string) {
// Logging logic
}
}
// Export type equal to the class to create an interface
export type ILoggerService = LoggerService;
export const ILoggerService = DI.createInterface<ILoggerService>('ILoggerService');
In this scenario, you must register the implementation with the DI container:
import { Registration } from 'aurelia';
container.register(
Registration.singleton(ILoggerService, LoggerService)
);
Exporting Classes Directly for Injectable Services
Simple Class Export
For services that don't require an interface, simply exporting a class is sufficient:
export class AuthService {
isAuthenticated(): boolean {
// Authentication logic
return true;
}
}
This service can be auto-registered or manually registered:
import { Registration } from 'aurelia';
container.register(
Registration.singleton(AuthService, AuthService)
);
Decorator-based Registration
Aurelia supports decorators for controlling service registration:
import { singleton } from 'aurelia';
@singleton()
export class AuthService {
isAuthenticated(): boolean {
// Authentication logic
return true;
}
}
The @singleton()
decorator ensures a single instance of AuthService
within the DI container.
Exporting a Type Equal to the Class
Sometimes, you may want to export a type equal to the class, which can serve as an interface. This approach reduces redundancy and keeps the service definition concise:
export class PaymentProcessor {
processPayment(amount: number) {
// Payment processing logic
}
}
// Export type equal to the class to create an interface
export type IPaymentProcessor = PaymentProcessor;
You can then use this type for injection and for defining the shape of your service without having to declare all methods explicitly. The important thing to remember is that a TypeScript type
alias is removed during compilation. That means IPaymentProcessor
exists only for tooling and static analysis—there is no runtime token by that name. When you ask the container for an instance, supply the class (or another runtime value) as the key:
import { resolve } from 'aurelia';
import { PaymentProcessor } from './payment-processor';
import type { IPaymentProcessor } from './payment-processor';
export class CheckoutWorkflow {
private readonly paymentProcessor: IPaymentProcessor = resolve(PaymentProcessor);
completeCheckout(amount: number) {
this.paymentProcessor.processPayment(amount);
}
}
If you'd rather rely on constructor injection, pair the runtime token with the @inject
decorator while keeping the parameter typed as the alias:
import { inject } from 'aurelia';
import { PaymentProcessor } from './payment-processor';
import type { IPaymentProcessor } from './payment-processor';
@inject(PaymentProcessor)
export class CheckoutController {
constructor(private readonly paymentProcessor: IPaymentProcessor) {}
completeCheckout(amount: number) {
this.paymentProcessor.processPayment(amount);
}
}
This pattern keeps the runtime registration focused on PaymentProcessor
while exposing the alias everywhere else for enhanced editor and refactoring support.
Conclusion
Aurelia provides multiple approaches to creating and registering injectable services, including DI.createInterface()
and directly exporting classes or types. By choosing the method that best fits your needs, you can achieve a clean and flexible architecture with easily injectable services, promoting code reuse and separation of concerns across your application.
Last updated
Was this helpful?