# -*- coding: utf-8 -*-
# Copyright © Cloud Linux GmbH & Cloud Linux Software, Inc 2010-2019 All Rights Reserved
#
# Licensed under CLOUD LINUX LICENSE AGREEMENT
# http://cloudlinux.com/docs/LICENSE.TXT
from __future__ import print_function
from __future__ import division
from __future__ import absolute_import
import json
import os
import pwd
from clselect.utils import check_call, list_dirs, run_command_full
from .interpreters import Interpreter
DEFAULT_PREFIX = 'nodevenv'
NODEVENV_BIN = os.path.join(os.path.dirname(__file__), 'nodevenv.py')
PYTHON_PATH = '/opt/cloudlinux/venv/bin/python3'
APP_CONFIG = 'package.json'
class Environment(object):
def __init__(self, name, user=None, prefix=None):
self.name = name
if user:
self.user = user
else:
self.user = pwd.getpwuid(os.getuid()).pw_name
if prefix is None:
self.prefix = DEFAULT_PREFIX
else:
self.prefix = prefix
self.path = os.path.join(_abs_prefix(self.user, self.prefix), name)
self._interpreter = None
self._npm = None
self.interpreter_name = 'node' + name
def __repr__(self):
return ("%s.%s(name='%s', user='%s', prefix='%s')" % (
self.__class__.__module__, self.__class__.__name__,
self.name, self.user, self.prefix))
def _demote(self):
user_pwd = pwd.getpwnam(self.user)
def func():
os.setgid(user_pwd.pw_gid)
os.setuid(user_pwd.pw_uid)
os.environ['USER'] = self.user
os.environ['HOME'] = user_pwd.pw_dir
return func
def as_dict(self, key=None):
e = {
'name': self.name,
'interpreter': self.interpreter(),
}
if key:
del e[key]
return {getattr(self, key): e}
return e
def as_deepdict(self, key=None):
e = {
'name': self.name,
'interpreter': self.interpreter().as_dict(),
}
if key:
del e[key]
return {getattr(self, key): e}
return e
def create(self, interpreter, destroy_first=False):
args = [
PYTHON_PATH, NODEVENV_BIN,
'--node', interpreter.binary,
]
if destroy_first:
args.append('--recreate-bin')
args.append(self.path)
kwargs = {'preexec_fn': self._demote()}
if os.getuid() == 0:
args = ['/bin/cagefs_enter'] + args
check_call(*args, **kwargs)
def destroy(self):
check_call('/bin/rm', '-r', '--interactive=never', self.path,
preexec_fn=self._demote())
def exists(self):
return os.path.exists(self.path)
def interpreter(self):
if not self._interpreter:
self._interpreter = Interpreter(prefix=self.path)
return self._interpreter
def npm(self):
if not self._npm:
self._npm = os.path.join(self.path, 'bin', 'npm')
return self._npm
def extension_install(self, extension='-', cwd=None):
"""
Install nodejs extension
:param cwd: current working directory for npm (it's full path to app root by default)
:param extension: name and version of extension
:return: None
"""
# git is need for npm install from git repositories
# we didn't add git to cagefs, because we use git from cpanel
command = (self.npm(), 'install')
# extension is defined
if extension != '-':
command += (extension,)
# npm takes list of extensions from package.json if extension is '-'
else:
config_path = os.path.join(cwd, APP_CONFIG)
if not os.path.isfile(config_path):
# package.json is absent
return 0
# command sample
# npm install