mirror of
https://github.com/penpot/penpot.git
synced 2025-01-22 06:32:38 -05:00
239 lines
5.6 KiB
Python
Executable file
239 lines
5.6 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
#
|
|
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
#
|
|
# Copyright (c) KALEIDOS INC
|
|
|
|
import argparse
|
|
import json
|
|
import socket
|
|
import sys
|
|
|
|
from tabulate import tabulate
|
|
from getpass import getpass
|
|
from urllib.parse import urlparse
|
|
|
|
PREPL_URI = "tcp://localhost:6063"
|
|
|
|
def get_prepl_conninfo():
|
|
uri_data = urlparse(PREPL_URI)
|
|
if uri_data.scheme != "tcp":
|
|
raise RuntimeError(f"invalid PREPL_URI: {PREPL_URI}")
|
|
|
|
if not isinstance(uri_data.netloc, str):
|
|
raise RuntimeError(f"invalid PREPL_URI: {PREPL_URI}")
|
|
|
|
host, port = uri_data.netloc.split(":", 2)
|
|
|
|
if port is None:
|
|
port = 6063
|
|
|
|
if isinstance(port, str):
|
|
port = int(port)
|
|
|
|
return host, port
|
|
|
|
def send_eval(expr):
|
|
host, port = get_prepl_conninfo()
|
|
|
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
|
s.connect((host, port))
|
|
s.send(expr.encode("utf-8"))
|
|
s.send(b":repl/quit\n\n")
|
|
|
|
with s.makefile() as f:
|
|
while True:
|
|
line = f.readline()
|
|
result = json.loads(line)
|
|
tag = result.get("tag", None)
|
|
if tag == "ret":
|
|
return result.get("val", None), result.get("exception", None)
|
|
elif tag == "out":
|
|
print(result.get("val"), end="")
|
|
else:
|
|
raise RuntimeError("unexpected response from PREPL")
|
|
|
|
def encode(val):
|
|
return json.dumps(json.dumps(val))
|
|
|
|
def print_error(res):
|
|
for error in res["via"]:
|
|
print("ERR:", error["message"])
|
|
break
|
|
|
|
def run_cmd(params):
|
|
try:
|
|
expr = "(app.srepl.cli/exec {})".format(encode(params))
|
|
res, failed = send_eval(expr)
|
|
if failed:
|
|
print_error(res)
|
|
sys.exit(-1)
|
|
|
|
return res
|
|
except Exception as cause:
|
|
print("EXC:", str(cause))
|
|
sys.exit(-2)
|
|
|
|
def create_profile(fullname, email, password):
|
|
params = {
|
|
"cmd": "create-profile",
|
|
"params": {
|
|
"fullname": fullname,
|
|
"email": email,
|
|
"password": password
|
|
}
|
|
}
|
|
|
|
res = run_cmd(params)
|
|
print(f"Created: {res['email']} / {res['id']}")
|
|
|
|
def update_profile(email, fullname, password, is_active):
|
|
params = {
|
|
"cmd": "update-profile",
|
|
"params": {
|
|
"email": email,
|
|
"fullname": fullname,
|
|
"password": password,
|
|
"is_active": is_active
|
|
}
|
|
}
|
|
|
|
res = run_cmd(params)
|
|
if res is True:
|
|
print(f"Updated")
|
|
else:
|
|
print(f"No profile found with email {email}")
|
|
|
|
def delete_profile(email, soft):
|
|
params = {
|
|
"cmd": "delete-profile",
|
|
"params": {
|
|
"email": email,
|
|
"soft": soft
|
|
}
|
|
}
|
|
|
|
res = run_cmd(params)
|
|
if res is True:
|
|
print(f"Deleted")
|
|
else:
|
|
print(f"No profile found with email {email}")
|
|
|
|
def search_profile(email):
|
|
params = {
|
|
"cmd": "search-profile",
|
|
"params": {
|
|
"email": email,
|
|
}
|
|
}
|
|
|
|
res = run_cmd(params)
|
|
|
|
if isinstance(res, list):
|
|
print(tabulate(res, headers="keys"))
|
|
|
|
def derive_password(password):
|
|
params = {
|
|
"cmd": "derive-password",
|
|
"params": {
|
|
"password": password,
|
|
}
|
|
}
|
|
|
|
res = run_cmd(params)
|
|
print(f"Derived password: \"{res}\"")
|
|
|
|
|
|
def migrate_components_v2():
|
|
params = {
|
|
"cmd": "migrate-v2",
|
|
"params": {}
|
|
}
|
|
|
|
run_cmd(params)
|
|
|
|
available_commands = (
|
|
"create-profile",
|
|
"update-profile",
|
|
"delete-profile",
|
|
"search-profile",
|
|
"derive-password",
|
|
"migrate-components-v2",
|
|
)
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description=(
|
|
"Penpot Command Line Interface (CLI)"
|
|
)
|
|
)
|
|
|
|
parser.add_argument("-V", "--version", action="version", version="Penpot CLI %%develop%%")
|
|
parser.add_argument("action", action="store", choices=available_commands)
|
|
parser.add_argument("-f", "--force", help="force operation", action="store_true")
|
|
parser.add_argument("-n", "--fullname", help="fullname", action="store")
|
|
parser.add_argument("-e", "--email", help="email", action="store")
|
|
parser.add_argument("-p", "--password", help="password", action="store")
|
|
parser.add_argument("-c", "--connect", help="connect to PREPL", action="store", default="tcp://localhost:6063")
|
|
|
|
args = parser.parse_args()
|
|
|
|
PREPL_URI = args.connect
|
|
|
|
if args.action == "create-profile":
|
|
email = args.email
|
|
password = args.password
|
|
fullname = args.fullname
|
|
|
|
if email is None:
|
|
email = input("Email: ")
|
|
|
|
if fullname is None:
|
|
fullname = input("Fullname: ")
|
|
|
|
if password is None:
|
|
password = getpass("Password: ")
|
|
|
|
create_profile(fullname, email, password)
|
|
|
|
elif args.action == "update-profile":
|
|
email = args.email
|
|
password = args.password
|
|
|
|
if email is None:
|
|
email = input("Email: ")
|
|
|
|
if password is None:
|
|
password = getpass("Password: ")
|
|
|
|
update_profile(email, None, password, None)
|
|
|
|
elif args.action == "derive-password":
|
|
password = args.password
|
|
|
|
if password is None:
|
|
password = getpass("Password: ")
|
|
|
|
derive_password(password)
|
|
|
|
elif args.action == "delete-profile":
|
|
email = args.email
|
|
soft = not args.force
|
|
|
|
if email is None:
|
|
email = input("Email: ")
|
|
|
|
delete_profile(email, soft)
|
|
|
|
elif args.action == "search-profile":
|
|
email = args.email
|
|
if email is None:
|
|
email = input("Email: ")
|
|
|
|
search_profile(email)
|
|
|
|
elif args.action == "migrate-components-v2":
|
|
migrate_components_v2()
|
|
|
|
|