initial
This commit is contained in:
commit
bec3d67171
41
.deploy/deploy-dev.yaml
Normal file
41
.deploy/deploy-dev.yaml
Normal file
@ -0,0 +1,41 @@
|
||||
version: "3.4"
|
||||
|
||||
|
||||
services:
|
||||
|
||||
nginx:
|
||||
image: mathwave/sprint-repo:battleship-nginx
|
||||
networks:
|
||||
- net
|
||||
ports:
|
||||
- "1236:80"
|
||||
deploy:
|
||||
mode: replicated
|
||||
restart_policy:
|
||||
condition: any
|
||||
placement:
|
||||
constraints: [node.role == manager]
|
||||
update_config:
|
||||
parallelism: 1
|
||||
order: start-first
|
||||
|
||||
backend:
|
||||
image: mathwave/sprint-repo:battleship-back
|
||||
networks:
|
||||
- net
|
||||
environment:
|
||||
DB_HOST: "pg.develop.sprinthub.ru"
|
||||
DB_PASSWORD: $DB_PASSWORD_DEV
|
||||
DEBUG: "true"
|
||||
command: ./manage.py runserver 0.0.0.0:8000
|
||||
deploy:
|
||||
mode: replicated
|
||||
restart_policy:
|
||||
condition: any
|
||||
update_config:
|
||||
parallelism: 1
|
||||
order: start-first
|
||||
|
||||
networks:
|
||||
net:
|
||||
driver: overlay
|
41
.deploy/deploy-prod.yaml
Normal file
41
.deploy/deploy-prod.yaml
Normal file
@ -0,0 +1,41 @@
|
||||
version: "3.4"
|
||||
|
||||
|
||||
services:
|
||||
|
||||
nginx:
|
||||
image: mathwave/sprint-repo:battleship-nginx
|
||||
networks:
|
||||
- net
|
||||
ports:
|
||||
- "1236:80"
|
||||
deploy:
|
||||
mode: replicated
|
||||
restart_policy:
|
||||
condition: any
|
||||
placement:
|
||||
constraints: [node.role == manager]
|
||||
update_config:
|
||||
parallelism: 1
|
||||
order: start-first
|
||||
|
||||
backend:
|
||||
image: mathwave/sprint-repo:battleship-back
|
||||
networks:
|
||||
- net
|
||||
environment:
|
||||
DB_HOST: "pg.develop.sprinthub.ru"
|
||||
DB_PASSWORD: $DB_PASSWORD_DEV
|
||||
DEBUG: "true"
|
||||
command: ./manage.py runserver 0.0.0.0:8000
|
||||
deploy:
|
||||
mode: replicated
|
||||
restart_policy:
|
||||
condition: any
|
||||
update_config:
|
||||
parallelism: 1
|
||||
order: start-first
|
||||
|
||||
networks:
|
||||
net:
|
||||
driver: overlay
|
119
.gitignore
vendored
Normal file
119
.gitignore
vendored
Normal file
@ -0,0 +1,119 @@
|
||||
# Django #
|
||||
*.log
|
||||
*.pot
|
||||
*.pyc
|
||||
__pycache__
|
||||
db.sqlite3
|
||||
media
|
||||
data
|
||||
*/__pycache__
|
||||
|
||||
# Backup files #
|
||||
*.bak
|
||||
|
||||
# If you are using PyCharm #
|
||||
.idea
|
||||
.idea/**/workspace.xml
|
||||
.idea/**/tasks.xml
|
||||
.idea/dictionaries
|
||||
.idea/**/dataSources/
|
||||
.idea/**/dataSources.ids
|
||||
.idea/**/dataSources.xml
|
||||
.idea/**/dataSources.local.xml
|
||||
.idea/**/sqlDataSources.xml
|
||||
.idea/**/dynamic.xml
|
||||
.idea/**/uiDesigner.xml
|
||||
.idea/**/gradle.xml
|
||||
.idea/**/libraries
|
||||
*.iws /out/
|
||||
|
||||
# Python #
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
|
||||
# Distribution / packaging
|
||||
.Python build/
|
||||
develop-eggs/
|
||||
downloads/
|
||||
eggs/
|
||||
.eggs/
|
||||
lib/
|
||||
lib64/
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
wheels/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
*.manifest
|
||||
*.spec
|
||||
|
||||
# Installer logs
|
||||
pip-log.txt
|
||||
pip-delete-this-directory.txt
|
||||
|
||||
# Unit test / coverage reports
|
||||
htmlcov/
|
||||
.tox/
|
||||
.coverage
|
||||
.coverage.*
|
||||
.cache
|
||||
.pytest_cache/
|
||||
nosetests.xml
|
||||
coverage.xml
|
||||
*.cover
|
||||
.hypothesis/
|
||||
postgres-data
|
||||
|
||||
# Jupyter Notebook
|
||||
.ipynb_checkpoints
|
||||
|
||||
# pyenv
|
||||
.python-version
|
||||
|
||||
# celery
|
||||
celerybeat-schedule.*
|
||||
|
||||
# SageMath parsed files
|
||||
*.sage.py
|
||||
|
||||
# Environments
|
||||
.venv
|
||||
env/
|
||||
ENV/
|
||||
venv/
|
||||
env.bak/
|
||||
venv.bak/
|
||||
|
||||
# mkdocs documentation
|
||||
/site
|
||||
|
||||
# mypy
|
||||
.mypy_cache/
|
||||
|
||||
# Sublime Text #
|
||||
*.tmlanguage.cache
|
||||
*.tmPreferences.cache
|
||||
*.stTheme.cache
|
||||
*.sublime-workspace
|
||||
*.sublime-project
|
||||
|
||||
# sftp configuration file
|
||||
sftp-config.json
|
||||
|
||||
# Package control specific files Package
|
||||
Control.last-run
|
||||
Control.ca-list
|
||||
Control.ca-bundle
|
||||
Control.system-ca-bundle
|
||||
GitHub.sublime-settings
|
||||
|
||||
# Visual Studio Code #
|
||||
.vscode
|
||||
.vscode/*
|
||||
!.vscode/settings.json
|
||||
!.vscode/tasks.json
|
||||
!.vscode/launch.json
|
||||
!.vscode/extensions.json
|
||||
.history
|
45
.gitlab-ci.yml
Normal file
45
.gitlab-ci.yml
Normal file
@ -0,0 +1,45 @@
|
||||
stages:
|
||||
- build
|
||||
- deploy-dev
|
||||
- deploy-prod
|
||||
|
||||
build:
|
||||
stage: build
|
||||
tags:
|
||||
- dev
|
||||
before_script:
|
||||
- docker login -u mathwave -p $DOCKERHUB_PASSWORD
|
||||
script:
|
||||
- docker build -t mathwave/sprint-repo:battleship-back .
|
||||
- docker push mathwave/sprint-repo:battleship-back
|
||||
- docker build -t mathwave/sprint-repo:battleship-nginx nginx
|
||||
- docker push mathwave/sprint-repo:battleship-nginx
|
||||
|
||||
.deploy:
|
||||
before_script:
|
||||
- docker login -u mathwave -p $DOCKERHUB_PASSWORD
|
||||
|
||||
deploy-dev:
|
||||
extends:
|
||||
- .deploy
|
||||
stage: deploy-dev
|
||||
tags:
|
||||
- dev
|
||||
rules:
|
||||
- if: '$CI_COMMIT_BRANCH == "master"'
|
||||
when: on_success
|
||||
- when: manual
|
||||
script:
|
||||
- docker stack deploy -c ./.deploy/deploy-dev.yaml sprint
|
||||
|
||||
deploy-prod:
|
||||
extends:
|
||||
- .deploy
|
||||
stage: deploy-prod
|
||||
tags:
|
||||
- prod
|
||||
only:
|
||||
- master
|
||||
when: manual
|
||||
script:
|
||||
- docker stack deploy -c ./.deploy/deploy-prod.yaml sprint
|
13
Dockerfile
Normal file
13
Dockerfile
Normal file
@ -0,0 +1,13 @@
|
||||
FROM python:3.8
|
||||
|
||||
ENV PYTHONUNBUFFERED 1
|
||||
ENV DJANGO_SETTINGS_MODULE battleship_back.settings
|
||||
RUN mkdir -p /usr/src/app/
|
||||
|
||||
COPY requirements.txt /usr/src/app/requirements.txt
|
||||
|
||||
WORKDIR /usr/src/app/
|
||||
|
||||
RUN pip3 install -r requirements.txt
|
||||
|
||||
COPY . /usr/src/app/
|
0
battleship/__init__.py
Normal file
0
battleship/__init__.py
Normal file
3
battleship/admin.py
Normal file
3
battleship/admin.py
Normal file
@ -0,0 +1,3 @@
|
||||
from django.contrib import admin
|
||||
|
||||
# Register your models here.
|
6
battleship/apps.py
Normal file
6
battleship/apps.py
Normal file
@ -0,0 +1,6 @@
|
||||
from django.apps import AppConfig
|
||||
|
||||
|
||||
class BattleshipConfig(AppConfig):
|
||||
default_auto_field = 'django.db.models.BigAutoField'
|
||||
name = 'battleship'
|
44
battleship/migrations/0001_initial.py
Normal file
44
battleship/migrations/0001_initial.py
Normal file
@ -0,0 +1,44 @@
|
||||
# Generated by Django 3.2.15 on 2022-08-22 11:35
|
||||
|
||||
import battleship.models
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
import django.utils.timezone
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='Game',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('last_move_ts', models.DateTimeField(default=django.utils.timezone.now)),
|
||||
('turn', models.IntegerField(default=0)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='Player',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('field', models.CharField(default=' ', max_length=100)),
|
||||
('number', models.IntegerField()),
|
||||
('attend_token', models.CharField(default=battleship.models.generate_token, max_length=30)),
|
||||
('token', models.CharField(blank=True, max_length=30, null=True)),
|
||||
('game', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='battleship.game')),
|
||||
],
|
||||
),
|
||||
migrations.AddConstraint(
|
||||
model_name='player',
|
||||
constraint=models.UniqueConstraint(fields=('token',), name='unique_player_token'),
|
||||
),
|
||||
migrations.AddConstraint(
|
||||
model_name='player',
|
||||
constraint=models.UniqueConstraint(fields=('attend_token',), name='unique_player_attend_token'),
|
||||
),
|
||||
]
|
0
battleship/migrations/__init__.py
Normal file
0
battleship/migrations/__init__.py
Normal file
30
battleship/models.py
Normal file
30
battleship/models.py
Normal file
@ -0,0 +1,30 @@
|
||||
import random
|
||||
|
||||
from django.db import models
|
||||
|
||||
# Create your models here.
|
||||
from django.utils import timezone
|
||||
|
||||
|
||||
def generate_token():
|
||||
letters = 'qwertyuioppasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM1234567890'
|
||||
return ''.join([random.choice(letters) for _ in range(30)])
|
||||
|
||||
|
||||
class Game(models.Model):
|
||||
last_move_ts = models.DateTimeField(default=timezone.now)
|
||||
turn = models.IntegerField(default=0)
|
||||
|
||||
|
||||
class Player(models.Model):
|
||||
game = models.ForeignKey(Game, on_delete=models.CASCADE)
|
||||
field = models.CharField(max_length=100, default=' ' * 100)
|
||||
number = models.IntegerField()
|
||||
attend_token = models.CharField(max_length=30, default=generate_token)
|
||||
token = models.CharField(max_length=30, null=True, blank=True)
|
||||
|
||||
class Meta:
|
||||
constraints = [
|
||||
models.UniqueConstraint(fields=['token'], name='unique_player_token'),
|
||||
models.UniqueConstraint(fields=['attend_token'], name='unique_player_attend_token')
|
||||
]
|
3
battleship/tests.py
Normal file
3
battleship/tests.py
Normal file
@ -0,0 +1,3 @@
|
||||
from django.test import TestCase
|
||||
|
||||
# Create your tests here.
|
99
battleship/views.py
Normal file
99
battleship/views.py
Normal file
@ -0,0 +1,99 @@
|
||||
from django.http import JsonResponse
|
||||
from django.shortcuts import render
|
||||
|
||||
# Create your views here.
|
||||
from battleship.models import Game, Player, generate_token
|
||||
|
||||
|
||||
def new_game(request):
|
||||
game = Game.objects.create()
|
||||
player1 = Player.objects.create(
|
||||
game=game,
|
||||
number=0,
|
||||
token=generate_token()
|
||||
)
|
||||
player2 = Player.objects.create(
|
||||
game=game,
|
||||
number=1
|
||||
)
|
||||
return JsonResponse({
|
||||
'game_id': game.id,
|
||||
'player_token': player2.attend_token,
|
||||
'my_token': player1.token
|
||||
})
|
||||
|
||||
|
||||
def attend_game(request):
|
||||
game_id = request.POST['game_id']
|
||||
attend_token = request.POST['attend_token']
|
||||
player = Player.objects.get(game_id=game_id, attend_token=attend_token)
|
||||
if player.token is not None:
|
||||
return JsonResponse({}, status=403)
|
||||
player.token = generate_token()
|
||||
player.save()
|
||||
return JsonResponse({
|
||||
'token': player.token
|
||||
})
|
||||
|
||||
|
||||
def place_ships(request):
|
||||
game_id = request.POST['game_id']
|
||||
token = request.POST['token']
|
||||
player = Player.objects.get(game_id=game_id, token=token)
|
||||
if player.field != ' ' * 100:
|
||||
return JsonResponse({}, status=403)
|
||||
player.field = request.POST['field']
|
||||
player.save()
|
||||
return JsonResponse({})
|
||||
|
||||
|
||||
def check_opponent(request):
|
||||
game_id = request.POST['game_id']
|
||||
token = request.POST['token']
|
||||
player = Player.objects.get(game_id=game_id, token=token)
|
||||
player2 = Player.objects.filter(game_id=game_id, number=(1 - player.number)).first()
|
||||
if player2 is None:
|
||||
return JsonResponse({"attend": False, "ready": False})
|
||||
return JsonResponse({"attend": True, "ready": player2.field != ' ' * 100})
|
||||
|
||||
|
||||
def shoot(request):
|
||||
game_id = request.POST['game_id']
|
||||
token = request.POST['token']
|
||||
player = Player.objects.get(game_id=game_id, token=token)
|
||||
if player.game.turn != player.number:
|
||||
return JsonResponse({}, status=403)
|
||||
player2 = Player.objects.get(game_id=game_id, number=(1 - player.number))
|
||||
h = request.POST['h']
|
||||
v = request.POST['v']
|
||||
pos = h * 10 + v
|
||||
if player2.field[pos] == 'x' or player2.field[pos] == '.':
|
||||
return JsonResponse({}, status=403)
|
||||
if player2.field[pos] == 'o':
|
||||
new_symb = 'x'
|
||||
else:
|
||||
new_symb = '.'
|
||||
player.game.turn = (1 - player.game.turn)
|
||||
player.game.save()
|
||||
if pos == 0:
|
||||
player2.field = new_symb + player2.field[1:]
|
||||
elif pos == 99:
|
||||
player2.field[:99] + new_symb
|
||||
else:
|
||||
player2.field = player2.field[:pos] + new_symb + player2.field[pos + 1:]
|
||||
player2.save()
|
||||
return JsonResponse({
|
||||
'shot': new_symb == 'x',
|
||||
'game_finish': 'o' not in player2.field
|
||||
})
|
||||
|
||||
|
||||
def check_status(request):
|
||||
game_id = request.POST['game_id']
|
||||
token = request.POST['token']
|
||||
player = Player.objects.get(game_id=game_id, token=token)
|
||||
player2 = Player.objects.get(game_id=game_id, number=1 - player.number)
|
||||
return JsonResponse({
|
||||
'my_turn': player.game.turn == player.number,
|
||||
'game_finished': 'o' not in player.field or 'o' not in player2.field
|
||||
})
|
0
battleship_back/__init__.py
Normal file
0
battleship_back/__init__.py
Normal file
16
battleship_back/asgi.py
Normal file
16
battleship_back/asgi.py
Normal file
@ -0,0 +1,16 @@
|
||||
"""
|
||||
ASGI config for battleship_back project.
|
||||
|
||||
It exposes the ASGI callable as a module-level variable named ``application``.
|
||||
|
||||
For more information on this file, see
|
||||
https://docs.djangoproject.com/en/3.2/howto/deployment/asgi/
|
||||
"""
|
||||
|
||||
import os
|
||||
|
||||
from django.core.asgi import get_asgi_application
|
||||
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'battleship_back.settings')
|
||||
|
||||
application = get_asgi_application()
|
131
battleship_back/settings.py
Normal file
131
battleship_back/settings.py
Normal file
@ -0,0 +1,131 @@
|
||||
"""
|
||||
Django settings for battleship_back project.
|
||||
|
||||
Generated by 'django-admin startproject' using Django 3.2.15.
|
||||
|
||||
For more information on this file, see
|
||||
https://docs.djangoproject.com/en/3.2/topics/settings/
|
||||
|
||||
For the full list of settings and their values, see
|
||||
https://docs.djangoproject.com/en/3.2/ref/settings/
|
||||
"""
|
||||
import os
|
||||
from pathlib import Path
|
||||
|
||||
# Build paths inside the project like this: BASE_DIR / 'subdir'.
|
||||
BASE_DIR = Path(__file__).resolve().parent.parent
|
||||
|
||||
|
||||
# Quick-start development settings - unsuitable for production
|
||||
# See https://docs.djangoproject.com/en/3.2/howto/deployment/checklist/
|
||||
|
||||
# SECURITY WARNING: keep the secret key used in production secret!
|
||||
SECRET_KEY = 'django-insecure-$x=irhk3ts1ae%b+kpr+g%mk3v$eb++^0eh(eg0)+-hb+ane82'
|
||||
|
||||
# SECURITY WARNING: don't run with debug turned on in production!
|
||||
DEBUG = True
|
||||
|
||||
ALLOWED_HOSTS = []
|
||||
|
||||
|
||||
# Application definition
|
||||
|
||||
INSTALLED_APPS = [
|
||||
'django.contrib.admin',
|
||||
'django.contrib.auth',
|
||||
'django.contrib.contenttypes',
|
||||
'django.contrib.sessions',
|
||||
'django.contrib.messages',
|
||||
'django.contrib.staticfiles',
|
||||
'battleship.apps.BattleshipConfig',
|
||||
]
|
||||
|
||||
MIDDLEWARE = [
|
||||
'django.middleware.security.SecurityMiddleware',
|
||||
'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 = 'battleship_back.urls'
|
||||
|
||||
TEMPLATES = [
|
||||
{
|
||||
'BACKEND': 'django.template.backends.django.DjangoTemplates',
|
||||
'DIRS': [os.path.join(BASE_DIR, 'templates')]
|
||||
,
|
||||
'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 = 'battleship_back.wsgi.application'
|
||||
|
||||
|
||||
# Database
|
||||
# https://docs.djangoproject.com/en/3.2/ref/settings/#databases
|
||||
|
||||
DATABASES = {
|
||||
"default": {
|
||||
"ENGINE": "django.db.backends.postgresql_psycopg2",
|
||||
"NAME": "battleship",
|
||||
"USER": "postgres",
|
||||
"PASSWORD": os.getenv("DB_PASSWORD", "password"),
|
||||
"HOST": os.getenv("DB_HOST", "127.0.0.1"),
|
||||
"PORT": 5432,
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
# Password validation
|
||||
# https://docs.djangoproject.com/en/3.2/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/3.2/topics/i18n/
|
||||
|
||||
LANGUAGE_CODE = 'en-us'
|
||||
|
||||
TIME_ZONE = 'UTC'
|
||||
|
||||
USE_I18N = True
|
||||
|
||||
USE_L10N = True
|
||||
|
||||
USE_TZ = True
|
||||
|
||||
|
||||
# Static files (CSS, JavaScript, Images)
|
||||
# https://docs.djangoproject.com/en/3.2/howto/static-files/
|
||||
|
||||
STATIC_URL = '/static/'
|
||||
|
||||
# Default primary key field type
|
||||
# https://docs.djangoproject.com/en/3.2/ref/settings/#default-auto-field
|
||||
|
||||
DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'
|
29
battleship_back/urls.py
Normal file
29
battleship_back/urls.py
Normal file
@ -0,0 +1,29 @@
|
||||
"""battleship_back URL Configuration
|
||||
|
||||
The `urlpatterns` list routes URLs to views. For more information please see:
|
||||
https://docs.djangoproject.com/en/3.2/topics/http/urls/
|
||||
Examples:
|
||||
Function views
|
||||
1. Add an import: from my_app import views
|
||||
2. Add a URL to urlpatterns: path('', views.home, name='home')
|
||||
Class-based views
|
||||
1. Add an import: from other_app.views import Home
|
||||
2. Add a URL to urlpatterns: path('', Home.as_view(), name='home')
|
||||
Including another URLconf
|
||||
1. Import the include() function: from django.urls import include, path
|
||||
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
|
||||
"""
|
||||
from django.contrib import admin
|
||||
from django.urls import path
|
||||
|
||||
from battleship import views
|
||||
|
||||
urlpatterns = [
|
||||
path('admin/', admin.site.urls),
|
||||
path('api/new_game', views.new_game),
|
||||
path('api/check_status', views.check_status),
|
||||
path('api/check_opponent', views.check_opponent),
|
||||
path('api/attend_game', views.attend_game),
|
||||
path('api/place_ships', views.place_ships),
|
||||
path('api/shoot', views.shoot)
|
||||
]
|
16
battleship_back/wsgi.py
Normal file
16
battleship_back/wsgi.py
Normal file
@ -0,0 +1,16 @@
|
||||
"""
|
||||
WSGI config for battleship_back project.
|
||||
|
||||
It exposes the WSGI callable as a module-level variable named ``application``.
|
||||
|
||||
For more information on this file, see
|
||||
https://docs.djangoproject.com/en/3.2/howto/deployment/wsgi/
|
||||
"""
|
||||
|
||||
import os
|
||||
|
||||
from django.core.wsgi import get_wsgi_application
|
||||
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'battleship_back.settings')
|
||||
|
||||
application = get_wsgi_application()
|
22
manage.py
Executable file
22
manage.py
Executable file
@ -0,0 +1,22 @@
|
||||
#!/usr/bin/env python
|
||||
"""Django's command-line utility for administrative tasks."""
|
||||
import os
|
||||
import sys
|
||||
|
||||
|
||||
def main():
|
||||
"""Run administrative tasks."""
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'battleship_back.settings')
|
||||
try:
|
||||
from django.core.management import execute_from_command_line
|
||||
except ImportError as exc:
|
||||
raise ImportError(
|
||||
"Couldn't import Django. Are you sure it's installed and "
|
||||
"available on your PYTHONPATH environment variable? Did you "
|
||||
"forget to activate a virtual environment?"
|
||||
) from exc
|
||||
execute_from_command_line(sys.argv)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
2
nginx/Dockerfile
Normal file
2
nginx/Dockerfile
Normal file
@ -0,0 +1,2 @@
|
||||
FROM nginx
|
||||
COPY nginx.conf /etc/nginx/nginx.conf
|
15
nginx/nginx.conf
Normal file
15
nginx/nginx.conf
Normal file
@ -0,0 +1,15 @@
|
||||
events {}
|
||||
|
||||
http {
|
||||
server {
|
||||
listen 80;
|
||||
|
||||
location /api/ {
|
||||
proxy_pass http://backend:8000/api/;
|
||||
}
|
||||
|
||||
; location / {
|
||||
; proxy_pass http://frontend:3000/;
|
||||
; }
|
||||
}
|
||||
}
|
6
requirements.txt
Normal file
6
requirements.txt
Normal file
@ -0,0 +1,6 @@
|
||||
asgiref==3.5.2
|
||||
Django==3.2.15
|
||||
psycopg2==2.9.3
|
||||
pytz==2022.2.1
|
||||
sqlparse==0.4.2
|
||||
typing_extensions==4.3.0
|
Loading…
Reference in New Issue
Block a user