insomnia/plugins/insomnia-plugin-response/index.js

283 lines
7.9 KiB
JavaScript
Raw Normal View History

const { JSONPath } = require('jsonpath-plus');
const iconv = require('iconv-lite');
2018-06-25 17:42:50 +00:00
const { query: queryXPath } = require('insomnia-xpath');
function isFilterableField(field) {
return field !== 'raw' && field !== 'url';
}
const defaultTriggerBehaviour = 'never';
2018-06-25 17:42:50 +00:00
module.exports.templateTags = [
{
name: 'response',
displayName: 'Response',
description: "reference values from other request's responses",
args: [
{
displayName: 'Attribute',
type: 'enum',
options: [
{
displayName: 'Body Attribute',
description: 'value of response body',
value: 'body',
2018-06-25 17:42:50 +00:00
},
{
displayName: 'Raw Body',
description: 'entire response body',
value: 'raw',
2018-06-25 17:42:50 +00:00
},
{
displayName: 'Header',
description: 'value of response header',
value: 'header',
},
{
displayName: 'Request URL',
description: 'Url of initiating request',
value: 'url',
},
],
2018-06-25 17:42:50 +00:00
},
{
displayName: 'Request',
type: 'model',
model: 'Request',
2018-06-25 17:42:50 +00:00
},
{
type: 'string',
encoding: 'base64',
hide: args => !isFilterableField(args[0].value),
2018-06-25 17:42:50 +00:00
displayName: args => {
switch (args[0].value) {
case 'body':
return 'Filter (JSONPath or XPath)';
case 'header':
return 'Header Name';
default:
return 'Filter';
}
},
},
{
displayName: 'Trigger Behavior',
help: 'Configure when to resend the dependent request',
type: 'enum',
defaultValue: defaultTriggerBehaviour,
options: [
{
2019-04-27 04:45:11 +00:00
displayName: 'Never',
description: 'never resend request',
value: 'never',
},
{
2019-04-27 04:45:11 +00:00
displayName: 'No History',
description: 'resend when no responses present',
value: 'no-history',
},
{
displayName: 'When Expired',
description: 'resend when existing response has expired',
value: 'when-expired',
},
{
2019-04-27 04:45:11 +00:00
displayName: 'Always',
description: 'resend request when needed',
value: 'always',
},
],
},
{
displayName: 'Max age (seconds)',
help: 'The maximum age of a response to use before it expires',
type: 'number',
hide: args => {
const triggerBehavior = (args[3] && args[3].value) || defaultTriggerBehaviour;
return triggerBehavior !== 'when-expired';
},
defaultValue: 60,
},
2018-06-25 17:42:50 +00:00
],
async run(context, field, id, filter, resendBehavior, maxAgeSeconds) {
2018-06-25 17:42:50 +00:00
filter = filter || '';
resendBehavior = (resendBehavior || defaultTriggerBehaviour).toLowerCase();
if (!['body', 'header', 'raw', 'url'].includes(field)) {
2018-06-25 17:42:50 +00:00
throw new Error(`Invalid response field ${field}`);
}
2018-06-25 17:42:50 +00:00
if (!id) {
throw new Error('No request specified');
}
2018-06-25 17:42:50 +00:00
const request = await context.util.models.request.getById(id);
if (!request) {
throw new Error(`Could not find request ${id}`);
}
const environmentId = context.context.getEnvironmentId();
let response = await context.util.models.response.getLatestForRequestId(id, environmentId);
let shouldResend = false;
switch (resendBehavior) {
case 'no-history':
shouldResend = !response;
break;
case 'when-expired':
if (!response) {
shouldResend = true;
} else {
const ageSeconds = (Date.now() - response.created) / 1000;
shouldResend = ageSeconds > maxAgeSeconds;
}
break;
case 'always':
shouldResend = true;
break;
case 'never':
default:
shouldResend = false;
break;
}
// Make sure we only send the request once per render so we don't have infinite recursion
const requestChain = context.context.getExtraInfo('requestChain') || [];
if (requestChain.some(id => id === request._id)) {
console.log('[response tag] Preventing recursive render');
shouldResend = false;
}
if (shouldResend && context.renderPurpose === 'send') {
console.log('[response tag] Resending dependency');
requestChain.push(request._id)
response = await context.network.sendRequest(request, [
{ name: 'requestChain', value: requestChain }
]);
}
2018-06-25 17:42:50 +00:00
if (!response) {
console.log('[response tag] No response found');
2018-06-25 17:42:50 +00:00
throw new Error('No responses for request');
}
if (response.error) {
console.log('[response tag] Response error ' + response.error);
throw new Error('Failed to send dependent request ' + response.error);
}
2018-06-25 17:42:50 +00:00
if (!response.statusCode) {
console.log('[response tag] Invalid status code ' + response.statusCode);
2018-06-25 17:42:50 +00:00
throw new Error('No successful responses for request');
}
2018-06-25 17:42:50 +00:00
if (isFilterableField(field) && !filter) {
2018-06-25 17:42:50 +00:00
throw new Error(`No ${field} filter specified`);
}
2018-06-25 17:42:50 +00:00
const sanitizedFilter = filter.trim();
const bodyBuffer = context.util.models.response.getBodyBuffer(response, '');
const match = response.contentType && response.contentType.match(/charset=([\w-]+)/);
const charset = match && match.length >= 2 ? match[1] : 'utf-8';
switch (field) {
case 'header':
return matchHeader(response.headers, sanitizedFilter);
case 'url':
return response.url;
case 'raw':
// Sometimes iconv conversion fails so fallback to regular buffer
try {
return iconv.decode(bodyBuffer, charset);
} catch (err) {
console.warn('[response] Failed to decode body', err);
return bodyBuffer.toString();
}
case 'body':
// Sometimes iconv conversion fails so fallback to regular buffer
let body;
try {
body = iconv.decode(bodyBuffer, charset);
} catch (err) {
console.warn('[response] Failed to decode body', err);
2022-03-17 13:19:08 +00:00
body = bodyBuffer.toString();
}
if (sanitizedFilter.indexOf('$') === 0) {
return matchJSONPath(body, sanitizedFilter);
} else {
return matchXPath(body, sanitizedFilter);
}
2018-06-25 17:42:50 +00:00
default:
throw new Error(`Unknown field ${field}`);
}
},
},
2018-06-25 17:42:50 +00:00
];
2018-06-25 17:42:50 +00:00
function matchJSONPath(bodyStr, query) {
let body;
let results;
try {
body = JSON.parse(bodyStr);
} catch (err) {
throw new Error(`Invalid JSON: ${err.message}`);
}
try {
results = JSONPath({json: body, path: query});
} catch (err) {
throw new Error(`Invalid JSONPath query: ${query}`);
}
if (results.length === 0) {
throw new Error(`Returned no results: ${query}`);
}
if (results.length > 1 ) {
return JSON.stringify(results);
}
if (typeof results[0] !== 'string') {
return JSON.stringify(results[0]);
} else {
return results[0];
}
}
2018-06-25 17:42:50 +00:00
function matchXPath(bodyStr, query) {
const results = queryXPath(bodyStr, query);
if (results.length === 0) {
throw new Error(`Returned no results: ${query}`);
} else if (results.length > 1) {
throw new Error(`Returned more than one result: ${query}`);
}
2017-09-22 13:48:47 +00:00
return results[0].inner;
}
2018-06-25 17:42:50 +00:00
function matchHeader(headers, name) {
if (!headers.length) {
throw new Error('No headers available');
}
2018-06-25 17:42:50 +00:00
const header = headers.find(h => h.name.toLowerCase() === name.toLowerCase());
if (!header) {
const names = headers.map(c => `"${c.name}"`).join(',\n\t');
2018-10-17 16:42:33 +00:00
throw new Error(`No header with name "${name}".\nChoices are [\n\t${names}\n]`);
}
return header.value;
}