UNPKG

apigeelint

Version:

Node module and tool to lint a bundle for an Apigee API Proxy or sharedflow.

492 lines (443 loc) 14 kB
/* 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"), util = require("node:util"), fs = require("node:fs"), path = require("node:path"), 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; 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 var result = ""; 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 > this.lines.length) { stop = this.lines.length; } if (stop < 0) { stop = 0; } if (start > stop) { start = stop; } for (var 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; } #flowTable = {}; #getFlow(expr) { debug(`> getFlow[${expr}]()`); if (this.#flowTable[expr] == null) { const n = xpath.select(expr, this.element); this.#flowTable[expr] = n && n[0] && new Flow(n[0], this); } debug(`< getFlow[${expr}]()`); return this.#flowTable[expr]; } getPreFlow = () => this.#getFlow("./PreFlow"); getPostFlow = () => this.#getFlow("./PostFlow"); getPostClientFlow = () => this.#getFlow("./PostClientFlow"); getEventFlow = () => this.#getFlow("./EventFlow"); getSharedFlow = () => this.#getFlow("/SharedFlow"); getRouteRules() { if (!this.routeRules) { let 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 let 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`); } let f = new Flow(elt, ep); debug(`getFlows() flow= ${f.element.tagName}`); flows.push(f); }); }); } this.flows = flows; } debug("< getFlows()"); return this.flows; } getAllFlows() { debug("> getAllFlows()"); if (!this.allFlows) { let 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()"); 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) { let 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(); debug(`> onSteps: endpoint name: '${endpoint.getName()}'`); try { 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.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(); debug(`onConditions: endpoint name: ${endpoint.getName()}`); try { 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.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"); let summary = { name: this.getName(), type: this.getType(), }; if (this.bundleType === bundleTypes.BundleType.SHAREDFLOW) { debug("summarize - is SharedFlow"); summary.flows = this.getSharedFlow().summarize(); } else { debug("summarize - is Apiproxy"); summary.proxyName = this.getProxyName(); let 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;