2.48.0 • Published 6 months ago

@memberjunction/actions v2.48.0

Weekly downloads
-
License
ISC
Repository
-
Last release
6 months ago

@memberjunction/actions

The @memberjunction/actions library provides the core server-side infrastructure for the MemberJunction Actions Framework. It includes base classes for actions and filters, the action execution engine, and support for entity-specific actions.

Overview

The Actions Framework is a powerful system for creating reusable, parameterized business logic that can be executed on demand. Actions are "verbs" in the MemberJunction ecosystem - they perform specific tasks and can be triggered through various mechanisms including entity events, API calls, or scheduled jobs.

IMPORTANT: This library should only be imported on the server side.

Key Features

  • Action Engine: Central execution engine for running actions with parameter validation, filtering, and logging
  • Entity Actions: Actions that can be triggered on entity lifecycle events (create, update, delete)
  • Code Generation: AI-powered automatic code generation for actions based on natural language prompts
  • Action Filters: Pre-execution filters to control when actions should run
  • Transaction Support: Built-in transaction management for complex multi-step operations
  • Comprehensive Logging: Automatic logging of all action executions with parameters and results

Installation

npm install @memberjunction/actions

Dependencies

This package depends on several other MemberJunction packages:

  • @memberjunction/global - Global utilities and class factory
  • @memberjunction/core - Core MJ functionality and base classes
  • @memberjunction/actions-base - Base types and interfaces for actions
  • @memberjunction/core-entities - Entity definitions
  • @memberjunction/ai - AI integration capabilities
  • @memberjunction/aiengine - AI engine functionality
  • @memberjunction/doc-utils - Documentation utilities

Usage

Creating a Custom Action

To create a custom action, extend the BaseAction class and implement the InternalRunAction method:

import { BaseAction } from '@memberjunction/actions';
import { ActionResultSimple, RunActionParams } from '@memberjunction/actions-base';
import { RegisterClass } from '@memberjunction/global';

@RegisterClass(BaseAction, 'MyCustomAction')
export class MyCustomAction extends BaseAction {
    protected async InternalRunAction(params: RunActionParams): Promise<ActionResultSimple> {
        // Access input parameters
        const inputParam = params.Params.find(p => p.Name === 'InputValue');
        
        // Perform your action logic
        const result = await this.performBusinessLogic(inputParam?.Value);
        
        // Return the result
        return {
            Success: true,
            ResultCode: 'SUCCESS',
            Message: 'Action completed successfully',
            Params: params.Params // Include any output parameters
        };
    }
    
    private async performBusinessLogic(value: any): Promise<any> {
        // Your custom logic here
        return value;
    }
}

Running Actions with ActionEngine

The ActionEngineServer class provides the main interface for executing actions:

import { ActionEngineServer } from '@memberjunction/actions';
import { RunActionParams } from '@memberjunction/actions-base';

// Get the singleton instance
const engine = ActionEngineServer.Instance;

// Configure the engine (only needs to be done once)
await engine.Config(false, currentUser);

// Run an action by ID
const result = await engine.RunActionByID({
    ActionID: 'your-action-id',
    ContextUser: currentUser,
    Params: [
        { Name: 'InputParam', Value: 'some value', Type: 'string' }
    ]
});

// Run an action with full parameters
const params: RunActionParams = {
    Action: actionEntity, // ActionEntity instance
    ContextUser: currentUser,
    Filters: [], // Optional filters
    Params: [
        { Name: 'InputParam', Value: 'some value', Type: 'string' }
    ]
};

const result = await engine.RunAction(params);

Entity Actions

Entity Actions are triggered automatically during entity lifecycle events. To work with entity actions, use the EntityActionEngineServer:

import { EntityActionEngineServer } from '@memberjunction/actions';
import { EntityActionInvocationParams } from '@memberjunction/actions-base';

const entityActionEngine = EntityActionEngineServer.Instance;

// Run an entity action
const params: EntityActionInvocationParams = {
    EntityAction: entityActionEntity,
    InvocationType: invocationTypeEntity, // e.g., 'BeforeCreate', 'AfterUpdate'
    EntityObject: entityInstance,
    ContextUser: currentUser
};

const result = await entityActionEngine.RunEntityAction(params);

Creating Custom Action Filters

Action filters determine whether an action should run. Create custom filters by extending BaseActionFilter:

import { BaseActionFilter } from '@memberjunction/actions';
import { RunActionParams } from '@memberjunction/actions-base';
import { ActionFilterEntity } from '@memberjunction/core-entities';
import { RegisterClass } from '@memberjunction/global';

@RegisterClass(BaseActionFilter, 'MyCustomFilter')
export class MyCustomFilter extends BaseActionFilter {
    protected async InternalRun(
        params: RunActionParams, 
        filter: ActionFilterEntity
    ): Promise<boolean> {
        // Implement your filter logic
        // Return true to allow action execution, false to skip
        return params.ContextUser.IsActive === true;
    }
}

API Reference

Classes

ActionEngineServer

The main engine for executing actions.

Methods:

  • RunAction(params: RunActionParams): Promise<ActionResult> - Executes an action with full control over parameters
  • RunActionByID(params: RunActionByNameParams): Promise<ActionResult> - Convenience method to run an action by its ID
  • Config(forceRefresh?: boolean, contextUser?: UserInfo): Promise<void> - Configures the engine (inherited from base)

BaseAction

Abstract base class for all actions.

Methods:

  • Run(params: RunActionParams): Promise<ActionResultSimple> - Public method called by the engine
  • InternalRunAction(params: RunActionParams): Promise<ActionResultSimple> - Abstract method to implement action logic

EntityActionEngineServer

Engine specifically for entity-related actions.

Methods:

  • RunEntityAction(params: EntityActionInvocationParams): Promise<EntityActionResult> - Executes an entity action

BaseActionFilter

Abstract base class for action filters.

Methods:

  • Run(params: RunActionParams, filter: ActionFilterEntity): Promise<boolean> - Public method called by the engine
  • InternalRun(params: RunActionParams, filter: ActionFilterEntity): Promise<boolean> - Abstract method to implement filter logic

ActionEntityServerEntity

Server-side entity class for Actions with AI-powered code generation.

Key Features:

  • Automatic code generation from natural language prompts
  • Code validation and improvement through AI
  • Library dependency management
  • Transaction-safe save operations

Types and Interfaces

The package exports all types from @memberjunction/actions-base, including:

  • RunActionParams - Parameters for running an action
  • ActionResult - Detailed result of action execution
  • ActionResultSimple - Simplified action result
  • ActionParam - Parameter definition for actions
  • EntityActionInvocationParams - Parameters for entity actions
  • EntityActionResult - Result of entity action execution

Entity Action Invocation Types

The framework supports various invocation types for entity actions:

Single Record Operations

  • Read - Triggered when reading an entity
  • BeforeCreate - Before creating a new record
  • AfterCreate - After creating a new record
  • BeforeUpdate - Before updating a record
  • AfterUpdate - After updating a record
  • BeforeDelete - Before deleting a record
  • AfterDelete - After deleting a record

Multiple Record Operations

  • List - Actions operating on a list of records
  • View - Actions operating on records from a view

Validation

  • Validate - Special invocation type for validation logic

Code Generation

The framework includes sophisticated AI-powered code generation capabilities:

  1. Natural Language Input: Define action behavior using plain English in the UserPrompt field
  2. Automatic Code Generation: The system generates TypeScript code based on your prompt
  3. Code Validation: Generated code is automatically validated and improved
  4. Library Management: Automatic tracking and importing of required libraries

Example workflow:

// In your database, create an Action record with:
// Name: "SendWelcomeEmail"
// Type: "Generated"
// UserPrompt: "Send a welcome email to a new user with their name and registration date"

// The system will automatically generate the implementation code

Best Practices

  1. Action Naming: Use clear, descriptive names for actions (e.g., SendInvoiceEmail, CalculateOrderTotal)

  2. Parameter Design: Design action parameters to be reusable and flexible:

    params: [
        { Name: 'EmailTemplate', Type: 'string', ValueType: 'Scalar' },
        { Name: 'RecipientUser', Type: 'User', ValueType: 'BaseEntity Sub-Class' },
        { Name: 'EmailSent', Type: 'boolean', ValueType: 'Scalar', IsInput: false }
    ]
  3. Error Handling: Always include proper error handling in your actions:

    try {
        // Action logic
        return { Success: true, ResultCode: 'SUCCESS', Message: 'Completed' };
    } catch (error) {
        return { Success: false, ResultCode: 'ERROR', Message: error.message };
    }
  4. Logging: The framework automatically logs action executions, but include additional logging for debugging:

    import { LogError, LogStatus } from '@memberjunction/core';
    
    LogStatus('Starting custom action processing...');
  5. Transaction Management: Use transaction groups for multi-step operations:

    const tg = await metadata.CreateTransactionGroup();
    try {
        // Multiple operations
        await tg.Submit();
    } catch (error) {
        // Automatic rollback on error
    }

Integration with Other MJ Packages

This package integrates seamlessly with:

  • @memberjunction/core: Provides base entity functionality and metadata access
  • @memberjunction/ai: Enables AI-powered code generation
  • @memberjunction/core-entities: Provides strongly-typed entity classes
  • @memberjunction/global: Manages class registration and instantiation

Advanced Topics

Custom Action Engines

You can create custom action engines by extending ActionEngineServer:

@RegisterClass(BaseEngine, 'ActionEngineBase', 1) // Higher priority
export class CustomActionEngine extends ActionEngineServer {
    protected async ValidateInputs(params: RunActionParams): Promise<boolean> {
        // Custom validation logic
        return super.ValidateInputs(params);
    }
    
    protected async RunFilters(params: RunActionParams): Promise<boolean> {
        // Custom filter logic
        return super.RunFilters(params);
    }
}

Script Evaluation in Entity Actions

Entity actions support dynamic script evaluation for parameter mapping:

// In EntityActionParam configuration:
{
    ValueType: 'Script',
    Value: `
        const user = EntityActionContext.entityObject;
        EntityActionContext.result = user.Email.toUpperCase();
    `
}

Troubleshooting

  1. Action Not Found: Ensure your action class is properly registered with @RegisterClass
  2. Code Generation Fails: Check that AI models are configured and API keys are set
  3. Filter Not Running: Verify filter is associated with the action in metadata
  4. Entity Action Not Triggering: Confirm invocation type matches the entity operation

License

This package is part of the MemberJunction ecosystem. See the main repository for license information.

2.27.1

10 months ago

2.23.2

10 months ago

2.46.0

7 months ago

2.23.1

10 months ago

2.27.0

10 months ago

2.34.0

8 months ago

2.30.0

9 months ago

2.19.4

11 months ago

2.19.5

11 months ago

2.19.2

11 months ago

2.19.3

11 months ago

2.19.0

11 months ago

2.19.1

11 months ago

2.15.2

11 months ago

2.34.2

8 months ago

2.15.0

11 months ago

2.34.1

8 months ago

2.15.1

11 months ago

2.38.0

7 months ago

2.45.0

7 months ago

2.22.1

11 months ago

2.22.0

11 months ago

2.41.0

7 months ago

2.22.2

11 months ago

2.26.1

10 months ago

2.26.0

10 months ago

2.33.0

8 months ago

2.18.3

11 months ago

2.18.1

11 months ago

2.18.2

11 months ago

2.18.0

11 months ago

2.37.1

8 months ago

2.37.0

8 months ago

2.14.0

12 months ago

2.21.0

11 months ago

2.44.0

7 months ago

2.40.0

7 months ago

2.29.0

10 months ago

2.29.2

10 months ago

2.29.1

10 months ago

2.25.0

10 months ago

2.48.0

6 months ago

2.32.0

9 months ago

2.32.2

9 months ago

2.32.1

9 months ago

2.17.0

11 months ago

2.13.4

12 months ago

2.36.0

8 months ago

2.13.2

1 year ago

2.13.3

1 year ago

2.13.0

1 year ago

2.36.1

8 months ago

2.13.1

1 year ago

2.43.0

7 months ago

2.20.2

11 months ago

2.20.3

11 months ago

2.20.0

11 months ago

2.20.1

11 months ago

2.28.0

10 months ago

2.47.0

7 months ago

2.24.1

10 months ago

2.24.0

10 months ago

2.31.0

9 months ago

2.12.0

1 year ago

2.39.0

7 months ago

2.16.1

11 months ago

2.35.1

8 months ago

2.35.0

8 months ago

2.16.0

11 months ago

2.42.1

7 months ago

2.42.0

7 months ago

2.23.0

10 months ago

2.11.0

1 year ago

2.10.0

1 year ago

2.9.0

1 year ago

2.8.0

1 year ago

2.7.0

1 year ago

2.6.1

1 year ago

2.5.2

1 year ago

2.6.0

1 year ago

2.7.1

1 year ago

2.5.1

1 year ago

2.5.0

1 year ago

2.4.1

1 year ago

2.4.0

1 year ago

2.3.3

1 year ago

2.3.2

1 year ago

2.3.1

1 year ago

2.3.0

1 year ago

2.2.1

1 year ago

2.2.0

1 year ago

2.1.5

1 year ago

2.1.4

1 year ago

2.1.3

1 year ago

2.1.2

1 year ago

2.1.1

1 year ago

2.1.0

1 year ago

2.0.0

1 year ago

1.8.1

2 years ago

1.8.0

2 years ago

1.7.1

2 years ago

1.7.0

2 years ago

1.6.1

2 years ago

1.6.0

2 years ago

1.5.3

2 years ago

1.5.2

2 years ago

1.5.1

2 years ago

1.5.0

2 years ago

1.4.1

2 years ago