Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e102db9f38 | ||
|
|
15c5f5293a | ||
|
|
f11f9a0d97 | ||
|
|
c80ef7441d | ||
|
|
a2e7a92280 | ||
|
|
b4ef050e1f | ||
|
|
c09558c0ff | ||
|
|
c3666783e4 | ||
|
|
54c6336e22 | ||
|
|
7da888aa09 | ||
|
|
07d98bf11d | ||
|
|
4584ba0143 | ||
|
|
6feac7ef2e | ||
|
|
5efb93ea68 | ||
|
|
1ee6b890ef | ||
|
|
18aab648c6 | ||
|
|
57328a7fc8 |
@@ -1,7 +1,6 @@
|
||||
variables:
|
||||
CI_PROJECT_DIR: "."
|
||||
CI_REGISTRY_IMAGE: hub.ducoterra.net/ducoterra/mysite
|
||||
DEPLOY: test
|
||||
|
||||
stages:
|
||||
- build
|
||||
@@ -17,24 +16,27 @@ build:
|
||||
name: gcr.io/kaniko-project/executor:debug
|
||||
entrypoint: [""]
|
||||
script:
|
||||
- echo $DEPLOY
|
||||
- /kaniko/executor --context $CI_PROJECT_DIR --dockerfile $CI_PROJECT_DIR/Dockerfile --destination $CI_REGISTRY_IMAGE:$CI_COMMIT_TAG
|
||||
|
||||
test:
|
||||
stage: test
|
||||
only:
|
||||
variables:
|
||||
- $CI_COMMIT_TAG
|
||||
stage: test
|
||||
image:
|
||||
name: $CI_REGISTRY_IMAGE:$CI_COMMIT_TAG
|
||||
entrypoint: [""]
|
||||
script:
|
||||
- python manage.py test
|
||||
|
||||
deploy:
|
||||
deploy_to_test:
|
||||
variables:
|
||||
DEPLOY: test
|
||||
stage: deploy
|
||||
only:
|
||||
variables:
|
||||
- $CI_COMMIT_TAG
|
||||
stage: deploy
|
||||
image:
|
||||
name: debian:10
|
||||
entrypoint: [""]
|
||||
@@ -44,6 +46,31 @@ deploy:
|
||||
- chmod +x ./kubectl
|
||||
- mkdir /deploy
|
||||
- for f in $(find k8s -regex '.*\.ya*ml'); do envsubst < $f > "/deploy/$(basename $f)"; done
|
||||
- for f in $(find k8s/test -regex '.*\.ya*ml'); do envsubst < $f > "/deploy/$(basename $f)"; done
|
||||
- ./kubectl apply -f /deploy
|
||||
- ./kubectl rollout status deploy $DEPLOY
|
||||
- ./kubectl exec $(./kubectl get pods --selector=app=$DEPLOY --output=jsonpath='{.items[*].metadata.name}') -- python manage.py migrate
|
||||
- POD=$(./kubectl get pods --selector=app=$DEPLOY --output=jsonpath='{.items[*].metadata.name}')
|
||||
- ./kubectl exec $POD -- python manage.py migrate
|
||||
|
||||
deploy_to_prod:
|
||||
variables:
|
||||
DEPLOY: prod
|
||||
stage: deploy
|
||||
only:
|
||||
variables:
|
||||
- $CI_COMMIT_TAG
|
||||
when: manual
|
||||
image:
|
||||
name: debian:10
|
||||
entrypoint: [""]
|
||||
script:
|
||||
- apt -qq update >> /dev/null && apt -qq install -y curl gettext >> /dev/null
|
||||
- curl -LO https://storage.googleapis.com/kubernetes-release/release/`curl -s https://storage.googleapis.com/kubernetes-release/release/stable.txt`/bin/linux/amd64/kubectl
|
||||
- chmod +x ./kubectl
|
||||
- mkdir /deploy
|
||||
- for f in $(find k8s -regex '.*\.ya*ml'); do envsubst < $f > "/deploy/$(basename $f)"; done
|
||||
- for f in $(find k8s/prod -regex '.*\.ya*ml'); do envsubst < $f > "/deploy/$(basename $f)"; done
|
||||
- ./kubectl apply -f /deploy
|
||||
- ./kubectl rollout status deploy $DEPLOY
|
||||
- POD=$(./kubectl get pods --selector=app=$DEPLOY --output=jsonpath='{.items[*].metadata.name}')
|
||||
- ./kubectl exec $POD -- python manage.py migrate
|
||||
13
.vscode/launch.json
vendored
13
.vscode/launch.json
vendored
@@ -5,7 +5,7 @@
|
||||
"version": "0.2.0",
|
||||
"configurations": [
|
||||
{
|
||||
"name": "Python: Django",
|
||||
"name": "Test",
|
||||
"type": "python",
|
||||
"request": "launch",
|
||||
"program": "${workspaceFolder}/manage.py",
|
||||
@@ -13,6 +13,17 @@
|
||||
"test",
|
||||
],
|
||||
"django": true
|
||||
},
|
||||
{
|
||||
"name": "Run Server",
|
||||
"type": "python",
|
||||
"request": "launch",
|
||||
"program": "${workspaceFolder}/manage.py",
|
||||
"args": [
|
||||
"runserver",
|
||||
"--noreload"
|
||||
],
|
||||
"django": true
|
||||
}
|
||||
]
|
||||
}
|
||||
@@ -18,7 +18,7 @@ from django.urls import path, include
|
||||
from django.http import JsonResponse
|
||||
|
||||
urlpatterns = [
|
||||
path('', include('api.urls')),
|
||||
# path('api/', include('api.urls')),
|
||||
path('', include('ui.urls')),
|
||||
path('admin/', admin.site.urls),
|
||||
# path('admin/', admin.site.urls),
|
||||
]
|
||||
|
||||
@@ -3,4 +3,4 @@ kind: ConfigMap
|
||||
metadata:
|
||||
name: $DEPLOY
|
||||
data:
|
||||
ALLOWED_HOSTS: localhost,test.ducoterra.net
|
||||
ALLOWED_HOSTS: localhost,$DEPLOY.ducoterra.net
|
||||
@@ -17,6 +17,8 @@ spec:
|
||||
envFrom:
|
||||
- configMapRef:
|
||||
name: $DEPLOY
|
||||
- secretRef:
|
||||
name: django-secrets
|
||||
volumeMounts:
|
||||
- mountPath: /app/db
|
||||
name: $DEPLOY
|
||||
|
||||
@@ -7,10 +7,10 @@ metadata:
|
||||
spec:
|
||||
tls:
|
||||
- hosts:
|
||||
- test.ducoterra.net
|
||||
- $DEPLOY.ducoterra.net
|
||||
secretName: letsencrypt
|
||||
rules:
|
||||
- host: test.ducoterra.net
|
||||
- host: $DEPLOY.ducoterra.net
|
||||
http:
|
||||
paths:
|
||||
- backend:
|
||||
@@ -29,7 +29,7 @@ spec:
|
||||
tls:
|
||||
secretName: letsencrypt
|
||||
routes:
|
||||
- match: Host(`test.ducoterra.net`)
|
||||
- match: Host(`$DEPLOY.ducoterra.net`)
|
||||
kind: Rule
|
||||
services:
|
||||
- name: $DEPLOY
|
||||
@@ -45,7 +45,7 @@ spec:
|
||||
entryPoints:
|
||||
- web
|
||||
routes:
|
||||
- match: Host(`test.ducoterra.net`)
|
||||
- match: Host(`$DEPLOY.ducoterra.net`)
|
||||
kind: Rule
|
||||
services:
|
||||
- name: $DEPLOY
|
||||
18
k8s/test/ingress.yaml
Normal file
18
k8s/test/ingress.yaml
Normal file
@@ -0,0 +1,18 @@
|
||||
apiVersion: networking.k8s.io/v1beta1
|
||||
kind: Ingress
|
||||
metadata:
|
||||
annotations:
|
||||
ingress.kubernetes.io/ssl-redirect: "true"
|
||||
name: $DEPLOY
|
||||
spec:
|
||||
tls:
|
||||
- hosts:
|
||||
- $DEPLOY.ducoterra.net
|
||||
secretName: letsencrypt
|
||||
rules:
|
||||
- host: $DEPLOY.ducoterra.net
|
||||
http:
|
||||
paths:
|
||||
- backend:
|
||||
serviceName: $DEPLOY
|
||||
servicePort: 8000
|
||||
29
ui/static/ui/achievement.css
Normal file
29
ui/static/ui/achievement.css
Normal file
@@ -0,0 +1,29 @@
|
||||
.achievement {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
text-align: center;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
display: flex;
|
||||
position: absolute;
|
||||
}
|
||||
|
||||
.achievement-animate {
|
||||
animation-name: moveup;
|
||||
animation-duration: 2s;
|
||||
animation-fill-mode: forwards;
|
||||
}
|
||||
|
||||
.achievement-text {
|
||||
font-size: 4rem;
|
||||
}
|
||||
|
||||
@keyframes moveup {
|
||||
from {bottom: 0px;}
|
||||
to {bottom: 200px; color: white;}
|
||||
}
|
||||
|
||||
@keyframes fadeout {
|
||||
from {}
|
||||
to {color: transparent;}
|
||||
}
|
||||
@@ -1,6 +1,8 @@
|
||||
html, body {
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
position: fixed;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.section, .container {
|
||||
|
||||
@@ -1,26 +1,53 @@
|
||||
function getCookie(name) {
|
||||
var cookieValue = null;
|
||||
if (document.cookie && document.cookie !== '') {
|
||||
var cookies = document.cookie.split(';');
|
||||
for (var i = 0; i < cookies.length; i++) {
|
||||
var cookie = cookies[i].trim();
|
||||
// Does this cookie string begin with the name we want?
|
||||
if (cookie.substring(0, name.length + 1) === (name + '=')) {
|
||||
cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return cookieValue;
|
||||
const csrftoken = getCookie('csrftoken');
|
||||
const button = document.getElementById("BUTTON");
|
||||
const count = document.getElementById("COUNT");
|
||||
const button_container = document.getElementById("button-container");
|
||||
const achievement = document.getElementById("achievement");
|
||||
|
||||
const achievements = {
|
||||
1: "Clicked!",
|
||||
2: "Clicked Twice!",
|
||||
4: "2^2",
|
||||
8: "2^3",
|
||||
16: "2^4",
|
||||
32: "2^5",
|
||||
64: "2^6",
|
||||
69: "Sixty-Nine",
|
||||
100: "one hundred",
|
||||
128: "2^7",
|
||||
200: "two hundred",
|
||||
250: "quarter thousand",
|
||||
256: "2^8",
|
||||
420: "Blaze it",
|
||||
500: "half thousand",
|
||||
512: "2^9",
|
||||
1000: "one thousand",
|
||||
1024: "2^10",
|
||||
2048: "2048!",
|
||||
2500: "Keep going!",
|
||||
4096: "2^11",
|
||||
5000: "halfway to ten thousand"
|
||||
}
|
||||
|
||||
var csrftoken = getCookie('csrftoken');
|
||||
var button = document.getElementById("BUTTON");
|
||||
var count = document.getElementById("COUNT");
|
||||
function add_achievement(text) {
|
||||
achievement.querySelector(".achievement-text").innerText = text;
|
||||
achievement.classList.remove("achievement-animate");
|
||||
void achievement.offsetWidth;
|
||||
achievement.classList.add("achievement-animate");
|
||||
}
|
||||
|
||||
function check_achievement(count) {
|
||||
current_achievement = achievements[Number(count)]
|
||||
if (current_achievement != undefined) {
|
||||
add_achievement(current_achievement)
|
||||
}
|
||||
}
|
||||
|
||||
// when button is clicked submit an empty post request
|
||||
button.addEventListener("click", event => {
|
||||
button.disabled = true;
|
||||
fetch('/button', {
|
||||
button.classList.add("is-loading");
|
||||
fetch('/button/', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
@@ -32,9 +59,12 @@ button.addEventListener("click", event => {
|
||||
})
|
||||
.then((data) => {
|
||||
count.innerText = data.pressed;
|
||||
check_achievement(data.pressed);
|
||||
}).finally(() => {
|
||||
button.disabled = false;
|
||||
button.classList.remove("is-loading");
|
||||
});
|
||||
});
|
||||
|
||||
// when the page is loaded automatically select the button
|
||||
button.focus();
|
||||
16
ui/static/ui/helper.js
Normal file
16
ui/static/ui/helper.js
Normal file
@@ -0,0 +1,16 @@
|
||||
// get cookies when fetching with django
|
||||
function getCookie(name) {
|
||||
var cookieValue = null;
|
||||
if (document.cookie && document.cookie !== '') {
|
||||
var cookies = document.cookie.split(';');
|
||||
for (var i = 0; i < cookies.length; i++) {
|
||||
var cookie = cookies[i].trim();
|
||||
// Does this cookie string begin with the name we want?
|
||||
if (cookie.substring(0, name.length + 1) === (name + '=')) {
|
||||
cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return cookieValue;
|
||||
}
|
||||
@@ -4,16 +4,22 @@
|
||||
|
||||
{% block css %}
|
||||
<link rel="stylesheet" href="{% static 'ui/button.css' %}">
|
||||
<link rel="stylesheet" href="{% static 'ui/achievement.css' %}">
|
||||
{% endblock %}
|
||||
|
||||
{% block js %}
|
||||
<script src="{% static 'ui/helper.js' %}"></script>
|
||||
<script src="{% static 'ui/button.js' %}"></script>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
{% block body %}
|
||||
<section class="section">
|
||||
<div class="container">
|
||||
<div id = "button-container" class="container">
|
||||
<div id = "achievement" class="achievement">
|
||||
<div>
|
||||
<div class="achievement-text"></div>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<h1 class="title">
|
||||
The Button
|
||||
|
||||
@@ -2,5 +2,5 @@ from django.urls import path
|
||||
from . import views
|
||||
|
||||
urlpatterns = [
|
||||
path('button', views.button, name = 'button'),
|
||||
path('button/', views.button, name = 'button'),
|
||||
]
|
||||
Reference in New Issue
Block a user