Strategy Pattern in Kawkab
The Strategy Pattern is one of the behavioral design patterns that allows you to isolate different algorithms within independent, interchangeable objects. In the Kawkab framework, you can use this pattern to enhance application flexibility, separating algorithms from the objects that depend on them.
What is the Strategy Pattern?
Simply put, the Strategy Pattern allows you to:
- Define a family of algorithms for a specific task.
- Switch the algorithm used at runtime without affecting other objects.
- Reduce duplication and make the code more extensible and maintainable.
When to Use the Strategy Pattern?
- When you have multiple ways or algorithms to solve a problem.
- When you need to switch the behavior of objects at runtime.
- To avoid long conditional statements (such as
if-else
orswitch-case
).
Creating a New Strategy in Kawkab
Creation Command
To easily add a new strategy in Kawkab, a custom command is provided:
npm run kawkab strategy:make <name> [module]
Parameters:
<name>
: The name of the strategy (e.g.,payment
,authentication
).[module]
(optional): The module to which the strategy will be added. If not specified, the strategy will be placed within the default modulemain
.
Practical Example:
npm run kawkab strategy:make payment
This example will generate:
- A strategy interface definition
IStrategy.ts
. - A strategy management class
PaymentStrategy.ts
. - An initial implementation of a specific strategy
StrategyA.ts
.
Generated File Structure
1. Strategy Interface (IStrategy.ts
)
Defines the contract that all strategies must adhere to:
export interface IStrategy {
execute(): void;
}
2. Main Strategy Class (PaymentStrategy.ts
)
Contains the logic responsible for determining and using the required strategy:
import { IStrategy } from './IStrategy';
import { StrategyA } from './StrategyA';
export class PaymentStrategy {
private strategy: IStrategy;
constructor(strategy: IStrategy) {
this.strategy = strategy;
}
static StrategyA(): PaymentStrategy {
return new PaymentStrategy(new StrategyA());
}
execute(): void {
return this.strategy.execute();
}
}
3. Specific Strategy Implementation (StrategyA.ts
)
Here, the specific algorithm logic for the strategy is written:
import { IStrategy } from './IStrategy';
export class StrategyA implements IStrategy {
execute(): void {
console.log('Executing Strategy A...');
// Strategy logic here
}
}
How to Use the Strategy?
Applying Strategies in a Controller
Let’s assume we want to use a login strategy (LoginStrategy
):
import { BaseController, inherit } from "kawkab";
import { LoginStrategy } from "../strategies/login/LoginStrategy";
export default class extends inherit(BaseController) {
async post() {
// Apply Strategy A
const strategy = LoginStrategy.StrategyA();
strategy.execute();
return {
status: true,
message: "Strategy executed successfully!"
};
}
}
Code Explanation:
LoginStrategy.StrategyA()
: Creates an instance of the strategy usingStrategyA
.execute()
: Executes the logic of the specified strategy.
Adding a New Strategy: StrategyB
1. Create the New Strategy File
Create a new file within the same module folder (e.g., strategies/login
) named StrategyB.ts
.
Content of StrategyB.ts
:
import { IStrategy } from './IStrategy';
export class StrategyB implements IStrategy {
execute(): void {
console.log('Executing Strategy B...');
// Strategy B logic
}
}
2. Update the Main Strategy Class
Update the LoginStrategy
class to support the new StrategyB
.
Updated code for LoginStrategy.ts
:
import { IStrategy } from './IStrategy';
import { StrategyA } from './StrategyA';
import { StrategyB } from './StrategyB';
export class LoginStrategy {
private strategy: IStrategy;
constructor(strategy: IStrategy) {
this.strategy = strategy;
}
// Add support for Strategy A
static StrategyA(): LoginStrategy {
return new LoginStrategy(new StrategyA());
}
// Add support for Strategy B
static StrategyB(): LoginStrategy {
return new LoginStrategy(new StrategyB());
}
execute(): void {
return this.strategy.execute();
}
}
3. Use Strategy B
Within a Controller:
You can now use StrategyB
in the same way as other strategies:
import { BaseController, inherit } from "kawkab";
import { LoginStrategy } from "../strategies/login/LoginStrategy";
export default class extends inherit(BaseController) {
async post() {
// Use Strategy B
const strategy = LoginStrategy.StrategyB();
strategy.execute();
return {
status: true,
message: "Strategy B executed successfully!"
};
}
}
How It Works?
-
LoginStrategy.StrategyB()
: Creates a new instance of the strategy using theStrategyB
class. -
execute()
: Executes the algorithm logic defined withinStrategyB
.
File Structure After Addition
/strategies
/login
- IStrategy.ts
- LoginStrategy.ts
- StrategyA.ts
- StrategyB.ts
Adding More Algorithms in the Future
To add a new strategy, you can repeat the same steps:
- Create a new class for the algorithm.
- Modify the main
LoginStrategy
class to support the new strategy.
Benefits of the Strategy Pattern
-
Flexibility and Runtime Switching:
- Behavior can be changed without affecting the main code.
-
Reduce Long Conditional Statements:
- Instead of using constructs like
if-else
, behaviors are managed by independent strategies.
- Instead of using constructs like
-
Improved Extensibility:
- Adding a new strategy simply means creating a new class adhering to the interface.
-
Separation of Responsibilities:
- Each object is responsible for executing specific logic only.
-
Ease of Testing:
- Each strategy can be tested individually without needing to test the entire system.
Best Practices
1. Organized Structure
Keep all strategies of a module in their own folder, for example:
/strategies
/login
- IStrategy.ts
- LoginStrategy.ts
- StrategyA.ts
- StrategyB.ts
2. Use Strong Interfaces
Using interfaces like IStrategy
ensures consistency and interchangeability of strategies.
3. Avoid Complex Logic Within Classes
Keep the strategy logic simple, clear, and separable.
Conclusion
The Strategy Pattern represents an ideal way to simplify handling multiple algorithms in your application. In the Kawkab framework, this pattern provides powerful tools that make creating and switching behaviors easier and more organized.
💡 Start applying the Strategy Pattern in your project and enjoy organized and flexible code!