Solucionado (ver solução)
Solucionado
(ver solução)
2
respostas

MySQL no Heroku

Boa noite.

Durante todos os cursos de DJango tenho utilizado um banco de dados MySQL local.

Ao acessar a API no HEROKU após o deploy estou recebendo o seguinte erro:

(2002, "Can't connect to local MySQL server through socket '/var/run/mysqld/mysqld.sock' (2)")

A configuração do meu settings.xml está como abaixo:

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'django-rest-api',
        'USER': 'admin',
        'PASSWORD': 'admin',
        'HOST': 'localhost',
        'PORT': '3306',
    }
}

Eu vi que na documentação do Heroku é solicitada a instalação local do PostgreSQL, mas quero utilizar o meu MySQL já configurado. Alguma dica de como fazer? Estou procurando na documentação.

Abaixo o erro completo que recebo:


Request Method:    GET
Request URL:    https://rf-django-rest-api-clientes.herokuapp.com/cliente/clientes/
Django Version:    4.1.2
Exception Type:    OperationalError
Exception Value:    
(2002, "Can't connect to local MySQL server through socket '/var/run/mysqld/mysqld.sock' (2)")
Exception Location:    /app/.heroku/python/lib/python3.10/site-packages/MySQLdb/connections.py, line 185, in __init__
Raised during:    cliente.views.ClientesViewSet
Python Executable:    /app/.heroku/python/bin/python
Python Version:    3.10.7
Python Path:    
['/app/setup/../apps',
 '/app/.heroku/python/bin',
 '/app',
 '/app/.heroku/python/lib/python310.zip',
 '/app/.heroku/python/lib/python3.10',
 '/app/.heroku/python/lib/python3.10/lib-dynload',
 '/app/.heroku/python/lib/python3.10/site-packages']
Server time:    Mon, 10 Oct 2022 22:46:48 -0300

Outra coisa que não entendo é como pode o Heroku se conectar no servidor de banco de dados da nossa máquina local. Não faz sentido para mim isso.

O óbvio para mim seria ele provisionar uma instância de banco lá no servidor remoto e ali manter os dados.

At.te.

2 respostas

Me respondendo...

Encontrei o seguinte na documentação do Heroku:

Heroku recommends running Postgres locally to ensure parity between environments.

Agora sim, faz sentido. O banco realmente é hospedado lá. Óbvio. O que eles solicitam é a instalação em desenvolvimento para ter paridade de scripts.

Vou tentar ajustar esses scripts na mão para não ter que realizar a instalação.

Agora preciso ver como fazer meu settings.xml apontar para o banco disponibilizado pelo Heroku. A lib django-heroku sugerida no curso não é atualizada há quatro anos.

At.te

solução!

Resolvido o meu problema.

Segue abaixo o meu settings.xml:

import os
import sys

from pathlib import Path

from dotenv import load_dotenv
import dj_database_url

# Carrega as variáveis de ambiente de um arquivo .env disponibilizado na raiz do projeto...
load_dotenv()

# Se existe a variável de ambiente 'DYNO' setada então a aplicação está rodando dentro do Heroku...
IS_HEROKU = "DYNO" in os.environ

# Build paths inside the project like this: BASE_DIR / 'subdir'.
BASE_DIR = Path(__file__).resolve().parent.parent

# Chave de segurançao do Django setado no arquivo .env...
SECRET_KEY = str(os.getenv('SECRET_KEY'))

# SECURITY WARNING: don't run with debug turned on in production!
DEBUG = True

# Se está rodando dentro do Heroku, então permite acesso de qualquer lugar. (Apenas para testes de deploy)
if IS_HEROKU:
    ALLOWED_HOSTS = ["*"]
else:
    ALLOWED_HOSTS = []


# Application definition

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'rest_framework',
    'django_filters',
    'escola',
    'cliente'
]

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    "whitenoise.middleware.WhiteNoiseMiddleware",  # Permite servir arquivos estáticos...
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

ROOT_URLCONF = 'setup.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

WSGI_APPLICATION = 'setup.wsgi.application'


# Database
# https://docs.djangoproject.com/en/4.1/ref/settings/#databases

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'OPTIONS': {
            'read_default_file': os.path.join(BASE_DIR, 'my.cnf')
        }
    }
}

# Se a variável de ambiente DATABASE_URL está setada então substitui a base de dados padrão...
if "DATABASE_URL" in os.environ:
    # Configure Django for DATABASE_URL environment variable.
    DATABASES["default"] = dj_database_url.config(
        conn_max_age=600, ssl_require=True)


# Password validation
# https://docs.djangoproject.com/en/4.1/ref/settings/#auth-password-validators

AUTH_PASSWORD_VALIDATORS = [
    {
        'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',
    },
]


# Internationalization
# https://docs.djangoproject.com/en/4.1/topics/i18n/

LANGUAGE_CODE = 'pt-br'

TIME_ZONE = 'America/Sao_Paulo'

USE_I18N = True

USE_TZ = True


# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/4.1/howto/static-files/

STATIC_URL = 'static/'
STATIC_ROOT = os.path.join(BASE_DIR, 'static')


# Default primary key field type
# https://docs.djangoproject.com/en/4.1/ref/settings/#default-auto-field

DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'

PROJECT_ROOT = os.path.dirname(__file__)
sys.path.insert(0, os.path.join(PROJECT_ROOT, '../apps'))

REST_FRAMEWORK = {
    'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination',
    'PAGE_SIZE': 10
}

Também foi necessário incluir um addon PostgreSQL na aplicação através do comando:

heroku addons:create heroku-postgresql:hobby-dev

Cuidado que esse addon será desativado no mês de Novembro de 2022.

At.te.