Не удалось подключиться к базе данных Heroku с помощью sequenceize-typescript

Мой Express API, использующий sequelize-typescript, не может подключиться к базе данных Heroku PostgreSQL.

Приложение Heroku было инициализировано выполнением следующих команд:

heroku create <my-app-name>
heroku addons:create heroku-postgresql:hobby-dev
git push heroku master

Это приложение работает без проблем в моей локальной среде.

Я могу подтвердить, что переменная среды DATABASE_URL верна на Heroku.

Мои package.json скрипты тоже кажутся правильными:

"scripts": {
  "build": "tsc",
  "dev": "node_modules/.bin/nodemon",
  "postinstall": "npm run build",
  "start": "node dist/server/server.js"
}

Что я пытался решить проблему от самого важного к наименее:

  1. Настройка NPM_CONFIG_PRODUCTION=false путем выполнения heroku config:set. Я сделал это после прочтения этого SO thread, который указывает на этот документ Heroku. Я думаю, что идея заключается в том, что это говорит Heroku пропустить шаг, на котором он обрезает devDependencies, чтобы Heroku мог скомпилировать файлы TypeScript в Javascript.

  2. Настройка dialectOptions: { ssl: true } в моем соединении Sequelize, когда process.env.NODE_ENV === 'production'. Я сделал это после прочтения нескольких документов SO по этой теме.

  3. Предоставление соединения Sequelize с полями имени пользователя и пароля. Я сделал это, выполнив:

// In the terminal
heroku config:set DB_USER=<database user from Heroku database credentials>
heroku config:set DB_PASSWORD=<database password from Heroku database credentials>

// In Sequelize connection
export const connection = new Sequelize({
    dialect: 'postgres',
    protocol: 'postgres',
    username: process.env.DB_USER || '',
    password: process.env.DB_PASSWORD || '',
    database: process.env.DATABASE_URL || 'fitnius-db',
    logging: false,
    dialectOptions: { ssl },
    models: [__dirname + '/models', __dirname + '/models' + '/joins']
})
  1. Добавление @types/pg в качестве зависимости

  2. Удаление моего приложения Heroku и создание нового

  3. Удаление "node_modules" из "exclude": [] в tsconfig.json

  4. Удаление всех переменных окружения из nodemon.json

  5. Создание Procfile и добавление web: node dist/server/server.js внутрь него

  6. Перемещение, @types/express и @types/sequelize между зависимостями и devDependencies. Я был в отчаянии и прочитал этот документ от Sequelize.

    • Also, installing @types/node and @types/validator as dependencies
  7. Установка скрипта сборки на "build": "tsc -p ." (Не совсем уверен, что делает флаг -p... но я это сделал)

  8. Установка относительного пути к скомпилированному файлу сервера в моем стартовом скрипте:

    • "start": "node ./dist/server/server.js"

Вопрос: как заставить Sequelize установить соединение с базой данных Heroku PostgreSQL и запустить мое приложение?

Ниже приведен соответствующий код, относящийся к проблеме.


Журнал ошибок от heroku logs --tail

{ SequelizeConnectionRefusedError: connect ECONNREFUSED 127.0.0.1:5432
at connection.connect.err (/app/node_modules/sequelize/lib/dialects/postgres/connection-manager.js:170:24)
at Connection.connectingErrorHandler (/app/node_modules/pg/lib/client.js:191:14)
at Connection.emit (events.js:198:13)
at Socket.reportStreamError (/app/node_modules/pg/lib/connection.js:72:10)
at Socket.emit (events.js:198:13)
at emitErrorNT (internal/streams/destroy.js:91:8)
at emitErrorAndCloseNT (internal/streams/destroy.js:59:3)
at process._tickCallback (internal/process/next_tick.js:63:19)
name: 'SequelizeConnectionRefusedError',
parent:
{ Error: connect ECONNREFUSED 127.0.0.1:5432
at TCPConnectWrap.afterConnect [as oncomplete] (net.js:1106:14)
errno: 'ECONNREFUSED',
code: 'ECONNREFUSED',
syscall: 'connect',
address: '127.0.0.1',
port: 5432 },

at=error code=H10 desc="App crashed" method=GET path="/" 
host=fitnius-web-api.herokuapp.com 
request_id=c8c76ce3-8201-4733-99fb-bee4121b09d1 fwd="100.1.44.6" 
dyno= connect= service= status=503 bytes= protocol=https

Последовательное подключение

import { Sequelize } from 'sequelize-typescript'

const ssl = process.env.NODE_ENV === 'production'

export const connection = new Sequelize({
    dialect: 'postgres',
    protocol: 'postgres',
    database: process.env.DATABASE_URL || 'fitnius-db',
    logging: false,
    dialectOptions: { ssl },
    models: [__dirname + '/models', __dirname + '/models' + '/joins']
})

Экспресс-сервер

import app from './app'
import { connection } from '../database/connection'

const PORT = process.env.PORT || 3000
connection
    .sync()
    .then(() => {
        app.listen(PORT, () => console.log(`Server is running on port ${PORT}`))
    })
    .catch(err => console.log(err))

tsconfig.json

{
  "compilerOptions": {
    "target": "es6",
    "module": "commonjs",
    "outDir": "dist",
    "strict": true,
    "declaration": true,
    "esModuleInterop": true,
    "emitDecoratorMetadata": true,
    "experimentalDecorators": true,
    "sourceMap": true,
    "lib": ["es6", "dom"]
  },
  "include": ["src/**/*"],
  "exclude": ["node_modules", ".vscode"]
}

package.json

{
  "name": "fitnius-server",
  "version": "1.0.0",
  "description": "",
  "main": "dist/server/server.js",
  "scripts": {
    "build": "tsc",
    "dev": "node_modules/.bin/nodemon",
    "postinstall": "npm run build",
    "start": "node dist/server/server.js"
  },
  "keywords": [],
  "author": "",
  "license": "ISC",
  "dependencies": {
    "@types/express": "^4.17.0",
    "@types/graphql": "^14.2.3",
    "@types/pg": "^7.4.14",
    "@types/sequelize": "^4.28.4",
    "express": "^4.17.1",
    "express-graphql": "^0.9.0",
    "graphql": "^14.4.2",
    "pg": "^7.12.0",
    "reflect-metadata": "^0.1.13",
    "sequelize": "^5.12.3",
    "sequelize-typescript": "^1.0.0-beta.3"
  },
  "devDependencies": {
    "nodemon": "^1.19.1",
    "ts-node": "^8.3.0",
    "typescript": "^3.5.3"
  }
}

Дерево каталогов

root
  dist (compiled by tsc)
  node_modules
  src
    database
      models
      connection.ts (Sequelize connection)
    server
      app.ts
      server.ts (Express server)
  nodemon.json
  tsconfig.json
  package.json

person khan    schedule 05.08.2019    source источник


Ответы (1)


Покопавшись в пакете npm sequelize-typescript, я обнаружил, что параметр database не относится к URL-адресу всей базы данных.

Мне удалось решить проблему, проанализировав следующий URL-адрес базы данных Heroku. В качестве решения я использовал этот парсер URL-адресов базы данных.

interface ParsedDatabaseOptions {
    protocol: string | undefined
    host: string | undefined
    username: string | undefined
    password: string | undefined
    database: string | undefined
}

После преобразования парсера в грубую версию TypeScript, которую можно найти по адресу здесь, затем я создал условное выражение, проверив NODE_ENV === 'production', и установил следующие параметры базы данных:

import { Sequelize } from 'sequelize-typescript'
// parseDatabaseUrl is currently a wonky piece of code
import parseDatabaseUrl from '../utils/parse-database-url'

const optionsProduction = parseDatabaseUrl(process.env.DATABASE_URL)
const optionsDevelopment = { database: <your local database> }

// Create conditional Sequelize database options here
const sequelizeOptions = process.env.NODE_ENV === 'production'
    ? optionsProduction
    : optionsDevelopment

const connection = new Sequelize({
    dialect: 'postgres',
    logging: false,
    ...sequelizeOptions, // Spread options here
    models: ...
})
person khan    schedule 05.08.2019