Compare commits
22 Commits
docker-api
...
0f4ac8e444
Author | SHA1 | Date | |
---|---|---|---|
0f4ac8e444 | |||
3cf501a393 | |||
180efa01af | |||
7ed847251f | |||
e3a4776a5d | |||
357d99cb91 | |||
f5c079ffc2 | |||
0e5dfe6bde | |||
2adbb8d94c | |||
8446d11720 | |||
a580b04659 | |||
2e6894ad14 | |||
2e6c6517f3 | |||
be6c1b38b7 | |||
aa8e129913 | |||
71e09807a7 | |||
4816170c03 | |||
6af241e8be | |||
c8ec53f207 | |||
908d02803f | |||
6e6bd2b143 | |||
47fbaab403 |
24
Dockerfile
24
Dockerfile
@ -1,8 +1,17 @@
|
|||||||
FROM python:3.8-alpine as build
|
FROM python:3.8-alpine as build
|
||||||
|
|
||||||
RUN apk add gettext git gcc python3-dev musl-dev \
|
RUN apk add --no-cache \
|
||||||
libffi-dev zlib-dev jpeg-dev libjpeg postgresql-dev build-base \
|
build-base \
|
||||||
--virtual .build-dependencies
|
gcc \
|
||||||
|
gettext \
|
||||||
|
git \
|
||||||
|
jpeg-dev \
|
||||||
|
libffi-dev \
|
||||||
|
libjpeg \
|
||||||
|
musl-dev \
|
||||||
|
postgresql-dev \
|
||||||
|
python3-dev \
|
||||||
|
zlib-dev
|
||||||
|
|
||||||
RUN mkdir -p /app/{code,venv}
|
RUN mkdir -p /app/{code,venv}
|
||||||
WORKDIR /app/code
|
WORKDIR /app/code
|
||||||
@ -17,7 +26,14 @@ RUN pipenv install --deploy --verbose
|
|||||||
|
|
||||||
FROM python:3.8-alpine
|
FROM python:3.8-alpine
|
||||||
|
|
||||||
RUN apk add --no-cache libpq libstdc++ libjpeg libvirt-client
|
RUN apk add --no-cache \
|
||||||
|
cloud-utils \
|
||||||
|
libjpeg \
|
||||||
|
libpq \
|
||||||
|
libstdc++ \
|
||||||
|
libvirt-client \
|
||||||
|
openssh-client \
|
||||||
|
virt-install
|
||||||
|
|
||||||
COPY . /app/code/
|
COPY . /app/code/
|
||||||
WORKDIR /app/code
|
WORKDIR /app/code
|
||||||
|
@ -26,12 +26,27 @@ class StdoutMockFlaskMail:
|
|||||||
def send(self, message: Message):
|
def send(self, message: Message):
|
||||||
current_app.logger.info(f"Email would have been sent if configured:\n\nto: {','.join(message.recipients)}\nsubject: {message.subject}\nbody:\n\n{message.body}\n\n")
|
current_app.logger.info(f"Email would have been sent if configured:\n\nto: {','.join(message.recipients)}\nsubject: {message.subject}\nbody:\n\n{message.body}\n\n")
|
||||||
|
|
||||||
|
|
||||||
load_dotenv(find_dotenv())
|
load_dotenv(find_dotenv())
|
||||||
|
|
||||||
|
for var_name in [
|
||||||
|
"SPOKE_HOST_TOKEN", "HUB_TOKEN", "STRIPE_SECRET_KEY",
|
||||||
|
"BTCPAY_PRIVATE_KEY", "MAIL_PASSWORD"
|
||||||
|
]:
|
||||||
|
var = os.environ.get(f"{var_name}_FILE")
|
||||||
|
if not var:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if not os.path.isfile(var):
|
||||||
|
continue
|
||||||
|
|
||||||
|
with open(var) as secret_file:
|
||||||
|
os.environ[var_name] = secret_file.read().rstrip('\n')
|
||||||
|
del os.environ[f"{var_name}_FILE"]
|
||||||
|
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
|
|
||||||
app.config.from_mapping(
|
app.config.from_mapping(
|
||||||
|
|
||||||
BASE_URL=os.environ.get("BASE_URL", default="http://localhost:5000"),
|
BASE_URL=os.environ.get("BASE_URL", default="http://localhost:5000"),
|
||||||
SECRET_KEY=os.environ.get("SECRET_KEY", default="dev"),
|
SECRET_KEY=os.environ.get("SECRET_KEY", default="dev"),
|
||||||
HUB_MODE_ENABLED=os.environ.get("HUB_MODE_ENABLED", default="True").lower() in ['true', '1', 't', 'y', 'yes'],
|
HUB_MODE_ENABLED=os.environ.get("HUB_MODE_ENABLED", default="True").lower() in ['true', '1', 't', 'y', 'yes'],
|
||||||
@ -153,6 +168,7 @@ is_running_server = ('flask run' in command_line) or ('gunicorn' in command_line
|
|||||||
app.logger.info(f"is_running_server: {is_running_server}")
|
app.logger.info(f"is_running_server: {is_running_server}")
|
||||||
|
|
||||||
if app.config['HUB_MODE_ENABLED']:
|
if app.config['HUB_MODE_ENABLED']:
|
||||||
|
|
||||||
if app.config['HUB_MODEL'] == "capsul-flask":
|
if app.config['HUB_MODEL'] == "capsul-flask":
|
||||||
app.config['HUB_MODEL'] = hub_model.CapsulFlaskHub()
|
app.config['HUB_MODEL'] = hub_model.CapsulFlaskHub()
|
||||||
|
|
||||||
@ -174,9 +190,7 @@ if app.config['HUB_MODE_ENABLED']:
|
|||||||
from capsulflask import db
|
from capsulflask import db
|
||||||
db.init_app(app, is_running_server)
|
db.init_app(app, is_running_server)
|
||||||
|
|
||||||
from capsulflask import (
|
from capsulflask import auth, landing, console, payment, metrics, cli, hub_api, admin
|
||||||
auth, landing, console, payment, metrics, cli, hub_api, publicapi, admin
|
|
||||||
)
|
|
||||||
|
|
||||||
app.register_blueprint(landing.bp)
|
app.register_blueprint(landing.bp)
|
||||||
app.register_blueprint(auth.bp)
|
app.register_blueprint(auth.bp)
|
||||||
@ -186,13 +200,13 @@ if app.config['HUB_MODE_ENABLED']:
|
|||||||
app.register_blueprint(cli.bp)
|
app.register_blueprint(cli.bp)
|
||||||
app.register_blueprint(hub_api.bp)
|
app.register_blueprint(hub_api.bp)
|
||||||
app.register_blueprint(admin.bp)
|
app.register_blueprint(admin.bp)
|
||||||
app.register_blueprint(publicapi.bp)
|
|
||||||
|
|
||||||
app.add_url_rule("/", endpoint="index")
|
app.add_url_rule("/", endpoint="index")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if app.config['SPOKE_MODE_ENABLED']:
|
if app.config['SPOKE_MODE_ENABLED']:
|
||||||
|
|
||||||
if app.config['SPOKE_MODEL'] == "shell-scripts":
|
if app.config['SPOKE_MODEL'] == "shell-scripts":
|
||||||
app.config['SPOKE_MODEL'] = spoke_model.ShellScriptSpoke()
|
app.config['SPOKE_MODEL'] = spoke_model.ShellScriptSpoke()
|
||||||
else:
|
else:
|
||||||
@ -220,6 +234,10 @@ def override_url_for():
|
|||||||
return dict(url_for=url_for_with_cache_bust)
|
return dict(url_for=url_for_with_cache_bust)
|
||||||
|
|
||||||
|
|
||||||
|
@app.context_processor
|
||||||
|
def load_config_vars():
|
||||||
|
return dict(config=app.config)
|
||||||
|
|
||||||
def url_for_with_cache_bust(endpoint, **values):
|
def url_for_with_cache_bust(endpoint, **values):
|
||||||
"""
|
"""
|
||||||
Add a query parameter based on the hash of the file, this acts as a cache bust
|
Add a query parameter based on the hash of the file, this acts as a cache bust
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
from base64 import b64decode
|
|
||||||
import functools
|
import functools
|
||||||
import re
|
import re
|
||||||
|
|
||||||
@ -25,15 +24,6 @@ def account_required(view):
|
|||||||
|
|
||||||
@functools.wraps(view)
|
@functools.wraps(view)
|
||||||
def wrapped_view(**kwargs):
|
def wrapped_view(**kwargs):
|
||||||
api_token = request.headers.get('authorization', None)
|
|
||||||
if api_token is not None:
|
|
||||||
email = get_model().authenticate_token(b64decode(api_token).decode('utf-8'))
|
|
||||||
|
|
||||||
if email is not None:
|
|
||||||
session.clear()
|
|
||||||
session["account"] = email
|
|
||||||
session["csrf-token"] = generate()
|
|
||||||
|
|
||||||
if session.get("account") is None or session.get("csrf-token") is None :
|
if session.get("account") is None or session.get("csrf-token") is None :
|
||||||
return redirect(url_for("auth.login"))
|
return redirect(url_for("auth.login"))
|
||||||
|
|
||||||
|
@ -1,9 +1,7 @@
|
|||||||
from base64 import b64encode
|
|
||||||
from datetime import datetime, timedelta
|
|
||||||
import json
|
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
|
import json
|
||||||
|
from datetime import datetime, timedelta
|
||||||
from flask import Blueprint
|
from flask import Blueprint
|
||||||
from flask import flash
|
from flask import flash
|
||||||
from flask import current_app
|
from flask import current_app
|
||||||
@ -100,6 +98,7 @@ def index():
|
|||||||
@bp.route("/<string:id>", methods=("GET", "POST"))
|
@bp.route("/<string:id>", methods=("GET", "POST"))
|
||||||
@account_required
|
@account_required
|
||||||
def detail(id):
|
def detail(id):
|
||||||
|
|
||||||
duration=request.args.get('duration')
|
duration=request.args.get('duration')
|
||||||
if not duration:
|
if not duration:
|
||||||
duration = "5m"
|
duration = "5m"
|
||||||
@ -189,13 +188,23 @@ def detail(id):
|
|||||||
duration=duration
|
duration=duration
|
||||||
)
|
)
|
||||||
|
|
||||||
def _create(vm_sizes, operating_systems, public_keys_for_account, server_data):
|
|
||||||
|
@bp.route("/create", methods=("GET", "POST"))
|
||||||
|
@account_required
|
||||||
|
def create():
|
||||||
|
vm_sizes = get_model().vm_sizes_dict()
|
||||||
|
operating_systems = get_model().operating_systems_dict()
|
||||||
|
public_keys_for_account = get_model().list_ssh_public_keys_for_account(session["account"])
|
||||||
|
account_balance = get_account_balance(get_vms(), get_payments(), datetime.utcnow())
|
||||||
|
capacity_avaliable = current_app.config["HUB_MODEL"].capacity_avaliable(512*1024*1024)
|
||||||
errors = list()
|
errors = list()
|
||||||
|
|
||||||
size = server_data.get("size")
|
if request.method == "POST":
|
||||||
os = server_data.get("os")
|
if "csrf-token" not in request.form or request.form['csrf-token'] != session['csrf-token']:
|
||||||
posted_keys_count = int(server_data.get("ssh_authorized_key_count"))
|
return abort(418, f"u want tea")
|
||||||
|
|
||||||
|
size = request.form["size"]
|
||||||
|
os = request.form["os"]
|
||||||
if not size:
|
if not size:
|
||||||
errors.append("Size is required")
|
errors.append("Size is required")
|
||||||
elif size not in vm_sizes:
|
elif size not in vm_sizes:
|
||||||
@ -206,14 +215,15 @@ def _create(vm_sizes, operating_systems, public_keys_for_account, server_data):
|
|||||||
elif os not in operating_systems:
|
elif os not in operating_systems:
|
||||||
errors.append(f"Invalid os {os}")
|
errors.append(f"Invalid os {os}")
|
||||||
|
|
||||||
|
posted_keys_count = int(request.form["ssh_authorized_key_count"])
|
||||||
posted_keys = list()
|
posted_keys = list()
|
||||||
|
|
||||||
if posted_keys_count > 1000:
|
if posted_keys_count > 1000:
|
||||||
errors.append("something went wrong with ssh keys")
|
errors.append("something went wrong with ssh keys")
|
||||||
else:
|
else:
|
||||||
for i in range(0, posted_keys_count):
|
for i in range(0, posted_keys_count):
|
||||||
if f"ssh_key_{i}" in server_data:
|
if f"ssh_key_{i}" in request.form:
|
||||||
posted_name = server_data.get(f"ssh_key_{i}")
|
posted_name = request.form[f"ssh_key_{i}"]
|
||||||
key = None
|
key = None
|
||||||
for x in public_keys_for_account:
|
for x in public_keys_for_account:
|
||||||
if x['name'] == posted_name:
|
if x['name'] == posted_name:
|
||||||
@ -226,9 +236,7 @@ def _create(vm_sizes, operating_systems, public_keys_for_account, server_data):
|
|||||||
if len(posted_keys) == 0:
|
if len(posted_keys) == 0:
|
||||||
errors.append("At least one SSH Public Key is required")
|
errors.append("At least one SSH Public Key is required")
|
||||||
|
|
||||||
capacity_avaliable = current_app.config["HUB_MODEL"].capacity_avaliable(
|
capacity_avaliable = current_app.config["HUB_MODEL"].capacity_avaliable(vm_sizes[size]['memory_mb']*1024*1024)
|
||||||
vm_sizes[size]['memory_mb']*1024*1024
|
|
||||||
)
|
|
||||||
|
|
||||||
if not capacity_avaliable:
|
if not capacity_avaliable:
|
||||||
errors.append("""
|
errors.append("""
|
||||||
@ -237,45 +245,19 @@ def _create(vm_sizes, operating_systems, public_keys_for_account, server_data):
|
|||||||
|
|
||||||
if len(errors) == 0:
|
if len(errors) == 0:
|
||||||
id = make_capsul_id()
|
id = make_capsul_id()
|
||||||
get_model().create_vm(
|
# we can't create the vm record in the DB yet because its IP address needs to be allocated first.
|
||||||
email=session["account"],
|
# so it will be created when the allocation happens inside the hub_api.
|
||||||
id=id,
|
|
||||||
size=size,
|
|
||||||
os=os,
|
|
||||||
ssh_authorized_keys=list(map(lambda x: x["name"], posted_keys))
|
|
||||||
)
|
|
||||||
current_app.config["HUB_MODEL"].create(
|
current_app.config["HUB_MODEL"].create(
|
||||||
email = session["account"],
|
email = session["account"],
|
||||||
id=id,
|
id=id,
|
||||||
|
os=os,
|
||||||
|
size=size,
|
||||||
template_image_file_name=operating_systems[os]['template_image_file_name'],
|
template_image_file_name=operating_systems[os]['template_image_file_name'],
|
||||||
vcpus=vm_sizes[size]['vcpus'],
|
vcpus=vm_sizes[size]['vcpus'],
|
||||||
memory_mb=vm_sizes[size]['memory_mb'],
|
memory_mb=vm_sizes[size]['memory_mb'],
|
||||||
ssh_authorized_keys=list(map(lambda x: x["content"], posted_keys))
|
ssh_authorized_keys=list(map(lambda x: dict(name=x['name'], content=x['content']), posted_keys))
|
||||||
)
|
)
|
||||||
return id, errors
|
|
||||||
|
|
||||||
return None, errors
|
|
||||||
|
|
||||||
@bp.route("/create", methods=("GET", "POST"))
|
|
||||||
@account_required
|
|
||||||
def create():
|
|
||||||
vm_sizes = get_model().vm_sizes_dict()
|
|
||||||
operating_systems = get_model().operating_systems_dict()
|
|
||||||
public_keys_for_account = get_model().list_ssh_public_keys_for_account(session["account"])
|
|
||||||
account_balance = get_account_balance(get_vms(), get_payments(), datetime.utcnow())
|
|
||||||
capacity_avaliable = current_app.config["HUB_MODEL"].capacity_avaliable(512*1024*1024)
|
|
||||||
|
|
||||||
if request.method == "POST":
|
|
||||||
if "csrf-token" not in request.form or request.form['csrf-token'] != session['csrf-token']:
|
|
||||||
return abort(418, f"u want tea")
|
|
||||||
id, errors = _create(
|
|
||||||
vm_sizes,
|
|
||||||
operating_systems,
|
|
||||||
public_keys_for_account,
|
|
||||||
request.form)
|
|
||||||
if len(errors) == 0:
|
|
||||||
for error in errors:
|
|
||||||
flash(error)
|
|
||||||
return redirect(f"{url_for('console.index')}?created={id}")
|
return redirect(f"{url_for('console.index')}?created={id}")
|
||||||
|
|
||||||
affordable_vm_sizes = dict()
|
affordable_vm_sizes = dict()
|
||||||
@ -286,6 +268,9 @@ def create():
|
|||||||
if vm_size["dollars_per_month"] <= account_balance+0.25:
|
if vm_size["dollars_per_month"] <= account_balance+0.25:
|
||||||
affordable_vm_sizes[key] = vm_size
|
affordable_vm_sizes[key] = vm_size
|
||||||
|
|
||||||
|
for error in errors:
|
||||||
|
flash(error)
|
||||||
|
|
||||||
if not capacity_avaliable:
|
if not capacity_avaliable:
|
||||||
current_app.logger.warning(f"when capsul capacity is restored, send an email to {session['account']}")
|
current_app.logger.warning(f"when capsul capacity is restored, send an email to {session['account']}")
|
||||||
|
|
||||||
@ -302,25 +287,23 @@ def create():
|
|||||||
vm_sizes=affordable_vm_sizes
|
vm_sizes=affordable_vm_sizes
|
||||||
)
|
)
|
||||||
|
|
||||||
@bp.route("/keys", methods=("GET", "POST"))
|
@bp.route("/ssh", methods=("GET", "POST"))
|
||||||
@account_required
|
@account_required
|
||||||
def ssh_api_keys():
|
def ssh_public_keys():
|
||||||
errors = list()
|
errors = list()
|
||||||
|
|
||||||
token = None
|
|
||||||
|
|
||||||
if request.method == "POST":
|
if request.method == "POST":
|
||||||
if "csrf-token" not in request.form or request.form['csrf-token'] != session['csrf-token']:
|
if "csrf-token" not in request.form or request.form['csrf-token'] != session['csrf-token']:
|
||||||
return abort(418, f"u want tea")
|
return abort(418, f"u want tea")
|
||||||
|
|
||||||
action = request.form["action"]
|
method = request.form["method"]
|
||||||
|
|
||||||
if action == 'upload_ssh_key':
|
|
||||||
content = None
|
content = None
|
||||||
|
if method == "POST":
|
||||||
content = request.form["content"].replace("\r", " ").replace("\n", " ").strip()
|
content = request.form["content"].replace("\r", " ").replace("\n", " ").strip()
|
||||||
|
|
||||||
name = request.form["name"]
|
name = request.form["name"]
|
||||||
if not name or len(name.strip()) < 1:
|
if not name or len(name.strip()) < 1:
|
||||||
|
if method == "POST":
|
||||||
parts = re.split(" +", content)
|
parts = re.split(" +", content)
|
||||||
if len(parts) > 2 and len(parts[2].strip()) > 0:
|
if len(parts) > 2 and len(parts[2].strip()) > 0:
|
||||||
name = parts[2].strip()
|
name = parts[2].strip()
|
||||||
@ -331,6 +314,7 @@ def ssh_api_keys():
|
|||||||
if not re.match(r"^[0-9A-Za-z_@:. -]+$", name):
|
if not re.match(r"^[0-9A-Za-z_@:. -]+$", name):
|
||||||
errors.append(f"Key name '{name}' must match \"^[0-9A-Za-z_@:. -]+$\"")
|
errors.append(f"Key name '{name}' must match \"^[0-9A-Za-z_@:. -]+$\"")
|
||||||
|
|
||||||
|
if method == "POST":
|
||||||
if not content or len(content.strip()) < 1:
|
if not content or len(content.strip()) < 1:
|
||||||
errors.append("Content is required")
|
errors.append("Content is required")
|
||||||
else:
|
else:
|
||||||
@ -343,36 +327,24 @@ def ssh_api_keys():
|
|||||||
if len(errors) == 0:
|
if len(errors) == 0:
|
||||||
get_model().create_ssh_public_key(session["account"], name, content)
|
get_model().create_ssh_public_key(session["account"], name, content)
|
||||||
|
|
||||||
elif action == "delete_ssh_key":
|
elif method == "DELETE":
|
||||||
|
|
||||||
|
if len(errors) == 0:
|
||||||
get_model().delete_ssh_public_key(session["account"], name)
|
get_model().delete_ssh_public_key(session["account"], name)
|
||||||
|
|
||||||
elif action == "generate_api_token":
|
|
||||||
name = request.form["name"]
|
|
||||||
if name == '':
|
|
||||||
name = datetime.utcnow().strftime('%y-%m-%d %H:%M:%S')
|
|
||||||
token = b64encode(
|
|
||||||
get_model().generate_api_token(session["account"], name).encode('utf-8')
|
|
||||||
).decode('utf-8')
|
|
||||||
|
|
||||||
elif action == "delete_api_token":
|
|
||||||
get_model().delete_api_token(session["account"], request.form["id"])
|
|
||||||
|
|
||||||
for error in errors:
|
for error in errors:
|
||||||
flash(error)
|
flash(error)
|
||||||
|
|
||||||
ssh_keys_list=list(map(
|
keys_list=list(map(
|
||||||
lambda x: dict(name=x['name'], content=f"{x['content'][:20]}...{x['content'][len(x['content'])-20:]}"),
|
lambda x: dict(name=x['name'], content=f"{x['content'][:20]}...{x['content'][len(x['content'])-20:]}"),
|
||||||
get_model().list_ssh_public_keys_for_account(session["account"])
|
get_model().list_ssh_public_keys_for_account(session["account"])
|
||||||
))
|
))
|
||||||
|
|
||||||
api_tokens_list = get_model().list_api_tokens(session["account"])
|
|
||||||
|
|
||||||
return render_template(
|
return render_template(
|
||||||
"keys.html",
|
"ssh-public-keys.html",
|
||||||
csrf_token = session["csrf-token"],
|
csrf_token = session["csrf-token"],
|
||||||
api_tokens=api_tokens_list,
|
ssh_public_keys=keys_list,
|
||||||
ssh_public_keys=ssh_keys_list,
|
has_ssh_public_keys=len(keys_list) > 0
|
||||||
generated_api_token=token,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def get_vms():
|
def get_vms():
|
||||||
@ -396,6 +368,7 @@ def get_vm_months_float(vm, as_of):
|
|||||||
return days / average_number_of_days_in_a_month
|
return days / average_number_of_days_in_a_month
|
||||||
|
|
||||||
def get_account_balance(vms, payments, as_of):
|
def get_account_balance(vms, payments, as_of):
|
||||||
|
|
||||||
vm_cost_dollars = 0.0
|
vm_cost_dollars = 0.0
|
||||||
for vm in vms:
|
for vm in vms:
|
||||||
vm_months = get_vm_months_float(vm, as_of)
|
vm_months = get_vm_months_float(vm, as_of)
|
||||||
@ -408,6 +381,7 @@ def get_account_balance(vms, payments, as_of):
|
|||||||
@bp.route("/account-balance")
|
@bp.route("/account-balance")
|
||||||
@account_required
|
@account_required
|
||||||
def account_balance():
|
def account_balance():
|
||||||
|
|
||||||
payment_sessions = get_model().list_payment_sessions_for_account(session['account'])
|
payment_sessions = get_model().list_payment_sessions_for_account(session['account'])
|
||||||
for payment_session in payment_sessions:
|
for payment_session in payment_sessions:
|
||||||
if payment_session['type'] == 'btcpay':
|
if payment_session['type'] == 'btcpay':
|
||||||
|
@ -33,7 +33,7 @@ def init_app(app, is_running_server):
|
|||||||
result = re.search(r"^\d+_(up|down)", filename)
|
result = re.search(r"^\d+_(up|down)", filename)
|
||||||
if not result:
|
if not result:
|
||||||
app.logger.error(f"schemaVersion {filename} must match ^\\d+_(up|down). exiting.")
|
app.logger.error(f"schemaVersion {filename} must match ^\\d+_(up|down). exiting.")
|
||||||
continue
|
exit(1)
|
||||||
key = result.group()
|
key = result.group()
|
||||||
with open(join(schemaMigrationsPath, filename), 'rb') as file:
|
with open(join(schemaMigrationsPath, filename), 'rb') as file:
|
||||||
schemaMigrations[key] = file.read().decode("utf8")
|
schemaMigrations[key] = file.read().decode("utf8")
|
||||||
@ -128,3 +128,4 @@ def close_db(e=None):
|
|||||||
if db_model is not None:
|
if db_model is not None:
|
||||||
db_model.cursor.close()
|
db_model.cursor.close()
|
||||||
current_app.config['PSYCOPG2_CONNECTION_POOL'].putconn(db_model.connection)
|
current_app.config['PSYCOPG2_CONNECTION_POOL'].putconn(db_model.connection)
|
||||||
|
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
|
|
||||||
# I was never able to get this type hinting to work correctly
|
# I was never able to get this type hinting to work correctly
|
||||||
# from psycopg2.extensions import connection as Psycopg2Connection, cursor as Psycopg2Cursor
|
# from psycopg2.extensions import connection as Psycopg2Connection, cursor as Psycopg2Cursor
|
||||||
import hashlib
|
|
||||||
from nanoid import generate
|
from nanoid import generate
|
||||||
from flask import current_app
|
from flask import current_app
|
||||||
from typing import List
|
from typing import List
|
||||||
@ -17,6 +17,7 @@ class DBModel:
|
|||||||
self.cursor = cursor
|
self.cursor = cursor
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# ------ LOGIN ---------
|
# ------ LOGIN ---------
|
||||||
|
|
||||||
|
|
||||||
@ -43,16 +44,6 @@ class DBModel:
|
|||||||
|
|
||||||
return (token, ignoreCaseMatches)
|
return (token, ignoreCaseMatches)
|
||||||
|
|
||||||
def authenticate_token(self, token):
|
|
||||||
m = hashlib.md5()
|
|
||||||
m.update(token.encode('utf-8'))
|
|
||||||
hash_token = m.hexdigest()
|
|
||||||
self.cursor.execute("SELECT email FROM api_tokens WHERE token = %s", (hash_token, ))
|
|
||||||
result = self.cursor.fetchall()
|
|
||||||
if len(result) == 1:
|
|
||||||
return result[0]
|
|
||||||
return None
|
|
||||||
|
|
||||||
def consume_token(self, token):
|
def consume_token(self, token):
|
||||||
self.cursor.execute("SELECT email FROM login_tokens WHERE token = %s and created > (NOW() - INTERVAL '20 min')", (token, ))
|
self.cursor.execute("SELECT email FROM login_tokens WHERE token = %s and created > (NOW() - INTERVAL '20 min')", (token, ))
|
||||||
row = self.cursor.fetchone()
|
row = self.cursor.fetchone()
|
||||||
@ -141,32 +132,6 @@ class DBModel:
|
|||||||
self.cursor.execute( "DELETE FROM ssh_public_keys where email = %s AND name = %s", (email, name) )
|
self.cursor.execute( "DELETE FROM ssh_public_keys where email = %s AND name = %s", (email, name) )
|
||||||
self.connection.commit()
|
self.connection.commit()
|
||||||
|
|
||||||
def list_api_tokens(self, email):
|
|
||||||
self.cursor.execute(
|
|
||||||
"SELECT id, token, name, created FROM api_tokens WHERE email = %s",
|
|
||||||
(email, )
|
|
||||||
)
|
|
||||||
return list(map(
|
|
||||||
lambda x: dict(id=x[0], token=x[1], name=x[2], created=x[3]),
|
|
||||||
self.cursor.fetchall()
|
|
||||||
))
|
|
||||||
|
|
||||||
def generate_api_token(self, email, name):
|
|
||||||
token = generate()
|
|
||||||
m = hashlib.md5()
|
|
||||||
m.update(token.encode('utf-8'))
|
|
||||||
hash_token = m.hexdigest()
|
|
||||||
self.cursor.execute(
|
|
||||||
"INSERT INTO api_tokens (email, name, token) VALUES (%s, %s, %s)",
|
|
||||||
(email, name, hash_token)
|
|
||||||
)
|
|
||||||
self.connection.commit()
|
|
||||||
return token
|
|
||||||
|
|
||||||
def delete_api_token(self, email, id_):
|
|
||||||
self.cursor.execute( "DELETE FROM api_tokens where email = %s AND id = %s", (email, id_))
|
|
||||||
self.connection.commit()
|
|
||||||
|
|
||||||
def list_vms_for_account(self, email):
|
def list_vms_for_account(self, email):
|
||||||
self.cursor.execute("""
|
self.cursor.execute("""
|
||||||
SELECT vms.id, vms.public_ipv4, vms.public_ipv6, vms.size, vms.os, vms.created, vms.deleted, vm_sizes.dollars_per_month
|
SELECT vms.id, vms.public_ipv4, vms.public_ipv6, vms.size, vms.os, vms.created, vms.deleted, vm_sizes.dollars_per_month
|
||||||
@ -514,3 +479,8 @@ class DBModel:
|
|||||||
#cursor.close()
|
#cursor.close()
|
||||||
|
|
||||||
return to_return
|
return to_return
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -160,7 +160,10 @@ def can_claim_create(payload, host_id) -> (str, str):
|
|||||||
if allocated_network_name is None or allocated_ipv4_address is None:
|
if allocated_network_name is None or allocated_ipv4_address is None:
|
||||||
return "", f"host \"{host_id}\" does not have any avaliable IP addresses on any of its networks."
|
return "", f"host \"{host_id}\" does not have any avaliable IP addresses on any of its networks."
|
||||||
|
|
||||||
payload["network_name"] = allocated_network_name
|
# payload["network_name"] = allocated_network_name
|
||||||
|
# hard-code the network name for now until we can fix the phantom dhcp lease issues.
|
||||||
|
|
||||||
|
payload["network_name"] = 'public3'
|
||||||
payload["public_ipv4"] = allocated_ipv4_address
|
payload["public_ipv4"] = allocated_ipv4_address
|
||||||
|
|
||||||
return payload, ""
|
return payload, ""
|
||||||
|
@ -215,11 +215,12 @@ class CapsulFlaskHub(VirtualizationInterface):
|
|||||||
# no need to do anything here since if it cant be parsed then generic_operation will handle it.
|
# no need to do anything here since if it cant be parsed then generic_operation will handle it.
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
if error_message != "":
|
||||||
|
raise ValueError(f"create capsul operation {operation_id} on {assigned_hosts} failed with {error_message}")
|
||||||
|
|
||||||
if number_of_assigned != 1:
|
if number_of_assigned != 1:
|
||||||
assigned_hosts_string = ", ".join(assigned_hosts)
|
assigned_hosts_string = ", ".join(assigned_hosts)
|
||||||
raise ValueError(f"expected create capsul operation {operation_id} to be assigned to one host, it was assigned to {number_of_assigned} ({assigned_hosts_string})")
|
raise ValueError(f"expected create capsul operation {operation_id} to be assigned to one host, it was assigned to {number_of_assigned} ({assigned_hosts_string})")
|
||||||
if error_message != "":
|
|
||||||
raise ValueError(f"create capsul operation {operation_id} on {assigned_hosts_string} failed with {error_message}")
|
|
||||||
|
|
||||||
|
|
||||||
def destroy(self, email: str, id: str):
|
def destroy(self, email: str, id: str):
|
||||||
|
@ -48,6 +48,10 @@ def validate_dollars():
|
|||||||
def btcpay_payment():
|
def btcpay_payment():
|
||||||
errors = list()
|
errors = list()
|
||||||
|
|
||||||
|
if current_app.config['BTCPAY_PRIVATE_KEY'] == "":
|
||||||
|
flash("BTCPay is not enabled on this server")
|
||||||
|
return redirect(url_for("console.account_balance"))
|
||||||
|
|
||||||
if request.method == "POST":
|
if request.method == "POST":
|
||||||
result = validate_dollars()
|
result = validate_dollars()
|
||||||
errors = result[0]
|
errors = result[0]
|
||||||
|
@ -1,40 +0,0 @@
|
|||||||
import datetime
|
|
||||||
|
|
||||||
from flask import Blueprint
|
|
||||||
from flask import current_app
|
|
||||||
from flask import jsonify
|
|
||||||
from flask import request
|
|
||||||
from flask import session
|
|
||||||
from nanoid import generate
|
|
||||||
|
|
||||||
from capsulflask.auth import account_required
|
|
||||||
from capsulflask.db import get_model
|
|
||||||
|
|
||||||
bp = Blueprint("publicapi", __name__, url_prefix="/api")
|
|
||||||
|
|
||||||
@bp.route("/capsul/create", methods=["POST"])
|
|
||||||
@account_required
|
|
||||||
def capsul_create():
|
|
||||||
email = session["account"]
|
|
||||||
|
|
||||||
from .console import _create,get_account_balance, get_payments, get_vms
|
|
||||||
|
|
||||||
vm_sizes = get_model().vm_sizes_dict()
|
|
||||||
operating_systems = get_model().operating_systems_dict()
|
|
||||||
public_keys_for_account = get_model().list_ssh_public_keys_for_account(session["account"])
|
|
||||||
account_balance = get_account_balance(get_vms(), get_payments(), datetime.datetime.utcnow())
|
|
||||||
capacity_avaliable = current_app.config["HUB_MODEL"].capacity_avaliable(512*1024*1024)
|
|
||||||
|
|
||||||
request.json['ssh_authorized_key_count'] = 1
|
|
||||||
|
|
||||||
id, errors = _create(
|
|
||||||
vm_sizes,
|
|
||||||
operating_systems,
|
|
||||||
public_keys_for_account,
|
|
||||||
request.json)
|
|
||||||
|
|
||||||
if id is not None:
|
|
||||||
return jsonify(
|
|
||||||
id=id,
|
|
||||||
)
|
|
||||||
return jsonify(errors=errors)
|
|
@ -1,2 +0,0 @@
|
|||||||
DROP TABLE api_keys;
|
|
||||||
UPDATE schemaversion SET version = 16;
|
|
@ -1,9 +0,0 @@
|
|||||||
CREATE TABLE api_tokens (
|
|
||||||
id SERIAL PRIMARY KEY,
|
|
||||||
email TEXT REFERENCES accounts(email) ON DELETE RESTRICT,
|
|
||||||
name TEXT NOT NULL,
|
|
||||||
created TIMESTAMP NOT NULL DEFAULT NOW(),
|
|
||||||
token TEXT NOT NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
UPDATE schemaversion SET version = 17;
|
|
@ -1,2 +0,0 @@
|
|||||||
DROP TABLE api_keys;
|
|
||||||
UPDATE schemaversion SET version = 16;
|
|
@ -1,9 +0,0 @@
|
|||||||
CREATE TABLE api_tokens (
|
|
||||||
id SERIAL PRIMARY KEY,
|
|
||||||
email TEXT REFERENCES accounts(email) ON DELETE RESTRICT,
|
|
||||||
name TEXT NOT NULL,
|
|
||||||
created TIMESTAMP NOT NULL DEFAULT NOW(),
|
|
||||||
token TEXT NOT NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
UPDATE schemaversion SET version = 17;
|
|
@ -3,7 +3,7 @@
|
|||||||
# check available RAM and IPv4s
|
# check available RAM and IPv4s
|
||||||
|
|
||||||
ram_bytes_to_allocate="$1"
|
ram_bytes_to_allocate="$1"
|
||||||
ram_bytes_available=$(grep -E "^(size|memory_available_bytes)" /proc/spl/kstat/zfs/arcstats | awk '{sum+=$3} END {printf "%.0f", sum}')
|
ram_bytes_available="$(($(grep Available /proc/meminfo | grep -o '[0-9]*') * 1024))"
|
||||||
ram_bytes_remainder="$((ram_bytes_available - ram_bytes_to_allocate))"
|
ram_bytes_remainder="$((ram_bytes_available - ram_bytes_to_allocate))"
|
||||||
|
|
||||||
if echo "$ram_bytes_to_allocate" | grep -vqE "^[0-9]+$"; then
|
if echo "$ram_bytes_to_allocate" | grep -vqE "^[0-9]+$"; then
|
||||||
@ -11,8 +11,8 @@ if echo "$ram_bytes_to_allocate" | grep -vqE "^[0-9]+$"; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# 20GB
|
# 0.25GB
|
||||||
if [ "$ram_bytes_remainder" -le $((20 * 1024 * 1024 * 1024)) ]; then
|
if [ "$ram_bytes_remainder" -le $((1 * 1024 * 1024 * 1024 / 4)) ]; then
|
||||||
echo "VM is requesting more RAM than $(hostname -f) has available."
|
echo "VM is requesting more RAM than $(hostname -f) has available."
|
||||||
echo "Bytes requested: $ram_bytes_to_allocate"
|
echo "Bytes requested: $ram_bytes_to_allocate"
|
||||||
echo "Bytes available: $ram_bytes_available"
|
echo "Bytes available: $ram_bytes_available"
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
|
|
||||||
vmname="$1"
|
vmname="$1"
|
||||||
template_file="/tank/img/$2"
|
template_file="/tank/img/$2"
|
||||||
|
qemu_tank_dir="/tank"
|
||||||
vcpus="$3"
|
vcpus="$3"
|
||||||
memory="$4"
|
memory="$4"
|
||||||
pubkeys="$5"
|
pubkeys="$5"
|
||||||
@ -50,40 +51,40 @@ if echo "$public_ipv4" | grep -vqE "^[0-9.]+$"; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
disk="/tank/vm/$vmname.qcow2"
|
disk="$vmname.qcow2"
|
||||||
cdrom="/tank/vm/$vmname.iso"
|
cdrom="$vmname.iso"
|
||||||
xml="/tank/vm/$vmname.xml"
|
xml="$vmname.xml"
|
||||||
|
|
||||||
if [ -f /tank/vm/$vmname.qcow2 ]; then
|
if [ -f /tank/vm/$vmname.qcow2 ]; then
|
||||||
echo "Randomly generated name matched an existing VM! Odds are like one in a billion. Buy a lotto ticket."
|
echo "Randomly generated name matched an existing VM! Odds are like one in a billion. Buy a lotto ticket."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
cp "$template_file" "$disk"
|
cp "$template_file" "/tank/vm/$disk"
|
||||||
cp /tank/config/cyberia-cloudinit.yml /tmp/cloudinit.yml
|
cp /tank/config/cyberia-cloudinit.yml /tmp/cloudinit.yml
|
||||||
echo "$pubkeys" | while IFS= read -r line; do
|
echo "$pubkeys" | while IFS= read -r line; do
|
||||||
echo " - $line" >> /tmp/cloudinit.yml
|
echo " - $line" >> /tmp/cloudinit.yml
|
||||||
done
|
done
|
||||||
|
|
||||||
cloud-localds "$cdrom" /tmp/cloudinit.yml
|
cloud-localds "/tank/vm/$cdrom" /tmp/cloudinit.yml
|
||||||
|
|
||||||
qemu-img resize "$disk" "$root_volume_size"
|
qemu-img resize "/tank/vm/$disk" "$root_volume_size"
|
||||||
virt-install \
|
virt-install \
|
||||||
--memory "$memory" \
|
--memory "$memory" \
|
||||||
--vcpus "$vcpus" \
|
--vcpus "$vcpus" \
|
||||||
--name "$vmname" \
|
--name "$vmname" \
|
||||||
--disk "$disk",bus=virtio \
|
--disk "$qemu_tank_dir/vm/$disk",bus=virtio \
|
||||||
--disk "$cdrom",device=cdrom \
|
--disk "$qemu_tank_dir/vm/$cdrom",device=cdrom \
|
||||||
--os-type Linux \
|
--os-type Linux \
|
||||||
--os-variant generic \
|
--os-variant generic \
|
||||||
--virt-type kvm \
|
--virt-type kvm \
|
||||||
--graphics vnc,listen=127.0.0.1 \
|
--graphics vnc,listen=127.0.0.1 \
|
||||||
--network network=$network_name,filterref=clean-traffic,model=virtio \
|
--network network=$network_name,model=virtio \
|
||||||
--import \
|
--import \
|
||||||
--print-xml > "$xml"
|
--print-xml > "/tank/vm/$xml"
|
||||||
|
|
||||||
chmod 0600 "$xml" "$disk" "$cdrom"
|
chmod 0600 "/tank/vm/$xml" "/tank/vm/$disk" "/tank/vm/$cdrom"
|
||||||
virsh define "$xml"
|
virsh define "/tank/vm/$xml"
|
||||||
virsh start "$vmname"
|
virsh start "$vmname"
|
||||||
|
|
||||||
echo "success"
|
echo "success"
|
||||||
|
@ -46,7 +46,9 @@
|
|||||||
<a href="/payment/stripe">Add funds with Credit/Debit (stripe)</a>
|
<a href="/payment/stripe">Add funds with Credit/Debit (stripe)</a>
|
||||||
<ul><li>notice: stripe will load nonfree javascript </li></ul>
|
<ul><li>notice: stripe will load nonfree javascript </li></ul>
|
||||||
</li>
|
</li>
|
||||||
|
{% if config['BTCPAY_PRIVATE_KEY'] != "" %}
|
||||||
<li><a href="/payment/btcpay">Add funds with Bitcoin/Litecoin/Monero (btcpay)</a></li>
|
<li><a href="/payment/btcpay">Add funds with Bitcoin/Litecoin/Monero (btcpay)</a></li>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
<li>Cash: email <a href="mailto:treasurer@cyberia.club">treasurer@cyberia.club</a></li>
|
<li>Cash: email <a href="mailto:treasurer@cyberia.club">treasurer@cyberia.club</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
<nav>
|
<nav>
|
||||||
<div class="row justify-space-between half-margin">
|
<div class="row justify-space-between half-margin">
|
||||||
<div>
|
<div>
|
||||||
<a href="/"><b>Capsul</b></a>💊
|
<a href="/"><b>YOLOCOLO</b></a>💊
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
|
|
||||||
@ -27,11 +27,10 @@
|
|||||||
<div class="row justify-center half-margin wrap nav-links">
|
<div class="row justify-center half-margin wrap nav-links">
|
||||||
<a href="/pricing">Pricing</a>
|
<a href="/pricing">Pricing</a>
|
||||||
<a href="/faq">FAQ</a>
|
<a href="/faq">FAQ</a>
|
||||||
<a href="/changelog">Changelog</a>
|
|
||||||
|
|
||||||
{% if session["account"] %}
|
{% if session["account"] %}
|
||||||
<a href="/console">Capsuls</a>
|
<a href="/console">Capsuls</a>
|
||||||
<a href="/console/keys">SSH & API Keys</a>
|
<a href="/console/ssh">SSH Public Keys</a>
|
||||||
<a href="/console/account-balance">Account Balance</a>
|
<a href="/console/account-balance">Account Balance</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
@ -47,11 +46,12 @@
|
|||||||
</main>
|
</main>
|
||||||
{% block subcontent %}{% endblock %}
|
{% block subcontent %}{% endblock %}
|
||||||
<footer>
|
<footer>
|
||||||
(c) Attribution-ShareAlike 4.0 International <br/>
|
This server runs <a
|
||||||
A service by Cyberia Computer Club 2020-<span class="bigtext">∞</span> <br/>
|
href="https://giit.cyberia.club/~forest/capsul-flask">capsul-flask</a> by
|
||||||
<br/>
|
Cyberia Computer Club, available under the <a
|
||||||
<br/>
|
href="https://creativecommons.org/licenses/by-sa/4.0/">Attribution-ShareAlike
|
||||||
<a href="https://giit.cyberia.club/~forest/capsul-flask/tree/master/capsulflask{% block pagesource %}{% endblock %}">View page source</a>
|
4.0 International</a> licence.<br/><br/>
|
||||||
|
<a href="https://git.autonomic.zone/3wordchant/capsul-flask/src/branch/yolocolo/capsulflask{% block pagesource %}{% endblock %}">View page source</a>
|
||||||
</footer>
|
</footer>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -101,7 +101,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="row justify-start">
|
<div class="row justify-start">
|
||||||
<label class="align" for="ssh_authorized_keys">SSH Authorized Keys</label>
|
<label class="align" for="ssh_authorized_keys">SSH Authorized Keys</label>
|
||||||
<a id="ssh_authorized_keys" href="/console/keys">{{ vm['ssh_authorized_keys'] }}</a>
|
<a id="ssh_authorized_keys" href="/console/ssh">{{ vm['ssh_authorized_keys'] }}</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -31,7 +31,7 @@
|
|||||||
<p>(At least one month of funding is required)</p>
|
<p>(At least one month of funding is required)</p>
|
||||||
{% elif no_ssh_public_keys %}
|
{% elif no_ssh_public_keys %}
|
||||||
<p>You don't have any ssh public keys yet.</p>
|
<p>You don't have any ssh public keys yet.</p>
|
||||||
<p>You must <a href="/console/keys">upload one</a> before you can create a Capsul.</p>
|
<p>You must <a href="/console/ssh">upload one</a> before you can create a Capsul.</p>
|
||||||
{% elif not capacity_avaliable %}
|
{% elif not capacity_avaliable %}
|
||||||
<p>Host(s) at capacity. No capsuls can be created at this time. sorry. </p>
|
<p>Host(s) at capacity. No capsuls can be created at this time. sorry. </p>
|
||||||
{% else %}
|
{% else %}
|
||||||
|
@ -10,81 +10,32 @@
|
|||||||
<p>
|
<p>
|
||||||
<ul>
|
<ul>
|
||||||
<li>
|
<li>
|
||||||
Which instance type should I buy?
|
What is this?
|
||||||
<p>There are no hard rules for this sort of thing, but here are some guidelines:</p>
|
<p>
|
||||||
<p>f1-xs: blog, vpn, bot, cgit</p>
|
This is a <strong>technical demo</strong> of <a
|
||||||
<p>f1-s: a bot, owncloud, gitea, popular blog</p>
|
href="https://giit.cyberia.club/~forest/capsul-flask">Capsul</a>, for the
|
||||||
<p>f1-m: docker host, build system</p>
|
as-yet-untitled <a href="https://coops.tech">Cotech</a> server hosting
|
||||||
<p>f1-l: large webservice, rotund java app</p>
|
initiative, which you can <a
|
||||||
<p>f1-x: gitlab (wow such memory very devops</p>
|
href="https://community.coops.tech/t/call-for-input-v2-co-op-vps-survey/2802/9">read
|
||||||
<p>f1-xx: something gargantuan</p>
|
about on the Cotech forum</a>.
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
How do I log in?
|
|
||||||
<p>ssh to the ip provided to you using the cyberian user.</p>
|
|
||||||
<pre class='code'>$ ssh cyberian@1.2.3.4</pre>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
How do I change to the root user?
|
|
||||||
<p>The cyberian user has passwordless sudo access by default. This should work:</p>
|
|
||||||
<pre class='code'>
|
|
||||||
# Linux
|
|
||||||
$ sudo su -
|
|
||||||
|
|
||||||
# OpenBSD
|
|
||||||
$ doas su -</pre>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Do you offer reverse DNS?
|
|
||||||
<p>We do, but right now it's a manual process. Shoot us an email and we'll get it done.</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
What if I don't pay / don't maintain my payments?
|
|
||||||
<p>Your VM will eventually be deleted.
|
|
||||||
Capsul will send you a few inoffensive reminders as that termination date approaches.
|
|
||||||
</p>
|
</p>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
Besides my virtual machines and payments, what information do you keep about me?
|
What do you mean, "technical demo"?
|
||||||
<p>We associate an email address with every VM so that we can track payment and respond to support requests.</p>
|
<p>No backups</p>
|
||||||
<p>If you pay with a credit card, Stripe stores some additional details about you that we literally cannot delete.</p>
|
<p>No service level agreement</p>
|
||||||
|
<p>"Best effort" support</p>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
What can I do with my VM?
|
Where can I get this, but, more reliable?
|
||||||
<p>Make it into a mailserver, a tor relay, a VPN host, whatever you'd like - we do have one small request, though.</p>
|
<p>Cyberia, the authors of this platform, run the canonical instance, <a
|
||||||
<p>Crypto mining on capsul is currently considered obnoxious behavior, because the hashrates on our CPUs is so low and because mining crypto consumes entire processor cores that could have otherwise been shared between many dozens of other users.</p>
|
href="https://capsul.org">Capsul.org</a>, on hardware they own. Please
|
||||||
<p>In the future, if we have plentiful CPU resources, we may come out with a tier more suitable for mining - maybe a high cpu tier or similar, where each VM gets a full dedicated core and sharing them is not anticipated.</p>
|
send them your money! (cash, crypto, or card accepted).</p>
|
||||||
<p>We will never snoop on your traffic or inspect what's going on inside of our customer virtual machines - we don't want to. We hope that you'll extend us a similar courtesy and try not to use too much of our shared CPU resources. Capsul is currently a shared (resource-wise) world, and we all must live in it together!</p>
|
|
||||||
<p>Also, mandatory: our systems exist within the USA, and as such those systems are bound by US law.</p>
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
Can you recover my passwords/insert new keys?
|
How do I use this system?
|
||||||
<p>Can we? Technically yes. Will we? No, never. It would violate the trust that our users have in us.
|
<p>Please see <a href="https://capsul.org/faq">the official Capsul FAQ
|
||||||
We have no interest in touching client VMs after they're running.
|
page</a>.</p>
|
||||||
We promise to keep your machines running smoothly.
|
|
||||||
If you lose access to your VM, that's on you.</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Do you offer refunds?
|
|
||||||
<p>Not now, but email us and we can probably figure something out.</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Where do the VMs run? Is it on a machine that you guys own/control?
|
|
||||||
<p>Capsul runs on a server named Baikal which Cyberia built from scratch & mailed to a datacenter
|
|
||||||
in Georgia called CyberWurx. CyberWurx staff installed it for us in a rack space that
|
|
||||||
Cyberia pays for. </p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Do you offer support?
|
|
||||||
<p>Yep, see <a href="/support">our support page</a>.</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Do you have an SLA?
|
|
||||||
<p>No, but we normally respond pretty quickly.</p>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
Will you implement feature X?
|
|
||||||
<p>Maybe! Email <a href="mailto:ops@cyberia.club">ops@cyberia.club</a> and ask us about it.</p>
|
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</p>
|
</p>
|
||||||
|
@ -1,31 +1,26 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1>CAPSUL</h1>
|
<h1>
|
||||||
<pre>
|
<pre>
|
||||||
.-.
|
_ _
|
||||||
/:::\
|
_ _ ___ | | ___ ___ ___ | | ___
|
||||||
/::::/
|
| | | |/ _ \| |/ _ \ / __/ _ \| |/ _ \
|
||||||
/ `-:/
|
| |_| | (_) | | (_) | (_| (_) | | (_) |
|
||||||
/ /
|
\__, |\___/|_|\___/ \___\___/|_|\___/
|
||||||
\ /
|
|___/
|
||||||
`"`
|
|
||||||
</pre>
|
</pre>
|
||||||
<span>Simple, fast, private compute by <a href="https://cyberia.club">cyberia.club</a></span>
|
<span>Co-operative hosting using <a href="https://cyberia.club">Cyberia</a>'s Capsul</span>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block subcontent %}
|
{% block subcontent %}
|
||||||
<p>
|
<p>
|
||||||
<ul>
|
<ul>
|
||||||
<li>Low friction: simply log in with your email address and fund your account with Credit/Debit or Cryptocurrency</li>
|
<li>Sign up for an account!</li>
|
||||||
<li>All root disks are backed up at no charge</li>
|
<li>Add some funds!</li>
|
||||||
<li>All storage is fast, local, and solid-state</li>
|
<li>Create a VPS!</li>
|
||||||
<li>All network connections are low latency</li>
|
<li>Give your feedback!</li>
|
||||||
<li>Supported by amazing volunteers from Cyberia</li>
|
|
||||||
<li>Upfront prices, no confusing billing</li>
|
|
||||||
<li>Operated by a Minnesota non-profit organization that will never exploit you</li>
|
|
||||||
<li>We donate a portion of our proceeds to likeminded hacker groups around the globe</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
</p>
|
</p>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,18 +1,17 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
{% block title %}SSH & API Keys{% endblock %}
|
{% block title %}SSH Public Keys{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="row third-margin">
|
<div class="row third-margin">
|
||||||
<h1>SSH PUBLIC KEYS</h1>
|
<h1>SSH PUBLIC KEYS</h1>
|
||||||
</div>
|
</div>
|
||||||
<div class="row third-margin"><div>
|
<div class="row third-margin"><div>
|
||||||
{% if ssh_public_keys|length > 0 %} <hr/> {% endif %}
|
{% if has_ssh_public_keys %} <hr/> {% endif %}
|
||||||
|
|
||||||
{% for ssh_public_key in ssh_public_keys %}
|
{% for ssh_public_key in ssh_public_keys %}
|
||||||
<form method="post">
|
<form method="post">
|
||||||
<input type="hidden" name="method" value="DELETE"></input>
|
<input type="hidden" name="method" value="DELETE"></input>
|
||||||
<input type="hidden" name="action" value="delete_ssh_key"></input>
|
|
||||||
<input type="hidden" name="name" value="{{ ssh_public_key['name'] }}"></input>
|
<input type="hidden" name="name" value="{{ ssh_public_key['name'] }}"></input>
|
||||||
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
@ -23,14 +22,13 @@
|
|||||||
</form>
|
</form>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
{% if ssh_public_keys|length > 0 %} <hr/> {% endif %}
|
{% if has_ssh_public_keys %} <hr/> {% endif %}
|
||||||
|
|
||||||
<div class="third-margin">
|
<div class="third-margin">
|
||||||
<h1>UPLOAD A NEW SSH PUBLIC KEY</h1>
|
<h1>UPLOAD A NEW SSH PUBLIC KEY</h1>
|
||||||
</div>
|
</div>
|
||||||
<form method="post">
|
<form method="post">
|
||||||
<input type="hidden" name="method" value="POST"></input>
|
<input type="hidden" name="method" value="POST"></input>
|
||||||
<input type="hidden" name="action" value="upload_ssh_key"></input>
|
|
||||||
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
||||||
<div class="row justify-start">
|
<div class="row justify-start">
|
||||||
<label class="align" for="content">File Contents</label>
|
<label class="align" for="content">File Contents</label>
|
||||||
@ -56,51 +54,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div></div>
|
</div></div>
|
||||||
<hr/>
|
|
||||||
<div class="row third-margin">
|
|
||||||
<h1>API KEYS</h1>
|
|
||||||
</div>
|
|
||||||
<div class="row third-margin"><div>
|
|
||||||
{% if generated_api_token %}
|
|
||||||
<hr/>
|
|
||||||
Generated key:
|
|
||||||
<span class="code">{{ generated_api_token }}</span>
|
|
||||||
{% endif %}
|
|
||||||
{% if api_tokens|length >0 %} <hr/>{% endif %}
|
|
||||||
{% for api_token in api_tokens %}
|
|
||||||
<form method="post">
|
|
||||||
<input type="hidden" name="method" value="DELETE"></input>
|
|
||||||
<input type="hidden" name="action" value="delete_api_token"></input>
|
|
||||||
<input type="hidden" name="id" value="{{ api_token['id'] }}"></input>
|
|
||||||
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
|
||||||
<div class="row">
|
|
||||||
<span class="code">{{ api_token['name'] }}</span>
|
|
||||||
created {{ api_token['created'].strftime("%b %d %Y") }}
|
|
||||||
<input type="submit" value="Delete">
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
{% endfor %}
|
|
||||||
{% if api_tokens|length >0 %} <hr/>{% endif %}
|
|
||||||
|
|
||||||
<div class="third-margin">
|
|
||||||
<h1>GENERATE A NEW API KEY</h1>
|
|
||||||
</div>
|
|
||||||
<form method="post">
|
|
||||||
<input type="hidden" name="method" value="POST"></input>
|
|
||||||
<input type="hidden" name="action" value="generate_api_token"></input>
|
|
||||||
<input type="hidden" name="csrf-token" value="{{ csrf_token }}"/>
|
|
||||||
<div class="smalltext">
|
|
||||||
<p>Generate a new API key, to integrate with other systems.</p>
|
|
||||||
</div>
|
|
||||||
<div class="row justify-start">
|
|
||||||
<label class="align" for="name">Key Name</label>
|
|
||||||
<input type="text" id="name" name="name"></input> (defaults to creation time)
|
|
||||||
</div>
|
|
||||||
<div class="row justify-end">
|
|
||||||
<input type="submit" value="Generate">
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div></div>
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block pagesource %}/templates/ssh-public-keys.html{% endblock %}
|
{% block pagesource %}/templates/ssh-public-keys.html{% endblock %}
|
@ -7,17 +7,26 @@ services:
|
|||||||
build: .
|
build: .
|
||||||
volumes:
|
volumes:
|
||||||
- "./:/app/code"
|
- "./:/app/code"
|
||||||
|
- "../tank:/tank"
|
||||||
|
- "/var/run/libvirt/libvirt-sock:/var/run/libvirt/libvirt-sock"
|
||||||
depends_on:
|
depends_on:
|
||||||
- db
|
- db
|
||||||
ports:
|
ports:
|
||||||
- "5000:5000"
|
- "5000:5000"
|
||||||
environment:
|
environment:
|
||||||
- "POSTGRES_CONNECTION_PARAMETERS=host=db port=5432 user=capsul password=capsul dbname=capsul"
|
- "POSTGRES_CONNECTION_PARAMETERS=host=db port=5432 user=capsul password=capsul dbname=capsul"
|
||||||
|
- SPOKE_MODEL=shell-scripts
|
||||||
|
#- FLASK_DEBUG=1
|
||||||
|
- BASE_URL=http://localhost:5000
|
||||||
|
- ADMIN_PANEL_ALLOW_EMAIL_ADDRESSES=3wc.capsul@doesthisthing.work
|
||||||
|
- VIRSH_DEFAULT_CONNECT_URI=qemu:///system
|
||||||
# The image uses gunicorn by default, let's override it with Flask's
|
# The image uses gunicorn by default, let's override it with Flask's
|
||||||
# built-in development server
|
# built-in development server
|
||||||
command: ["flask", "run", "-h", "0.0.0.0", "-p", "5000"]
|
command: ["flask", "run", "-h", "0.0.0.0", "-p", "5000"]
|
||||||
|
devices:
|
||||||
|
- "/dev/kvm:/dev/kvm"
|
||||||
db:
|
db:
|
||||||
image: "postgres:9.6.5"
|
image: "postgres:9.6.5-alpine"
|
||||||
volumes:
|
volumes:
|
||||||
- "postgres:/var/lib/postgresql/data"
|
- "postgres:/var/lib/postgresql/data"
|
||||||
environment:
|
environment:
|
||||||
|
Reference in New Issue
Block a user