Skip to content

Commit

Permalink
Feature: Record and Replay Backend Traffic (#49)
Browse files Browse the repository at this point in the history
  • Loading branch information
aniket-k-mukherjee authored Oct 1, 2024
1 parent 804fdb1 commit ddd3638
Show file tree
Hide file tree
Showing 11 changed files with 586 additions and 31 deletions.
18 changes: 16 additions & 2 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 3 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
{
"name": "appium-interceptor",
"version": "1.0.0-beta.10",
"version": "1.0.0-beta.11",
"description": "Appium 2.0 plugin to mock api calls for android apps",
"main": "./lib/index.js",
"types": "./lib/types/index.d.ts",
"scripts": {
"build": "npx tsc",
"test": "mocha --require ts-node/register -p test/plugin.spec.js --exit --timeout 260000",
"record-replay-test": "mocha --require ts-node/register -p test/record.replay.js --exit",
"prepublish": "npx tsc",
"lint": "eslint '**/*.js' --fix",
"prettier": "prettier '**/*.js' --write --single-quote",
Expand Down Expand Up @@ -104,6 +105,7 @@
],
"dependencies": {
"@appium/support": "^4.1.11",
"queue-typescript": "^1.0.1",
"ajv": "^6.12.6",
"appium-adb": "^11.0.9",
"axios": "^0.27.0",
Expand Down
6 changes: 5 additions & 1 deletion src/api-sniffer.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { RequestInfo, SniffConfig } from './types';
import { doesUrlMatch } from './utils/proxy';
import log from './logger';

export class ApiSniffer {
private readonly requests: RequestInfo[] = [];
Expand All @@ -23,7 +24,10 @@ export class ApiSniffer {
private doesRequestMatchesConfig(request: RequestInfo) {
const doesIncludeRuleMatches = !this.config.include
? true
: this.config.include.some((config) => doesUrlMatch(config.url, request.url));
: this.config.include.some((config) => {
log.info(`Matching include url ${request.url} with request ${config.url}`);
doesUrlMatch(config.url, request.url)
});
const doesExcludeRuleMatches = !this.config.exclude
? true
: !this.config.exclude.some((config) => doesUrlMatch(config.url, request.url));
Expand Down
1 change: 1 addition & 0 deletions src/interceptor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import stream from 'stream';
import { constructURLFromHttpRequest } from './utils/proxy';
import responseDecoder from './response-decoder';
import parseHeader from 'parse-headers';
import log from './logger';

function readBodyFromStream(writable: stream.Writable | undefined, callback: (value: any) => void) {
if (!writable) {
Expand Down
70 changes: 67 additions & 3 deletions src/plugin.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { BasePlugin } from 'appium/plugin';
import http from 'http';
import { Application } from 'express';
import { CliArg, ISessionCapability, MockConfig, RequestInfo, SniffConfig } from './types';
import { CliArg, ISessionCapability, MockConfig, RecordConfig, RequestInfo, ReplayConfig, SniffConfig } from './types';
import _ from 'lodash';
import { configureWifiProxy, isRealDevice } from './utils/adb';
import { cleanUpProxyServer, sanitizeMockConfig, setupProxyServer } from './utils/proxy';
Expand Down Expand Up @@ -40,6 +40,26 @@ export class AppiumInterceptorPlugin extends BasePlugin {
command: 'stopListening',
params: { optional: ['id'] },
},

'interceptor: startRecording': {
command: 'startRecording',
params: { optional: ['config'] },
},

'interceptor: stopRecording': {
command: 'stopRecording',
params: { optional: ['id'] },
},

'interceptor: startReplaying': {
command: 'startReplaying',
params: { required: ['replayConfig'] },
},

'interceptor: stopReplaying': {
command: 'stopReplaying',
params: { optional: ['id'] },
},
};

constructor(name: string, cliArgs: CliArg) {
Expand Down Expand Up @@ -77,6 +97,7 @@ export class AppiumInterceptorPlugin extends BasePlugin {
await configureWifiProxy(adb, deviceUDID, realDevice, proxy);
proxyCache.add(sessionId, proxy);
}
log.info("Creating session for appium interceptor");
return response;
}

Expand Down Expand Up @@ -162,10 +183,53 @@ export class AppiumInterceptorPlugin extends BasePlugin {
}

log.info(`Stopping listener with id: ${id}`);
return proxy.removeSniffer(id);
return proxy.removeSniffer(false, id);
}

async startRecording(next: any, driver: any, config: SniffConfig): Promise<string> {
const proxy = proxyCache.get(driver.sessionId);
if (!proxy) {
logger.error('Proxy is not running');
throw new Error('Proxy is not active for current session');
}

log.info(`Adding listener with config ${config}`);
return proxy?.addSniffer(config);
}

async stopRecording(next: any, driver: any, id: any): Promise<RecordConfig[]> {
const proxy = proxyCache.get(driver.sessionId);
if (!proxy) {
logger.error('Proxy is not running');
throw new Error('Proxy is not active for current session');
}

log.info(`Stopping recording with id: ${id}`);
return proxy.removeSniffer(true, id);
}

async startReplaying(next:any, driver:any, replayConfig: ReplayConfig) {
const proxy = proxyCache.get(driver.sessionId);
if (!proxy) {
logger.error('Proxy is not running');
throw new Error('Proxy is not active for current session');
}
log.info('Starting replay traffic');
proxy.startReplaying();
return proxy.getRecordingManager().replayTraffic(replayConfig);
}

async stopReplaying(next: any, driver:any, id:any) {
const proxy = proxyCache.get(driver.sessionId);
if (!proxy) {
logger.error('Proxy is not running');
throw new Error('Proxy is not active for current session');
}
log.info("Initiating stop replaying traffic");
proxy.getRecordingManager().stopReplay(id);
}

async execute(next: any, driver: any, script: any, args: any) {
return await this.executeMethod(next, driver, script, args);
}
}
}
69 changes: 48 additions & 21 deletions src/proxy.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { MockConfig, RequestInfo, SniffConfig } from './types';
import { MockConfig, RecordConfig, RequestInfo, SniffConfig } from './types';
import { Proxy as HttpProxy, IContext, IProxyOptions } from 'http-mitm-proxy';
import { v4 as uuid } from 'uuid';
import {
Expand All @@ -19,7 +19,8 @@ import { Mock } from './mock';
import { RequestInterceptor } from './interceptor';
import { ApiSniffer } from './api-sniffer';
import _ from 'lodash';
import logger from './logger';
import log from './logger';
import { RecordingManager } from './recording-manager';

export interface ProxyOptions {
deviceUDID: string;
Expand All @@ -31,20 +32,35 @@ export interface ProxyOptions {

export class Proxy {
private _started = false;
private _replayStarted = false;
private readonly mocks = new Map<string, Mock>();
private readonly sniffers = new Map<string, ApiSniffer>();

private readonly httpProxy: HttpProxy;
private readonly recordingManager: RecordingManager;

public isStarted(): boolean {
return this._started;
}

public isReplayStarted(): boolean {
return this._replayStarted;
}

public startReplaying(): void {
this._replayStarted = true;
}

constructor(private readonly options: ProxyOptions) {
this.httpProxy = new HttpProxy();
this.recordingManager = new RecordingManager(options);
addDefaultMocks(this);
}

public getRecordingManager(): RecordingManager {
return this.recordingManager;
}

public get port(): number {
return this.options.port;
}
Expand Down Expand Up @@ -81,7 +97,7 @@ export class Proxy {
this.httpProxy.onRequest(this.handleMockApiRequest.bind(this));

this.httpProxy.onError((context, error, errorType) => {
logger.error(`${errorType}: ${error}`);
log.error(`${errorType}: ${error}`);
});

await new Promise((resolve) => {
Expand Down Expand Up @@ -123,26 +139,37 @@ export class Proxy {
return id;
}

public removeSniffer(id?: string): RequestInfo[] {
const _sniffers = [...this.sniffers.values()];
if (id && !_.isNil(this.sniffers.get(id))) {
_sniffers.push(this.sniffers.get(id)!);
public removeSniffer(record: boolean, id?: string): RequestInfo[] {
const _sniffers = [...this.sniffers.values()];
if (id && !_.isNil(this.sniffers.get(id))) {
_sniffers.push(this.sniffers.get(id)!);
}
let apiRequests;
if (record) {
apiRequests = this.recordingManager.getCapturedTraffic(_sniffers);
}
else {
apiRequests = _sniffers.reduce((acc, sniffer) => {
acc.push(...sniffer.getRequests());
return acc;
}, [] as RequestInfo[]);
}
_sniffers.forEach((sniffer) => this.sniffers.delete(sniffer.getId()));
return apiRequests;
}
const apiRequests = _sniffers.reduce((acc, sniffer) => {
acc.push(...sniffer.getRequests());
return acc;
}, [] as RequestInfo[]);
_sniffers.forEach((sniffer) => this.sniffers.delete(sniffer.getId()));
return apiRequests;
}

private async handleMockApiRequest(ctx: IContext, next: () => void): Promise<void> {
const matchedMocks = await this.findMatchingMocks(ctx);
if (matchedMocks.length) {
const compiledMock = compileMockConfig(matchedMocks);
this.applyMockToRequest(ctx, compiledMock, next);
} else {
next();
if (this.isReplayStarted()) {
this.recordingManager.handleRecordingApiRequest(ctx, next);
} else if (!this.isReplayStarted()) {
const matchedMocks = await this.findMatchingMocks(ctx);
if (matchedMocks.length) {
const compiledMock = compileMockConfig(matchedMocks);
this.applyMockToRequest(ctx, compiledMock, next);
}
else {
next();
}
}
}

Expand Down Expand Up @@ -197,4 +224,4 @@ export class Proxy {
next();
}
}
}
}
27 changes: 27 additions & 0 deletions src/record.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import { RecordConfig } from './types';

export class Record {
private enabled = true;

constructor(private id: string, private config: RecordConfig) {}

getId() {
return this.id;
}

getConfig() {
return this.config;
}

isEnabled() {
return this.enabled;
}

setEnableStatus(enbaleStatus: boolean) {
this.enabled = enbaleStatus;
}

updateConfig(config: RecordConfig) {
this.config = config;
}
}
Loading

0 comments on commit ddd3638

Please sign in to comment.