2021-11-23 09:35:16 +08:00
|
|
|
import errno
|
|
|
|
import os
|
|
|
|
import pickle
|
|
|
|
import sys
|
|
|
|
from argparse import ArgumentParser
|
|
|
|
from base64 import b64decode
|
|
|
|
from glob import glob
|
|
|
|
from google.auth.transport.requests import Request
|
|
|
|
from google_auth_oauthlib.flow import InstalledAppFlow
|
|
|
|
from googleapiclient.discovery import build
|
|
|
|
from googleapiclient.errors import HttpError
|
2024-01-13 23:00:50 +08:00
|
|
|
from json import loads
|
|
|
|
from random import choice
|
|
|
|
from time import sleep
|
2021-11-23 09:35:16 +08:00
|
|
|
|
2024-03-20 01:08:54 +08:00
|
|
|
SCOPES = [
|
|
|
|
"https://www.googleapis.com/auth/drive",
|
|
|
|
"https://www.googleapis.com/auth/cloud-platform",
|
|
|
|
"https://www.googleapis.com/auth/iam",
|
|
|
|
]
|
2021-11-23 09:35:16 +08:00
|
|
|
project_create_ops = []
|
|
|
|
current_key_dump = []
|
|
|
|
sleep_time = 30
|
|
|
|
|
|
|
|
|
|
|
|
# Create count SAs in project
|
|
|
|
def _create_accounts(service, project, count):
|
|
|
|
batch = service.new_batch_http_request(callback=_def_batch_resp)
|
|
|
|
for _ in range(count):
|
2024-03-20 01:08:54 +08:00
|
|
|
aid = _generate_id("mfc-")
|
2024-01-13 22:25:39 +08:00
|
|
|
batch.add(
|
|
|
|
service.projects()
|
|
|
|
.serviceAccounts()
|
|
|
|
.create(
|
2024-03-20 01:08:54 +08:00
|
|
|
name=f"projects/{project}",
|
2024-01-13 22:25:39 +08:00
|
|
|
body={
|
2024-03-20 01:08:54 +08:00
|
|
|
"accountId": aid,
|
|
|
|
"serviceAccount": {"displayName": aid},
|
2024-01-13 22:25:39 +08:00
|
|
|
},
|
|
|
|
)
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
batch.execute()
|
|
|
|
|
|
|
|
|
|
|
|
# Create accounts needed to fill project
|
|
|
|
def _create_remaining_accounts(iam, project):
|
2024-03-20 01:08:54 +08:00
|
|
|
print(f"Creating accounts in {project}")
|
2021-11-23 09:35:16 +08:00
|
|
|
sa_count = len(_list_sas(iam, project))
|
|
|
|
while sa_count != 100:
|
|
|
|
_create_accounts(iam, project, 100 - sa_count)
|
|
|
|
sa_count = len(_list_sas(iam, project))
|
|
|
|
|
|
|
|
|
|
|
|
# Generate a random id
|
2024-03-20 01:08:54 +08:00
|
|
|
def _generate_id(prefix="saf-"):
|
|
|
|
chars = "-abcdefghijklmnopqrstuvwxyz1234567890"
|
|
|
|
return prefix + "".join(choice(chars) for _ in range(25)) + choice(chars[1:])
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
# List projects using service
|
|
|
|
def _get_projects(service):
|
2024-03-20 01:08:54 +08:00
|
|
|
return [i["projectId"] for i in service.projects().list().execute()["projects"]]
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
# Default batch callback handler
|
|
|
|
def _def_batch_resp(id, resp, exception):
|
|
|
|
if exception is not None:
|
2024-03-20 01:08:54 +08:00
|
|
|
if str(exception).startswith("<HttpError 429"):
|
2021-11-23 09:35:16 +08:00
|
|
|
sleep(sleep_time / 100)
|
|
|
|
else:
|
2024-01-13 22:25:39 +08:00
|
|
|
print(exception)
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
# Project Creation Batch Handler
|
|
|
|
def _pc_resp(id, resp, exception):
|
|
|
|
global project_create_ops
|
|
|
|
if exception is not None:
|
2024-01-13 22:25:39 +08:00
|
|
|
print(exception)
|
2021-11-23 09:35:16 +08:00
|
|
|
else:
|
|
|
|
for i in resp.values():
|
|
|
|
project_create_ops.append(i)
|
|
|
|
|
|
|
|
|
|
|
|
# Project Creation
|
|
|
|
def _create_projects(cloud, count):
|
|
|
|
global project_create_ops
|
|
|
|
batch = cloud.new_batch_http_request(callback=_pc_resp)
|
|
|
|
new_projs = []
|
|
|
|
for _ in range(count):
|
|
|
|
new_proj = _generate_id()
|
|
|
|
new_projs.append(new_proj)
|
2024-03-20 01:08:54 +08:00
|
|
|
batch.add(cloud.projects().create(body={"project_id": new_proj}))
|
2021-11-23 09:35:16 +08:00
|
|
|
batch.execute()
|
|
|
|
|
|
|
|
for i in project_create_ops:
|
|
|
|
while True:
|
|
|
|
resp = cloud.operations().get(name=i).execute()
|
2024-03-20 01:08:54 +08:00
|
|
|
if "done" in resp and resp["done"]:
|
2021-11-23 09:35:16 +08:00
|
|
|
break
|
|
|
|
sleep(3)
|
|
|
|
return new_projs
|
|
|
|
|
|
|
|
|
|
|
|
# Enable services ste for projects in projects
|
|
|
|
def _enable_services(service, projects, ste):
|
|
|
|
batch = service.new_batch_http_request(callback=_def_batch_resp)
|
|
|
|
for i in projects:
|
|
|
|
for j in ste:
|
2024-03-20 01:08:54 +08:00
|
|
|
batch.add(service.services().enable(name=f"projects/{i}/services/{j}"))
|
2021-11-23 09:35:16 +08:00
|
|
|
batch.execute()
|
|
|
|
|
|
|
|
|
|
|
|
# List SAs in project
|
|
|
|
def _list_sas(iam, project):
|
2024-01-13 22:25:39 +08:00
|
|
|
resp = (
|
|
|
|
iam.projects()
|
|
|
|
.serviceAccounts()
|
2024-03-20 01:08:54 +08:00
|
|
|
.list(name=f"projects/{project}", pageSize=100)
|
2024-01-13 22:25:39 +08:00
|
|
|
.execute()
|
|
|
|
)
|
2024-03-20 01:08:54 +08:00
|
|
|
return resp["accounts"] if "accounts" in resp else []
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
# Create Keys Batch Handler
|
|
|
|
def _batch_keys_resp(id, resp, exception):
|
|
|
|
global current_key_dump
|
|
|
|
if exception is not None:
|
|
|
|
current_key_dump = None
|
|
|
|
sleep(sleep_time / 100)
|
|
|
|
elif current_key_dump is None:
|
|
|
|
sleep(sleep_time / 100)
|
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
current_key_dump.append(
|
|
|
|
(
|
|
|
|
resp["name"][resp["name"].rfind("/") :],
|
|
|
|
b64decode(resp["privateKeyData"]).decode("utf-8"),
|
|
|
|
)
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
|
|
|
|
# Create Keys
|
|
|
|
def _create_sa_keys(iam, projects, path):
|
|
|
|
global current_key_dump
|
|
|
|
for i in projects:
|
|
|
|
current_key_dump = []
|
2024-03-20 01:08:54 +08:00
|
|
|
print(f"Downloading keys from {i}")
|
2021-11-23 09:35:16 +08:00
|
|
|
while current_key_dump is None or len(current_key_dump) != 100:
|
|
|
|
batch = iam.new_batch_http_request(callback=_batch_keys_resp)
|
|
|
|
total_sas = _list_sas(iam, i)
|
|
|
|
for j in total_sas:
|
2024-01-13 22:25:39 +08:00
|
|
|
batch.add(
|
|
|
|
iam.projects()
|
|
|
|
.serviceAccounts()
|
|
|
|
.keys()
|
|
|
|
.create(
|
|
|
|
name=f"projects/{i}/serviceAccounts/{j['uniqueId']}",
|
|
|
|
body={
|
2024-03-20 01:08:54 +08:00
|
|
|
"privateKeyType": "TYPE_GOOGLE_CREDENTIALS_FILE",
|
|
|
|
"keyAlgorithm": "KEY_ALG_RSA_2048",
|
2024-01-13 22:25:39 +08:00
|
|
|
},
|
|
|
|
)
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
batch.execute()
|
|
|
|
if current_key_dump is None:
|
2024-03-20 01:08:54 +08:00
|
|
|
print(f"Redownloading keys from {i}")
|
2021-11-23 09:35:16 +08:00
|
|
|
current_key_dump = []
|
|
|
|
else:
|
|
|
|
for index, j in enumerate(current_key_dump):
|
2024-03-20 01:08:54 +08:00
|
|
|
with open(f"{path}/{index}.json", "w+") as f:
|
2021-11-23 09:35:16 +08:00
|
|
|
f.write(j[1])
|
|
|
|
|
|
|
|
|
|
|
|
# Delete Service Accounts
|
|
|
|
def _delete_sas(iam, project):
|
|
|
|
sas = _list_sas(iam, project)
|
|
|
|
batch = iam.new_batch_http_request(callback=_def_batch_resp)
|
|
|
|
for i in sas:
|
2024-03-20 01:08:54 +08:00
|
|
|
batch.add(iam.projects().serviceAccounts().delete(name=i["name"]))
|
2021-11-23 09:35:16 +08:00
|
|
|
batch.execute()
|
|
|
|
|
|
|
|
|
|
|
|
def serviceaccountfactory(
|
2024-03-20 01:08:54 +08:00
|
|
|
credentials="credentials.json",
|
|
|
|
token="token_sa.pickle",
|
|
|
|
path=None,
|
|
|
|
list_projects=False,
|
|
|
|
list_sas=None,
|
|
|
|
create_projects=None,
|
|
|
|
max_projects=12,
|
|
|
|
enable_services=None,
|
|
|
|
services=["iam", "drive"],
|
|
|
|
create_sas=None,
|
|
|
|
delete_sas=None,
|
|
|
|
download_keys=None,
|
2021-11-23 09:35:16 +08:00
|
|
|
):
|
|
|
|
selected_projects = []
|
2024-03-20 01:08:54 +08:00
|
|
|
proj_id = loads(open(credentials, "r").read())["installed"]["project_id"]
|
2021-11-23 09:35:16 +08:00
|
|
|
creds = None
|
|
|
|
if os.path.exists(token):
|
2024-03-20 01:08:54 +08:00
|
|
|
with open(token, "rb") as t:
|
2021-11-23 09:35:16 +08:00
|
|
|
creds = pickle.load(t)
|
|
|
|
if not creds or not creds.valid:
|
|
|
|
if creds and creds.expired and creds.refresh_token:
|
|
|
|
creds.refresh(Request())
|
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
flow = InstalledAppFlow.from_client_secrets_file(credentials, SCOPES)
|
2021-11-23 09:35:16 +08:00
|
|
|
|
2022-04-18 07:28:17 +08:00
|
|
|
creds = flow.run_local_server(port=0, open_browser=False)
|
2021-11-23 09:35:16 +08:00
|
|
|
|
2024-03-20 01:08:54 +08:00
|
|
|
with open(token, "wb") as t:
|
2021-11-23 09:35:16 +08:00
|
|
|
pickle.dump(creds, t)
|
|
|
|
|
2024-03-20 01:08:54 +08:00
|
|
|
cloud = build("cloudresourcemanager", "v1", credentials=creds)
|
|
|
|
iam = build("iam", "v1", credentials=creds)
|
|
|
|
serviceusage = build("serviceusage", "v1", credentials=creds)
|
2021-11-23 09:35:16 +08:00
|
|
|
|
|
|
|
projs = None
|
|
|
|
while projs is None:
|
|
|
|
try:
|
|
|
|
projs = _get_projects(cloud)
|
|
|
|
except HttpError as e:
|
2024-03-20 01:08:54 +08:00
|
|
|
if (
|
|
|
|
loads(e.content.decode("utf-8"))["error"]["status"]
|
|
|
|
== "PERMISSION_DENIED"
|
|
|
|
):
|
2021-11-23 09:35:16 +08:00
|
|
|
try:
|
|
|
|
serviceusage.services().enable(
|
2024-03-20 01:08:54 +08:00
|
|
|
name=f"projects/{proj_id}/services/cloudresourcemanager.googleapis.com"
|
2024-01-13 22:25:39 +08:00
|
|
|
).execute()
|
2021-11-23 09:35:16 +08:00
|
|
|
except HttpError as e:
|
|
|
|
print(e._get_reason())
|
2024-03-20 01:08:54 +08:00
|
|
|
input("Press Enter to retry.")
|
2021-11-23 09:35:16 +08:00
|
|
|
if list_projects:
|
|
|
|
return _get_projects(cloud)
|
|
|
|
if list_sas:
|
|
|
|
return _list_sas(iam, list_sas)
|
|
|
|
if create_projects:
|
2024-01-13 22:25:39 +08:00
|
|
|
print(f"creat projects: {create_projects}")
|
2021-11-23 09:35:16 +08:00
|
|
|
if create_projects > 0:
|
|
|
|
current_count = len(_get_projects(cloud))
|
|
|
|
if current_count + create_projects <= max_projects:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("Creating %d projects" % (create_projects))
|
2021-11-23 09:35:16 +08:00
|
|
|
nprjs = _create_projects(cloud, create_projects)
|
|
|
|
selected_projects = nprjs
|
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
sys.exit(
|
|
|
|
"No, you cannot create %d new project (s).\n"
|
|
|
|
"Please reduce value of --quick-setup.\n"
|
|
|
|
"Remember that you can totally create %d projects (%d already).\n"
|
|
|
|
"Please do not delete existing projects unless you know what you are doing"
|
|
|
|
% (create_projects, max_projects, current_count)
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
print(
|
|
|
|
"Will overwrite all service accounts in existing projects.\n"
|
|
|
|
"So make sure you have some projects already."
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
input("Press Enter to continue...")
|
|
|
|
|
|
|
|
if enable_services:
|
|
|
|
ste = [enable_services]
|
2024-03-20 01:08:54 +08:00
|
|
|
if enable_services == "~":
|
2021-11-23 09:35:16 +08:00
|
|
|
ste = selected_projects
|
2024-03-20 01:08:54 +08:00
|
|
|
elif enable_services == "*":
|
2021-11-23 09:35:16 +08:00
|
|
|
ste = _get_projects(cloud)
|
2024-03-20 01:08:54 +08:00
|
|
|
services = [f"{i}.googleapis.com" for i in services]
|
|
|
|
print("Enabling services")
|
2021-11-23 09:35:16 +08:00
|
|
|
_enable_services(serviceusage, ste, services)
|
|
|
|
if create_sas:
|
|
|
|
stc = [create_sas]
|
2024-03-20 01:08:54 +08:00
|
|
|
if create_sas == "~":
|
2021-11-23 09:35:16 +08:00
|
|
|
stc = selected_projects
|
2024-03-20 01:08:54 +08:00
|
|
|
elif create_sas == "*":
|
2021-11-23 09:35:16 +08:00
|
|
|
stc = _get_projects(cloud)
|
|
|
|
for i in stc:
|
|
|
|
_create_remaining_accounts(iam, i)
|
|
|
|
if download_keys:
|
|
|
|
try:
|
|
|
|
os.mkdir(path)
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno != errno.EEXIST:
|
|
|
|
raise
|
|
|
|
std = [download_keys]
|
2024-03-20 01:08:54 +08:00
|
|
|
if download_keys == "~":
|
2021-11-23 09:35:16 +08:00
|
|
|
std = selected_projects
|
2024-03-20 01:08:54 +08:00
|
|
|
elif download_keys == "*":
|
2021-11-23 09:35:16 +08:00
|
|
|
std = _get_projects(cloud)
|
|
|
|
_create_sa_keys(iam, std, path)
|
|
|
|
if delete_sas:
|
|
|
|
std = []
|
|
|
|
std.append(delete_sas)
|
2024-03-20 01:08:54 +08:00
|
|
|
if delete_sas == "~":
|
2021-11-23 09:35:16 +08:00
|
|
|
std = selected_projects
|
2024-03-20 01:08:54 +08:00
|
|
|
elif delete_sas == "*":
|
2021-11-23 09:35:16 +08:00
|
|
|
std = _get_projects(cloud)
|
|
|
|
for i in std:
|
2024-03-20 01:08:54 +08:00
|
|
|
print(f"Deleting service accounts in {i}")
|
2021-11-23 09:35:16 +08:00
|
|
|
_delete_sas(iam, i)
|
|
|
|
|
|
|
|
|
2024-03-20 01:08:54 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
parse = ArgumentParser(description="A tool to create Google service accounts.")
|
|
|
|
parse.add_argument(
|
|
|
|
"--path",
|
|
|
|
"-p",
|
|
|
|
default="accounts",
|
|
|
|
help="Specify an alternate directory to output the credential files.",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--token", default="token_sa.pickle", help="Specify the pickle token file path."
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--credentials",
|
|
|
|
default="credentials.json",
|
|
|
|
help="Specify the credentials file path.",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--list-projects",
|
|
|
|
default=False,
|
|
|
|
action="store_true",
|
|
|
|
help="List projects viewable by the user.",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--list-sas", default=False, help="List service accounts in a project."
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--create-projects", type=int, default=None, help="Creates up to N projects."
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--max-projects",
|
|
|
|
type=int,
|
|
|
|
default=12,
|
|
|
|
help="Max amount of project allowed. Default: 12",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--enable-services",
|
|
|
|
default=None,
|
|
|
|
help="Enables services on the project. Default: IAM and Drive",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--services",
|
|
|
|
nargs="+",
|
|
|
|
default=["iam", "drive"],
|
|
|
|
help="Specify a different set of services to enable. Overrides the default.",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--create-sas", default=None, help="Create service accounts in a project."
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--delete-sas", default=None, help="Delete service accounts in a project."
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--download-keys",
|
|
|
|
default=None,
|
|
|
|
help="Download keys for all the service accounts in a project.",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--quick-setup",
|
|
|
|
default=None,
|
|
|
|
type=int,
|
|
|
|
help="Create projects, enable services, create service accounts and download keys. ",
|
|
|
|
)
|
|
|
|
parse.add_argument(
|
|
|
|
"--new-only",
|
|
|
|
default=False,
|
|
|
|
action="store_true",
|
|
|
|
help="Do not use exisiting projects.",
|
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
args = parse.parse_args()
|
|
|
|
# If credentials file is invalid, search for one.
|
|
|
|
if not os.path.exists(args.credentials):
|
2024-03-20 01:08:54 +08:00
|
|
|
options = glob("*.json")
|
|
|
|
print(
|
|
|
|
"No credentials found at %s. Please enable the Drive API in:\n"
|
|
|
|
"https://developers.google.com/drive/api/v3/quickstart/python\n"
|
|
|
|
"and save the json file as credentials.json" % args.credentials
|
|
|
|
)
|
2024-01-13 22:25:39 +08:00
|
|
|
if not options:
|
2021-11-23 09:35:16 +08:00
|
|
|
exit(-1)
|
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("Select a credentials file below.")
|
|
|
|
inp_options = [str(i) for i in list(range(1, len(options) + 1))] + options
|
2021-11-23 09:35:16 +08:00
|
|
|
for i in range(len(options)):
|
2024-03-20 01:08:54 +08:00
|
|
|
print(" %d) %s" % (i + 1, options[i]))
|
2021-11-23 09:35:16 +08:00
|
|
|
inp = None
|
|
|
|
while True:
|
2024-03-20 01:08:54 +08:00
|
|
|
inp = input("> ")
|
2021-11-23 09:35:16 +08:00
|
|
|
if inp in inp_options:
|
|
|
|
break
|
|
|
|
args.credentials = inp if inp in options else options[int(inp) - 1]
|
2024-01-13 22:25:39 +08:00
|
|
|
print(
|
2024-03-20 01:08:54 +08:00
|
|
|
f"Use --credentials {args.credentials} next time to use this credentials file."
|
2024-01-13 22:25:39 +08:00
|
|
|
)
|
2021-11-23 09:35:16 +08:00
|
|
|
if args.quick_setup:
|
2024-03-20 01:08:54 +08:00
|
|
|
opt = "~" if args.new_only else "*"
|
|
|
|
args.services = ["iam", "drive"]
|
2021-11-23 09:35:16 +08:00
|
|
|
args.create_projects = args.quick_setup
|
|
|
|
args.enable_services = opt
|
|
|
|
args.create_sas = opt
|
|
|
|
args.download_keys = opt
|
|
|
|
resp = serviceaccountfactory(
|
|
|
|
path=args.path,
|
|
|
|
token=args.token,
|
|
|
|
credentials=args.credentials,
|
|
|
|
list_projects=args.list_projects,
|
|
|
|
list_sas=args.list_sas,
|
|
|
|
create_projects=args.create_projects,
|
|
|
|
max_projects=args.max_projects,
|
|
|
|
create_sas=args.create_sas,
|
|
|
|
delete_sas=args.delete_sas,
|
|
|
|
enable_services=args.enable_services,
|
|
|
|
services=args.services,
|
2024-03-20 01:08:54 +08:00
|
|
|
download_keys=args.download_keys,
|
2021-11-23 09:35:16 +08:00
|
|
|
)
|
|
|
|
if resp is not None:
|
|
|
|
if args.list_projects:
|
|
|
|
if resp:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("Projects (%d):" % len(resp))
|
2021-11-23 09:35:16 +08:00
|
|
|
for i in resp:
|
2024-03-20 01:08:54 +08:00
|
|
|
print(f" {i}")
|
2021-11-23 09:35:16 +08:00
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("No projects.")
|
2021-11-23 09:35:16 +08:00
|
|
|
elif args.list_sas:
|
|
|
|
if resp:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("Service accounts in %s (%d):" % (args.list_sas, len(resp)))
|
2021-11-23 09:35:16 +08:00
|
|
|
for i in resp:
|
2024-01-13 22:25:39 +08:00
|
|
|
print(f" {i['email']} ({i['uniqueId']})")
|
2021-11-23 09:35:16 +08:00
|
|
|
else:
|
2024-03-20 01:08:54 +08:00
|
|
|
print("No service accounts.")
|