Skip to content

Commit

Permalink
fix: handle DNS error codes as ErrnoException
Browse files Browse the repository at this point in the history
DNS error codes are not included in the OS error constants
(os.constants) / https://nodejs.org/docs/latest-v18.x/api/os.html#error-constants

This commit adds DNS error codes to the list of handled errno errors.

Fixes #5217
  • Loading branch information
stefreak committed Oct 31, 2023
1 parent d460f3f commit a60a8b1
Show file tree
Hide file tree
Showing 3 changed files with 48 additions and 14 deletions.
4 changes: 2 additions & 2 deletions core/src/analytics/analytics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import { Profile } from "../util/profiling"
import { ModuleConfig } from "../config/module"
import { UserResult } from "@garden-io/platform-api-types"
import { uuidv4 } from "../util/random"
import { GardenError, NodeJSErrnoErrorCodes, StackTraceMetadata } from "../exceptions"
import { GardenError, NodeJSErrnoException, StackTraceMetadata } from "../exceptions"
import { ActionConfigMap } from "../actions/types"
import { actionKinds } from "../actions/types"
import { getResultErrorProperties } from "./helpers"
Expand Down Expand Up @@ -163,7 +163,7 @@ export type AnalyticsGardenErrorDetail = {
/**
* If this error was caused by an underlying NodeJSErrnoException, this will be the code.
*/
code?: NodeJSErrnoErrorCodes
code?: NodeJSErrnoException["code"]

stackTrace?: StackTraceMetadata
}
Expand Down
46 changes: 37 additions & 9 deletions core/src/exceptions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,19 +14,47 @@ import stripAnsi from "strip-ansi"
import { Cycle } from "./graph/common"
import indentString from "indent-string"
import { constants } from "os"
import dns from "node:dns"

// Unfortunately, NodeJS does not provide a list of all error codes, so we have to maintain this list manually.
const dnsErrorCodes = [
dns.NODATA,
dns.FORMERR,
dns.SERVFAIL,
dns.NOTFOUND,
dns.NOTIMP,
dns.REFUSED,
dns.BADQUERY,
dns.BADNAME,
dns.BADFAMILY,
dns.BADRESP,
dns.CONNREFUSED,
dns.TIMEOUT,
dns.EOF,
dns.FILE,
dns.NOMEM,
dns.DESTRUCTION,
dns.BADSTR,
dns.BADFLAGS,
dns.NONAME,
dns.BADHINTS,
dns.NOTINITIALIZED,
dns.LOADIPHLPAPI,
dns.ADDRGETNETWORKPARAMS,
dns.CANCELLED,
]

// See https://nodejs.org/api/os.html#error-constants
type NodeJSErrnoErrors = typeof constants.errno
export type NodeJSErrnoErrorCodes = keyof NodeJSErrnoErrors
const errnoErrors = Object.keys(constants.errno)

const errnoErrorCodeSet = new Set(Object.keys(constants.errno))
const errnoErrorCodeSet = new Set([errnoErrors, dnsErrorCodes].flat())

/**
* NodeJS native errors with a code property.
*/
export type NodeJSErrnoException = NodeJS.ErrnoException & {
code: NodeJSErrnoErrorCodes
errno: number
// Unfortunately we can't make this type more concrete, as DNS error codes are not known at typescript compile time.
code: string
}

export type EAddrInUseException = NodeJSErrnoException & {
Expand All @@ -37,7 +65,7 @@ export type EAddrInUseException = NodeJSErrnoException & {
}

export function isErrnoException(err: any): err is NodeJSErrnoException {
return typeof err.code === "string" && typeof err.errno === "number" && errnoErrorCodeSet.has(err.code)
return typeof err.code === "string" && errnoErrorCodeSet.has(err.code)
}

export function isEAddrInUseException(err: any): err is EAddrInUseException {
Expand All @@ -64,7 +92,7 @@ export interface GardenErrorParams {
*/
readonly taskType?: string

readonly code?: NodeJSErrnoErrorCodes
readonly code?: NodeJSErrnoException["code"]
}

export abstract class GardenError extends Error {
Expand All @@ -81,7 +109,7 @@ export abstract class GardenError extends Error {
/**
* If there was an underlying NodeJSErrnoException, the error code
*/
public code?: NodeJSErrnoErrorCodes
public code?: NodeJSErrnoException["code"]

public wrappedErrors?: GardenError[]

Expand Down Expand Up @@ -354,7 +382,7 @@ export class InternalError extends GardenError {
static wrapError(error: Error | string | any, prefix?: string): InternalError {
let message: string | undefined
let stack: string | undefined
let code: NodeJSErrnoErrorCodes | undefined
let code: NodeJSErrnoException["code"] | undefined

if (isErrnoException(error)) {
message = error.message
Expand Down
12 changes: 9 additions & 3 deletions core/src/plugins/kubernetes/retry.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,9 @@ import { dedent, deline } from "../../util/string"
import { LogLevel } from "../../logger/logger"
import { KubernetesError } from "./api"
import requestErrors = require("request-promise/errors")
import { InternalError, NodeJSErrnoErrorCodes, isErrnoException } from "../../exceptions"
import { InternalError, NodeJSErrnoException, isErrnoException } from "../../exceptions"
import { ErrorEvent } from "ws"
import dns from "node:dns"

/**
* The flag {@code forceRetry} can be used to avoid {@link shouldRetry} helper call in case if the error code
Expand Down Expand Up @@ -93,7 +94,7 @@ export function toKubernetesError(err: unknown, context: string): KubernetesErro
let response: KubernetesClientHttpError["response"] | undefined
let body: any | undefined
let responseStatusCode: number | undefined
let code: NodeJSErrnoErrorCodes | undefined
let code: NodeJSErrnoException["code"] | undefined

if (err instanceof KubernetesClientHttpError) {
errorType = "HttpError"
Expand Down Expand Up @@ -159,7 +160,7 @@ export function toKubernetesError(err: unknown, context: string): KubernetesErro
export function shouldRetry(error: unknown, context: string): boolean {
const err = toKubernetesError(error, context)

if (err.code === "ECONNRESET") {
if (err.code && errorCodesForRetry.includes(err.code)) {
return true
}

Expand All @@ -185,6 +186,11 @@ export const statusCodesForRetry: number[] = [
524, // A Timeout Occurred
]

const errorCodesForRetry: NodeJSErrnoException["code"][] = [
"ECONNRESET",
dns.NOTFOUND,
]

const errorMessageRegexesForRetry = [
/ETIMEDOUT/,
/ENOTFOUND/,
Expand Down

0 comments on commit a60a8b1

Please sign in to comment.