Skip to content

Commit

Permalink
T5743: HTTPS API ability to import PKI certificates
Browse files Browse the repository at this point in the history
  • Loading branch information
natali-rs1985 committed Aug 13, 2024
1 parent 7a54689 commit 27fb633
Show file tree
Hide file tree
Showing 5 changed files with 96 additions and 14 deletions.
2 changes: 1 addition & 1 deletion data/templates/https/nginx.default.j2
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ server {
ssl_ciphers 'ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-AES256-GCM-SHA384:DHE-RSA-AES128-GCM-SHA256:DHE-DSS-AES128-GCM-SHA256:kEDH+AESGCM:ECDHE-RSA-AES128-SHA256:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA:ECDHE-ECDSA-AES128-SHA:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA:ECDHE-ECDSA-AES256-SHA:DHE-RSA-AES128-SHA256:DHE-RSA-AES128-SHA:DHE-DSS-AES128-SHA256:DHE-RSA-AES256-SHA256:DHE-DSS-AES256-SHA:DHE-RSA-AES256-SHA:!aNULL:!eNULL:!EXPORT:!DES:!RC4:!3DES:!MD5:!PSK';

# proxy settings for HTTP API, if enabled; 503, if not
location ~ ^/(retrieve|configure|config-file|image|container-image|generate|show|reboot|reset|poweroff|docs|openapi.json|redoc|graphql) {
location ~ ^/(retrieve|configure|config-file|image|import-pki|container-image|generate|show|reboot|reset|poweroff|docs|openapi.json|redoc|graphql) {
{% if api is vyos_defined %}
proxy_pass http://unix:/run/api.sock;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
Expand Down
11 changes: 11 additions & 0 deletions python/vyos/configsession.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,9 @@
SAVE_CONFIG = ['/usr/libexec/vyos/vyos-save-config.py']
INSTALL_IMAGE = ['/usr/libexec/vyos/op_mode/image_installer.py',
'--action', 'add', '--no-prompt', '--image-path']
IMPORT_PKI = ['/opt/vyatta/bin/vyatta-op-cmd-wrapper', 'import']
IMPORT_PKI_NO_PROMPT = ['/usr/libexec/vyos/op_mode/pki.py',
'--action', 'import', '--no-prompt']
REMOVE_IMAGE = ['/usr/libexec/vyos/op_mode/image_manager.py',
'--action', 'delete', '--no-prompt', '--image-name']
SET_DEFAULT_IMAGE = ['/usr/libexec/vyos/op_mode/image_manager.py',
Expand Down Expand Up @@ -239,6 +242,14 @@ def remove_image(self, name):
out = self.__run_command(REMOVE_IMAGE + [name])
return out

def import_pki(self, path):
out = self.__run_command(IMPORT_PKI + path)
return out

def import_pki_no_prompt(self, path):
out = self.__run_command(IMPORT_PKI_NO_PROMPT + path)
return out

def set_default_image(self, name):
out = self.__run_command(SET_DEFAULT_IMAGE + [name])
return out
Expand Down
2 changes: 1 addition & 1 deletion python/vyos/pki.py
Original file line number Diff line number Diff line change
Expand Up @@ -271,7 +271,7 @@ def load_private_key(raw_data, passphrase=None, wrap_tags=True):

try:
return serialization.load_pem_private_key(bytes(raw_data, 'utf-8'), password=passphrase)
except ValueError:
except (ValueError, TypeError):
return False

def load_openssh_public_key(raw_data, type):
Expand Down
33 changes: 21 additions & 12 deletions src/op_mode/pki.py
Original file line number Diff line number Diff line change
Expand Up @@ -693,7 +693,7 @@ def generate_wireguard_psk(interface=None, peer=None, install=False):
print(f'Pre-shared key: {psk}')

# Import functions
def import_ca_certificate(name, path=None, key_path=None):
def import_ca_certificate(name, path=None, key_path=None, no_prompt=False, passphrase=None):
if path:
if not os.path.exists(path):
print(f'File not found: {path}')
Expand All @@ -717,19 +717,20 @@ def import_ca_certificate(name, path=None, key_path=None):
return

key = None
passphrase = ask_input('Enter private key passphrase: ') or None
if not no_prompt:
passphrase = ask_input('Enter private key passphrase: ') or None

with open(key_path) as f:
key_data = f.read()
key = load_private_key(key_data, passphrase=passphrase, wrap_tags=False)

if not key:
print(f'Invalid private key or passphrase: {path}')
print(f'Invalid private key or passphrase: {key_path}')
return

install_certificate(name, private_key=key, is_ca=True)

def import_certificate(name, path=None, key_path=None):
def import_certificate(name, path=None, key_path=None, no_prompt=False, passphrase=None):
if path:
if not os.path.exists(path):
print(f'File not found: {path}')
Expand All @@ -753,14 +754,15 @@ def import_certificate(name, path=None, key_path=None):
return

key = None
passphrase = ask_input('Enter private key passphrase: ') or None
if not no_prompt:
passphrase = ask_input('Enter private key passphrase: ') or None

with open(key_path) as f:
key_data = f.read()
key = load_private_key(key_data, passphrase=passphrase, wrap_tags=False)

if not key:
print(f'Invalid private key or passphrase: {path}')
print(f'Invalid private key or passphrase: {key_path}')
return

install_certificate(name, private_key=key, is_ca=False)
Expand Down Expand Up @@ -799,7 +801,7 @@ def import_dh_parameters(name, path):

install_dh_parameters(name, dh)

def import_keypair(name, path=None, key_path=None):
def import_keypair(name, path=None, key_path=None, no_prompt=False, passphrase=None):
if path:
if not os.path.exists(path):
print(f'File not found: {path}')
Expand All @@ -823,14 +825,15 @@ def import_keypair(name, path=None, key_path=None):
return

key = None
passphrase = ask_input('Enter private key passphrase: ') or None
if not no_prompt:
passphrase = ask_input('Enter private key passphrase: ') or None

with open(key_path) as f:
key_data = f.read()
key = load_private_key(key_data, passphrase=passphrase, wrap_tags=False)

if not key:
print(f'Invalid private key or passphrase: {path}')
print(f'Invalid private key or passphrase: {key_path}')
return

install_keypair(name, None, private_key=key, prompt=False)
Expand Down Expand Up @@ -1011,6 +1014,9 @@ def show_crl(name=None, pem=False):
parser.add_argument('--filename', help='Write certificate into specified filename', action='store')
parser.add_argument('--key-filename', help='Write key into specified filename', action='store')

parser.add_argument('--no-prompt', action='store_true', help='Perform action non-interactively')
parser.add_argument('--passphrase', help='A passphrase to decrypt the private key')

args = parser.parse_args()

try:
Expand Down Expand Up @@ -1054,15 +1060,18 @@ def show_crl(name=None, pem=False):
generate_wireguard_psk(args.interface, peer=args.peer, install=args.install)
elif args.action == 'import':
if args.ca:
import_ca_certificate(args.ca, path=args.filename, key_path=args.key_filename)
import_ca_certificate(args.ca, path=args.filename, key_path=args.key_filename,
no_prompt=args.no_prompt, passphrase=args.passphrase)
elif args.certificate:
import_certificate(args.certificate, path=args.filename, key_path=args.key_filename)
import_certificate(args.certificate, path=args.filename, key_path=args.key_filename,
no_prompt=args.no_prompt, passphrase=args.passphrase)
elif args.crl:
import_crl(args.crl, args.filename)
elif args.dh:
import_dh_parameters(args.dh, args.filename)
elif args.keypair:
import_keypair(args.keypair, path=args.filename, key_path=args.key_filename)
import_keypair(args.keypair, path=args.filename, key_path=args.key_filename,
no_prompt=args.no_prompt, passphrase=args.passphrase)
elif args.openvpn:
import_openvpn_secret(args.openvpn, args.filename)
elif args.action == 'show':
Expand Down
62 changes: 62 additions & 0 deletions src/services/vyos-http-api-server
Original file line number Diff line number Diff line change
Expand Up @@ -212,6 +212,22 @@ class ImageModel(ApiModel):
}
}

class ImportPkiModel(ApiModel):
op: StrictStr
path: List[StrictStr]
passphrase: StrictStr = None

class Config:
schema_extra = {
"example": {
"key": "id_key",
"op": "import_pki",
"path": ["op", "mode", "path"],
"passphrase": "passphrase",
}
}


class ContainerImageModel(ApiModel):
op: StrictStr
name: StrictStr = None
Expand Down Expand Up @@ -585,6 +601,14 @@ def _configure_op(data: Union[ConfigureModel, ConfigureListModel,

return success(msg)

def create_path_import_pki_no_prompt(path):
correct_paths = ['ca', 'certificate', 'key-pair']
if path[1] not in correct_paths:
return False
path[1] = '--' + path[1].replace('-', '')
path[3] = '--key-filename'
return path[1:]

@app.post('/configure')
def configure_op(data: Union[ConfigureModel,
ConfigureListModel],
Expand Down Expand Up @@ -814,6 +838,44 @@ def reset_op(data: ResetModel):

return success(res)

@app.post('/import-pki')
def import_pki(data: ImportPkiModel):
session = app.state.vyos_session

op = data.op
path = data.path

lock.acquire()

try:
if op == 'import-pki':
# need to get rid or interactive mode for private key
if len(path) == 5 and path[3] in ['key-file', 'private-key']:
path_no_prompt = create_path_import_pki_no_prompt(path)
if not path_no_prompt:
return error(400, f"Invalid command: {' '.join(path)}")
if data.passphrase:
path_no_prompt += ['--passphrase', data.passphrase]
res = session.import_pki_no_prompt(path_no_prompt)
else:
res = session.import_pki(path)
if not res[0].isdigit():
return error(400, res)
# commit changes
session.commit()
res = res.split('. ')[0]
else:
return error(400, f"'{op}' is not a valid operation")
except ConfigSessionError as e:
return error(400, str(e))
except Exception as e:
logger.critical(traceback.format_exc())
return error(500, "An internal error occured. Check the logs for details.")
finally:
lock.release()

return success(res)

@app.post('/poweroff')
def poweroff_op(data: PoweroffModel):
session = app.state.vyos_session
Expand Down

0 comments on commit 27fb633

Please sign in to comment.