@fnet/inter-service-js is a JavaScript library designed to facilitate communication between different services or components running in separate browser windows or frames. This project aims to simplify the integration of web applications that need to exchange messages or invoke functionalities from each other, efficiently creating a network of interconnected services within a browser environment.
The library functions by establishing communication channels between services using the browser's messaging capabilities such as the postMessage
API and MessageChannel
. By associating unique identifiers with each service, it ensures that messages are accurately routed to their intended destination. It includes mechanisms to find available routes for communication dynamically, ensuring robust and reliable message exchanges even in complex multi-window setups.
- Service Connection: Establishes connections to external services using unique service IDs, ensuring reliable routing of messages.
- Message Routing: Supports finding routers capable of directing communication to appropriate services or frames.
- Handler Registration: Allows services to register handlers for specific methods, enabling customized functionality exchange.
- Proxy Communication: Facilitates invoking methods on a service from another service without direct dependence.
- Event Handling: Built-in event emitter for managing event-driven interactions between services.
- Safety Checks: Includes checks to prevent self-connection and other potential misconfigurations.
@fnet/inter-service-js is valuable for developers looking to create interconnected web applications where components need to communicate efficiently across different environments. By abstracting and simplifying the communication process, it helps developers focus on building feature-rich applications without worrying about the intricacies of cross-window messaging.
The @fnet/inter-service-js
library provides a communication framework for connecting services running in different window contexts or frames. It facilitates message-passing between these services using the postMessage
API, making it easier to build applications with multiple services communicating in a loosely-coupled manner. The library includes functionality for connecting services, proxying methods across services, and managing communication channels securely and efficiently.
Before diving into the usage, it's important to understand some key concepts:
- Service: A JavaScript instance that can communicate with other services. Each service has a unique ID.
- Router: A service that can route messages between other services. A router is essential for communication between services in different windows/frames.
- Context: The window environment where a service runs. Each context has a unique ID.
- Handler: A function registered to a service that can be called by other services.
- Proxy: A method that forwards calls to another service.
To install this library, use npm or yarn. Simply run:
npm install @fnet/inter-service-js
or
yarn add @fnet/inter-service-js
To get started with @fnet/inter-service-js
, you need to initialize the library and create communication channels between services.
import interServiceJs from '@fnet/inter-service-js';
async function setupService() {
// Initialize a service with a unique ID
// Set router: true to enable this service to act as a router
const service = await interServiceJs({
id: 'my-service-id',
verbose: true, // Enable logging for debugging
router: true, // Enable router capabilities
});
// Example: Register a method that echoes the received message
service.register({
method: 'echo',
handler: (args) => {
console.log('Echo received:', args);
return args;
}
});
// Example: Connect to another service
try {
const result = await service.connect({
window: otherWindow,
serviceId: 'other-service-id'
});
console.log('Connected to service:', result);
} catch (error) {
console.error('Connection failed:', error.message);
}
}
setupService();
The typical communication flow between services follows these steps:
- Service Initialization: Each service is initialized with a unique ID.
-
Router Discovery: Services find routers using the
findRouter
method. -
Connection Establishment: Services connect to routers using the
connect
method. - Method Registration: Services register methods that can be called by other services.
-
Method Invocation: Services call methods on other services using the
proxy
method or registered proxy methods.
const service = await interServiceJs({
id: 'my-service-id', // Required: Unique service ID (lowercase with - or _)
verbose: true, // Optional: Enable logging (default: false)
router: true // Optional: Enable router capabilities (default: false)
});
service.register({
method: 'methodName', // Required: Name of the method
handler: (args, context) => {
// args: Arguments passed to the method
// context: Additional information about the call
return result; // Return value will be sent back to the caller
},
configurable: false // Optional: Whether the method can be overwritten (default: false)
});
service.registerProxy({
alias: 'localMethodName', // Required: Local name for the method
method: 'remoteMethodName', // Required: Name of the method on the remote service
service: 'remote-service-id', // Required: ID of the remote service
configurable: false // Optional: Whether the method can be overwritten (default: false)
});
const result = await service.proxy({
method: 'methodName', // Required: Name of the method to call
service: 'service-id', // Optional: ID of the target service
args: { /* arguments */ }, // Optional: Arguments to pass to the method
options: { // Optional: Additional options
callback: (event) => { // Optional: Callback for progress updates
console.log('Progress:', event);
},
timeout: 5000 // Optional: Custom timeout in milliseconds
}
});
const router = await service.findRouter({
timeout: 2000, // Optional: Maximum time to wait (default: 1000ms)
interval: 100, // Optional: Interval between attempts (default: 100ms)
where: { // Optional: Criteria for finding a specific router
id: 'router-id', // Optional: Match by service ID
serviceId: 'service-id', // Optional: Match by internal service ID
contextId: 'context-id', // Optional: Match by context ID
scriptId: 'script-id', // Optional: Match by script ID
client: 'client-id' // Optional: Match by connected client ID
}
});
const result = await service.connect({
window: targetWindow, // Required: Window object of the target service
serviceId: 'service-id' // Required: Internal service ID of the target
});
service.unregister({
method: 'methodName' // Required: Name of the method to unregister
});
service.dispose(); // Cleans up event listeners and resources
The library uses EventEmitter for event handling:
// Listen for connection events
service.on('connected', (client) => {
console.log('New client connected:', client);
});
// Remove event listener
service.off('connected', handlerFunction);
// Listen for an event only once
service.once('connected', handlerFunction);
You can use callbacks to receive progress updates during long-running operations:
const result = await service.proxy({
method: 'longOperation',
service: 'worker-service',
args: { /* arguments */ },
options: {
callback: (progressData) => {
console.log('Operation progress:', progressData);
updateProgressBar(progressData.percentage);
}
}
});
You can find routers that match specific criteria:
// Find a router in a specific context
const router = await service.findRouter({
where: { contextId: specificContextId }
});
// Find a router with specific connected clients
const router = await service.findRouter({
where: { client: ['client1', 'client2'] }
});
// Find a router by ID
const router = await service.findRouter({
where: { id: 'main-router' }
});
For complex applications, you can create a network of services:
// Main application with router capabilities
const mainService = await interServiceJs({
id: 'main-app',
router: true
});
// Feature-specific services
const authService = await interServiceJs({ id: 'auth-service' });
const dataService = await interServiceJs({ id: 'data-service' });
const uiService = await interServiceJs({ id: 'ui-service' });
// Connect all services to the main router
const router = await authService.findRouter();
await authService.connect(router);
await dataService.connect(router);
await uiService.connect(router);
// Register cross-service methods
authService.register({
method: 'login',
handler: async (credentials) => {
// Authentication logic
return { success: true, token: 'abc123' };
}
});
// Create proxy methods for easier access
uiService.registerProxy({
alias: 'login',
method: 'login',
service: 'auth-service'
});
// Now UI service can call auth service methods directly
const loginResult = await uiService.login({
username: 'user',
password: 'pass'
});
-
Use Descriptive Service IDs: Choose meaningful IDs for your services to make debugging easier.
-
Enable Verbose Mode During Development: Set
verbose: true
during development to see detailed logs. -
Implement Error Handling: Always use try/catch blocks when calling methods on other services.
-
Set Appropriate Timeouts: Adjust timeouts based on the expected response time of operations.
-
Use Event Listeners for Connection Management: Listen for 'connected' events to track service connections.
-
Clean Up Resources: Call
dispose()
when a service is no longer needed to prevent memory leaks. -
Structure Your Services Logically: Divide your application into services based on functionality.
-
Use Proxy Methods for Cleaner Code: Register proxy methods to make remote calls look like local calls.
-
Connection Failures:
- Ensure the target window is accessible
- Check that the service ID is correct
- Verify that the router is enabled on the target service
-
Method Not Found Errors:
- Ensure the method is registered on the target service
- Check that the service ID is correct in the proxy call
-
Timeout Errors:
- Increase the timeout value for long-running operations
- Check for network issues or high latency
-
Cross-Origin Issues:
- Ensure both windows are from the same origin or have appropriate CORS settings
-
Enable verbose mode to see detailed logs:
const service = await interServiceJs({ id: 'service-id', verbose: true });
-
Use browser developer tools to inspect postMessage events.
-
Add custom logging to your handlers to track method calls.
The development of @fnet/inter-service-js
leverages the nanoid
library for generating unique identifiers and the eventemitter3
for event handling. Special thanks to the developers and contributors of these libraries for providing essential functionality that enhances our library's capabilities.
$schema: https://json-schema.org/draft/2020-12/schema
type: object
properties:
id:
type: string
pattern: ^([a-z\-_]+)$
description: The unique service ID. Should be lowercase and may include '-' and '_'
verbose:
type: boolean
description: Enable verbose output for debugging
router:
type: boolean
description: Indicate if the service can act as a router
required:
- id