apigeelint
Version:
Node module and tool to lint a bundle for an Apigee API Proxy or sharedflow.
526 lines (476 loc) • 15.2 kB
JavaScript
/*
Copyright 2019-2025 Google LLC
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
https://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
const Flow = require("./Flow.js"),
ConfigElement = require("./ConfigElement.js"),
RouteRule = require("./RouteRule.js"),
FaultRule = require("./FaultRule.js"),
xpath = require("xpath"),
lintUtil = require("./lintUtil.js"),
debug = require("debug")("apigeelint:Endpoint"),
fs = require("node:fs"),
getcb = lintUtil.curry(lintUtil.diagcb, debug),
HTTPProxyConnection = require("./HTTPProxyConnection.js"),
HTTPTargetConnection = require("./HTTPTargetConnection.js"),
bundleTypes = require("./BundleTypes.js");
class Endpoint extends ConfigElement {
constructor(element, bundle, fname, bundletype) {
debug(`> new Endpoint() ${fname}`);
super(element, bundle);
this.bundleType = bundletype ? bundletype : "apiproxy"; //default to apiproxy if bundletype not passed
this.fileName = fname;
this.preFlow = null;
this.postFlow = null;
this.postClientFlow = null;
this.eventFlow = null;
this.sharedFlow = null;
this.httpProxyConnection = null;
this.httpTargetConnection = null;
this.report = {
filePath: lintUtil.effectivePath(bundle, fname),
errorCount: 0,
warningCount: 0,
fixableErrorCount: 0,
fixableWarningCount: 0,
messages: [],
};
debug(`< new Endpoint() ${fname}`);
}
getName() {
if (!this.name) {
this.name = xpath.select("string(./@name)", this.element);
}
return this.name;
}
getReport() {
// sort messages by line, column
this.report.messages.sort((a, b) => {
if (a.line && !b.line) return 1;
if (b.line && !a.line) return -1;
if (a.line == b.line) {
if (a.column && !b.column) return 1;
if (b.column && !a.column) return -1;
if (!b.column && !a.column) return 1;
return a.column - b.column;
}
return a.line - b.line;
});
return this.report;
}
getFileName = () => this.fileName;
getLines(start, stop) {
//actually parse the source into lines if we haven't already and return the requested subset
if (!this.lines) {
this.lines = fs.readFileSync(this.getFileName()).toString().split("\n");
}
//check start and stop
if (!stop || stop > this.lines.length) {
stop = this.lines.length;
}
if (!start || start < 0) {
start = 0;
}
if (stop < 0) {
stop = 0;
}
if (start > stop) {
start = stop;
}
let result = "";
for (let i = start; i <= stop; i++) {
result += this.lines[i] + "\n";
}
return result;
}
getSource() {
if (!this.source) {
const start = this.element.lineNumber - 1,
stop = Number.MAX_SAFE_INTEGER;
if (this.element.nextSibling && this.element.nextSibling.lineNumber) {
this.source = this.element.nextSibling.lineNumber - 1;
} else {
this.source = this.getLines(start, stop);
}
}
return this.source;
}
getType = () => this.element.tagName;
getProxyName() {
if (!this.proxyName) {
this.proxyName =
this.fileName +
":" +
lintUtil.buildTagBreadCrumb(this.element) +
this.getName();
}
return this.proxyName;
}
#valueTable = {};
#getFlow(expr) {
debug(`> getFlow[${expr}]()`);
if (expr in this.#valueTable) {
debug(`< getFlow[${expr}]() cached`);
} else {
const n = xpath.select(expr, this.element);
this.#valueTable[expr] = n && n[0] && new Flow(n[0], this);
debug(`< getFlow[${expr}]()`);
}
return this.#valueTable[expr]; // maybe null or false
}
getPreFlow = () => this.#getFlow("./PreFlow");
getPostFlow = () => this.#getFlow("./PostFlow");
getPostClientFlow = () => this.#getFlow("./PostClientFlow");
getEventFlow = () => this.#getFlow("./EventFlow");
getSharedFlow = () => this.#getFlow("/SharedFlow");
getRouteRules() {
if (!this.routeRules) {
const ep = this;
ep.routeRules = xpath
.select("./RouteRule", this.element)
.map((elt) => new RouteRule(elt, ep));
}
return this.routeRules;
}
getHTTPProxyConnection() {
if (this.httpProxyConnection == null) {
const n = xpath.select("./HTTPProxyConnection", this.element);
this.httpProxyConnection =
n && n[0] && new HTTPProxyConnection(n[0], this);
}
return this.httpProxyConnection;
}
getHTTPTargetConnection() {
if (this.httpTargetConnection == null) {
const n = xpath.select("./HTTPTargetConnection", this.element);
this.httpTargetConnection =
n && n[0] && new HTTPTargetConnection(n[0], this);
}
return this.httpTargetConnection;
}
getFlows() {
debug("> getFlows()");
if (!this.flows) {
// Flows is always a child of toplevel element
const nodes = xpath.select("/*/Flows", this.element),
ep = this,
flows = [];
if (nodes) {
nodes.forEach(function (flowsElement) {
// get a child flow from here
xpath.select("Flow", flowsElement).forEach((elt) => {
if (elt.attributes && elt.attributes[0]) {
debug(
`Flow ${elt.attributes[0].name}='${elt.attributes[0].value}'`,
);
} else {
debug(`Flow`);
}
const flow = new Flow(elt, ep);
debug(`getFlows() flow= ${flow.element.tagName}`);
flows.push(flow);
});
});
}
this.flows = flows;
}
debug("< getFlows()");
return this.flows;
}
getAllFlows() {
debug("> getAllFlows()");
if (!this.allFlows) {
const allFlows = [];
if (this.bundleType === bundleTypes.BundleType.SHAREDFLOW) {
allFlows.push(this.getSharedFlow());
} else {
allFlows.push(this.getPreFlow());
allFlows.push(this.getPostFlow());
allFlows.push(this.getPostClientFlow());
allFlows.push(this.getEventFlow());
}
allFlows.push.apply(allFlows, this.getFlows());
this.allFlows = allFlows.filter((e) => !!e);
debug("< getAllFlows()");
} else {
debug("< getAllFlows() cached");
}
return this.allFlows;
}
getFaultRules() {
debug("> getFaultRules()");
if (!this.faultRules) {
const n = xpath.select("./FaultRules/FaultRule", this.element),
ep = this;
ep.faultRules = n.length ? n.map((fr) => new FaultRule(fr, ep)) : [];
}
debug("< getFaultRules()");
return this.faultRules;
}
getDefaultFaultRule() {
if (!this.defaultFaultRule) {
const n = xpath.select("./DefaultFaultRule", this.element);
this.defaultFaultRule = n.length ? new FaultRule(n[0], this) : null;
}
return this.defaultFaultRule;
}
getSteps() {
debug("> getSteps()");
if (!this.steps) {
const ep = this,
steps = [];
ep.getAllFlows().forEach((flow) => {
if (flow) {
const flowphases = flow.getSharedFlow()
? [flow.getSharedFlow()]
: [flow.getFlowRequest(), flow.getFlowResponse()];
flowphases.forEach((flowphase) => {
flowphase && steps.push(...flowphase.getSteps());
});
}
});
ep.getFaultRules().forEach(
(fr) => fr && fr.getSteps().forEach((step) => steps.push(step)),
);
if (ep.getDefaultFaultRule()) {
ep.getDefaultFaultRule()
.getSteps()
.forEach((step) => steps.push(step));
}
this.steps = steps;
debug(` getSteps() found ${steps.length} steps`);
}
debug(`< getSteps()`);
return this.steps;
}
onSteps(pluginFunction, callback) {
const endpoint = this,
flows = endpoint.getFlows(),
faultrules = endpoint.getFaultRules(),
sharedflow = endpoint.getSharedFlow();
debug(`> onSteps: endpoint name: '${endpoint.getName()}'`);
try {
if (sharedflow) {
sharedflow.onSteps(pluginFunction, getcb(`onSteps sharedflow `));
} else {
if (endpoint.getPreFlow()) {
endpoint
.getPreFlow()
.onSteps(pluginFunction, getcb("onSteps preflow"));
}
if (flows && flows.length > 0) {
flows.forEach((fl) =>
fl.onSteps(pluginFunction, getcb(`onSteps flow '${fl.getName()}'`)),
);
}
if (endpoint.getPostFlow()) {
endpoint
.getPostFlow()
.onSteps(pluginFunction, getcb("onSteps postflow"));
} else {
debug("onSteps: no postflow");
}
if (endpoint.getPostClientFlow()) {
endpoint
.getPostClientFlow()
.onSteps(pluginFunction, getcb("onSteps postClientFlow"));
} else {
debug("onSteps: no postClientFlow");
}
}
if (endpoint.getDefaultFaultRule()) {
debug("onSteps: defaultFaultRule");
endpoint
.getDefaultFaultRule()
.onSteps(pluginFunction, getcb("onSteps dfr"));
} else {
debug("onSteps: no defaultFaultRule");
}
if (faultrules && faultrules.length > 0) {
faultrules.forEach((fr) =>
fr.onSteps(pluginFunction, getcb("onSteps faultrules")),
);
} else {
debug("onSteps: no faultRules");
}
} catch (exc1) {
debug("exception: " + exc1);
debug(exc1.stack);
}
if (callback) callback(null, {});
debug(`< onSteps: endpoint name: '${endpoint.getName()}'`);
}
onConditions(pluginFunction, callback) {
const endpoint = this,
flows = endpoint.getFlows(),
faultrules = endpoint.getFaultRules(),
routerules = endpoint.getRouteRules(),
sharedflow = endpoint.getSharedFlow();
debug(`onConditions: endpoint name: ${endpoint.getName()}`);
try {
if (sharedflow) {
sharedflow.onConditions(
pluginFunction,
getcb(`onConditions sharedflow`),
);
} else {
if (endpoint.getPreFlow()) {
endpoint
.getPreFlow()
.onConditions(pluginFunction, getcb("onConditions preflow"));
} else {
debug("onConditions: no PreFlow");
}
if (flows && flows.length > 0) {
flows.forEach((fl) =>
fl.onConditions(
pluginFunction,
getcb(`onConditions flow '${fl.getName()}'`),
),
);
} else {
debug("onConditions: no Flows");
}
if (endpoint.getPostFlow()) {
endpoint
.getPostFlow()
.onConditions(pluginFunction, getcb("onConditions postflow"));
} else {
debug("onConditions: no PostFlow");
}
if (endpoint.getPostClientFlow()) {
endpoint
.getPostClientFlow()
.onConditions(pluginFunction, getcb("onConditions postClientFlow"));
} else {
debug("onConditions: no PostClientFlow");
}
if (endpoint.getDefaultFaultRule()) {
endpoint
.getDefaultFaultRule()
.getSteps()
.forEach((step) =>
step.onConditions(pluginFunction, getcb("onConditions dfr")),
);
} else {
debug("onConditions: no DefaultFaultRule");
}
if (faultrules && faultrules.length > 0) {
faultrules.forEach((fr) => {
fr.onConditions(
pluginFunction,
getcb(`faultrule '${fr.getName()}'`),
);
fr.getSteps().forEach((step) =>
step.onConditions(
pluginFunction,
getcb(`onConditions faultrule '${fr.getName()}'`),
),
);
});
} else {
debug("onConditions: no FaultRules");
}
if (routerules && routerules.length > 0) {
routerules.forEach((rr) =>
rr.onConditions(
pluginFunction,
getcb(`onConditions routerule '${rr.getName()}'`),
),
);
} else {
debug("onConditions: no RouteRules");
}
}
} catch (exc1) {
debug("exception: " + exc1);
debug(exc1.stack);
}
if (callback) callback(null, {});
}
addMessage(msg) {
debug(`> addMessage`);
//Severity should be one of the following: 0 = off, 1 = warning, 2 = error
if (msg.hasOwnProperty("plugin")) {
msg.ruleId = msg.plugin.ruleId;
if (!msg.severity) msg.severity = msg.plugin.severity;
msg.nodeType = msg.plugin.nodeType;
delete msg.plugin;
}
debug(`addMessage ${msg.ruleId}: ${msg.message}`);
if (!msg.hasOwnProperty("entity")) {
msg.entity = this;
}
if (
!msg.hasOwnProperty("source") &&
msg.entity.hasOwnProperty("getSource")
) {
msg.source = msg.entity.getSource();
}
if (
!msg.hasOwnProperty("line") &&
msg.entity.hasOwnProperty("getElement")
) {
msg.line = msg.entity.getElement().lineNumber;
}
if (
!msg.hasOwnProperty("column") &&
msg.entity.hasOwnProperty("getElement")
) {
msg.column = msg.entity.getElement().columnNumber;
}
delete msg.entity;
this.report.messages.push(msg);
switch (msg.severity) {
case 1:
this.report.warningCount++;
break;
case 2:
this.report.errorCount++;
break;
}
}
summarize() {
debug("> summarize");
const summary = {
name: this.getName(),
type: this.getType(),
};
if (this.bundleType === bundleTypes.BundleType.SHAREDFLOW) {
debug("summarize - is SharedFlow");
summary.flows = this.getAllFlows().map((flow) => flow.summarize());
// summary.flows = this.getSharedFlow().summarize();
} else {
debug("summarize - is Apiproxy");
summary.proxyName = this.getProxyName();
const faultRules = this.getFaultRules();
if (faultRules) {
summary.faultRules = [];
faultRules.forEach(function (fr) {
summary.faultRules.push(fr.summarize());
});
}
summary.defaultFaultRule =
(this.getDefaultFaultRule() &&
this.getDefaultFaultRule().summarize()) ||
{};
summary.preFlow =
(this.getPreFlow() && this.getPreFlow().summarize()) || {};
summary.flows = this.getFlows().map((flow) => flow.summarize());
summary.postFlow =
(this.getPostFlow() && this.getPostFlow().summarize()) || {};
summary.routeRules = this.getRouteRules().map((rr) => rr.summarize());
}
return summary;
}
}
//Public
module.exports = Endpoint;