Articles

Solving the TS2769 Error in Upstash

Planted September 10, 2023

Upstash is a serverless data platform that supports Redis and Kafka. It is a great tool for building serverless applications. However, when I was trying to use it in my project, I encountered a TypeScript error TS2769: No overload matches this call.. This article will show you how to solve this error.

Setting Sail with Upstash

Upstash provides a Node.js package @upstash/redis that allows you to connect to Upstash Redis instances, along with some example code to get you started:

import { Redis } from '@upstash/redis'

const redis = new Redis({
  url: 'https://eu1-example.upstash.io',
  token: '********',
})
   
const data = await redis.set('foo', 'bar');

It is not recommended to hardcode the URL and token in the code, so it’s butter put them in .env.local instead:

UPSTASH_REDIS_REST_URL=https://eu1-example.upstash.io
UPSTASH_REDIS_REST_TOKEN=********

And then use process.env to access them:

import { Redis } from '@upstash/redis'

const redis = new Redis({
  url: process.env.UPSTASH_REDIS_REST_URL,
  token: process.env.UPSTASH_REDIS_REST_TOKEN,
})

const data = await redis.set('foo', 'bar');

Unexpected Error TS2769

After setting up the Upstash Redis client, I noticed that my IDE (WebStorm) is showing an error:

TS2769: No overload matches this call.   
Overload 1 of 2, '(config: RedisConfigNodejs): Redis', gave the following error.     
Type 'string | undefined' is not assignable to type 'string'.       
Type 'undefined' is not assignable to type 'string'.   

Overload 2 of 2, '(requesters: Requester): Redis', gave the following error.     
Argument of type '{ url: string | undefined; token: string; }' is not assignable to parameter of type 'Requester'.       
Object literal may only specify known properties, and 'url' does not exist in type 'Requester'. 

nodejs.d.ts(12, 5): The expected type comes from property 'url' which is declared here on type 'RedisConfigNodejs'

The error message indicated that the types string | undefined is not assignable to type string. This is because the environment variable returned by process.env is of type string | undefined, and the RedisConfigNodejs type only accepts string as the url property.

The Solution

The solution to this problem is simple:

const redis = Redis.fromEnv();

The fromEnv method will read the UPSTASH_REDIS_REST_URL and UPSTASH_REDIS_REST_TOKEN environment variables and return a Redis instance.

It does the validation for you, so you don’t have to worry about the types:

// source: upstash-redis/platforms/nodejs.ts
/**
   * Create a new Upstash Redis instance from environment variables.
   *
   * Use this to automatically load connection secrets from your environment
   * variables. For instance when using the Vercel integration.
   *
   * This tries to load `UPSTASH_REDIS_REST_URL` and `UPSTASH_REDIS_REST_TOKEN` from
   * your environment using `process.env`.
   */
  static fromEnv(config?: Omit<RedisConfigNodejs, "url" | "token">): Redis {
    // @ts-ignore process will be defined in node
    if (typeof process?.env === "undefined") {
      throw new Error(
        'Unable to get environment variables, `process.env` is undefined. If you are deploying to cloudflare, please import from "@upstash/redis/cloudflare" instead',
      );
    }
    // @ts-ignore process will be defined in node
    const url = process?.env["UPSTASH_REDIS_REST_URL"];
    if (!url) {
      throw new Error(
        "Unable to find environment variable: `UPSTASH_REDIS_REST_URL`",
      );
    }
    // @ts-ignore process will be defined in node
    const token = process?.env["UPSTASH_REDIS_REST_TOKEN"];
    if (!token) {
      throw new Error(
        "Unable to find environment variable: `UPSTASH_REDIS_REST_TOKEN`",
      );
    }
    return new Redis({ ...config, url, token });
  }
}