Skip to content

Commit 9021223

Browse files
author
samuel.casanova
committed
add proxy design pattern real world example
1 parent bcce5af commit 9021223

File tree

2 files changed

+114
-0
lines changed

2 files changed

+114
-0
lines changed

src/Proxy/RealWorld/Output.txt

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
Client: Executing the client code with a real weather service:
2+
The weather forecast is 4.539002575927763º average temperature and 54.50427119342287% max precipitation probability.
3+
The weather forecast is 12.87183981383º average temperature and 60.041907992745514% max precipitation probability.
4+
The weather forecast is 27.63639710822103º average temperature and 2.2777239912981972% max precipitation probability.
5+
6+
Client: Executing the same client code with a proxied weather service:
7+
Requesting forecast on date 2023-06-01T07:45:08.143Z.
8+
Invalid cache. Calling real weather service...
9+
Request processed in 1002 milliseconds
10+
The weather forecast is 27.65024492643842º average temperature and 46.87027596334876% max precipitation probability.
11+
Requesting forecast on date 2023-06-01T07:45:09.146Z.
12+
Request processed in 0 milliseconds
13+
The weather forecast is 27.65024492643842º average temperature and 46.87027596334876% max precipitation probability.
14+
Requesting forecast on date 2023-06-01T07:45:09.147Z.
15+
Request processed in 0 milliseconds
16+
The weather forecast is 27.65024492643842º average temperature and 46.87027596334876% max precipitation probability.

src/Proxy/RealWorld/index.ts

+98
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
/**
2+
* EN: Real World Example for the Proxy Design Pattern
3+
*
4+
* Need: Cache and log access to an external weather service SDK
5+
*
6+
* Solution: Create a proxy class to cache the SDK calls and log the requests
7+
*/
8+
9+
/**
10+
* EN: The WeatherService defines the SDK interface and response
11+
*/
12+
interface WeatherService {
13+
request(): Promise<WeatherForecast>;
14+
}
15+
16+
interface WeatherForecast {
17+
avgTemperature: number;
18+
maxPrecipitationProbability: number;
19+
}
20+
21+
/**
22+
* EN: The real service emulates a network request to an external service with
23+
* a 1 second delay
24+
*/
25+
class RealWeatherServiceSDK implements WeatherService {
26+
public async request(): Promise<WeatherForecast> {
27+
const randomWeatherForecast = {
28+
avgTemperature: Math.random() * 35,
29+
maxPrecipitationProbability: Math.random() * 100,
30+
};
31+
32+
return new Promise((resolve) => {
33+
setTimeout(() => resolve(randomWeatherForecast), 1000);
34+
});
35+
}
36+
}
37+
38+
/**
39+
* EN: The Proxy implements the same interface than the real service. However
40+
* the proxy uses an internal cache to store responses. Subsequent calls to the
41+
* proxied service will go faster as they won't call the real slow service. The
42+
* proxy also logs useful information about the cache usage and timmings.
43+
*/
44+
class ProxyWeatherService implements WeatherService {
45+
private cachedResponse : WeatherForecast;
46+
private cacheDate : Date;
47+
private expirationTimeInMillis = 24 * 60 * 60 * 1000;
48+
49+
constructor(private realWeatherService: WeatherService) {
50+
}
51+
52+
public async request(): Promise<WeatherForecast> {
53+
console.log(`Requesting forecast on date ${new Date().toISOString()}.`);
54+
const initialTime = Date.now();
55+
if (this.isCacheExpired()) {
56+
console.log('Invalid cache. Calling real weather service...');
57+
this.setCache(await this.realWeatherService.request());
58+
}
59+
const requestTimeInMillis = Date.now() - initialTime;
60+
console.log(`Request processed in ${requestTimeInMillis} milliseconds`);
61+
return this.cachedResponse;
62+
}
63+
64+
private isCacheExpired() : boolean {
65+
return this.cachedResponse ?
66+
Date.now() > this.cacheDate.getTime() + this.expirationTimeInMillis :
67+
true;
68+
}
69+
70+
private setCache(weatherForecast: WeatherForecast) {
71+
this.cachedResponse = weatherForecast;
72+
this.cacheDate = new Date();
73+
}
74+
}
75+
76+
/**
77+
* EN: The client code works with real and proxied services
78+
*/
79+
async function clientCode(weatherService: WeatherService) {
80+
for (let i = 0; i < 3; i += 1) {
81+
const weatherForecast = await weatherService.request();
82+
console.log(`The weather forecast is ${weatherForecast.avgTemperature}º average temperature and ${weatherForecast.maxPrecipitationProbability}% max precipitation probability.`);
83+
}
84+
}
85+
86+
async function main() {
87+
console.log('Client: Executing the client code with a real weather service:');
88+
const realWeatherService = new RealWeatherServiceSDK();
89+
await clientCode(realWeatherService);
90+
91+
console.log('');
92+
93+
console.log('Client: Executing the same client code with a proxied weather service:');
94+
const proxy = new ProxyWeatherService(realWeatherService);
95+
await clientCode(proxy);
96+
}
97+
98+
main();

0 commit comments

Comments
 (0)