diff --git a/cms/engine/config/__init__.py b/cms/engine/config/__init__.py new file mode 100644 index 0000000..0523e4d --- /dev/null +++ b/cms/engine/config/__init__.py @@ -0,0 +1,28 @@ +""" +This file handles all things configuration i.e even the parts of the configuration we are interested in +""" +import os +import json + +def get (path) : + if os.path.exists(path) : + f = open(path) + _conf = json.loads(f.read()) + f.close() + else: + _conf = {} + return _conf +def _isvalid(_allowed,**_args): + + if not list(set(_allowed) - set(_args.keys())) : + _pargs = {} + for key in _allowed : + _pargs [key] = _args[key] + return _pargs + return False + +def write(_config, path): + f = open(path,'w') + f.write( json.dumps(_config)) ; + f.close() + diff --git a/cms/engine/config/structure.py b/cms/engine/config/structure.py new file mode 100644 index 0000000..93aca9c --- /dev/null +++ b/cms/engine/config/structure.py @@ -0,0 +1,77 @@ +""" +This file handles the structure (strict) of the configuration file, +Not all elements are programmatically editable (for now) +""" +import meta +class Section : + + def build (self,**_args): + _data = {} + for _attr in dir(self): + if not _attr.startswith('_') and _attr not in ['build','update']: + _kwargs = _args if _attr not in _args or type(_args[_attr]) !=dict else _args[_attr] + _data[_attr] = getattr(self,_attr)(**_kwargs) + _name = type (self).__name__.lower() + + return {_name:_data } + + + + def update(self,_default,**_args) : + for _attr in _default.keys(): + + if _attr in _args : + _default[_attr] = _args[_attr] + return _default + +class System (Section) : + + def logo(self,**_args): + + return 'www/html/_assets/images/logo.png' if 'logo' not in _args else _args['logo'] + def theme (self,**_args): + """ + setting the theme given the name of the theme + :name name of the theme to set (optional) + """ + return 'default' if 'name' not in _args else _args['name'] + def version(self,**_args): + return meta.__version__ if 'version' not in _args else _args['version'] + def context (self,**_args): + return "" if 'context' not in _args else _args['context'] + def app (self,**_args): + _data = {'debug':True,'port':8084,'threaded':True,'host':'0.0.0.0'} + return self.update(_data,**_args) + + def source(self,**_args): + _data = {'id':'disk'} + if set(['key','auth']) & set(_args.keys()): + # + # key: reboot the app & auth: for cloud access + # + for _attr in ['key','auth'] : + if _attr in _args: + _data[_attr] = _args[_attr] + + _data = self.update(_data,**_args) + return _data + +class Layout (Section): + def index (self,**_args): + return "index.html" if 'index' not in _args else _args['index'] + def root(self,**_args): + return 'wwww/html' if 'root' not in _args else _args['root'] + def footer(self,**_args): + return [{'text':'Powered by QCMS'}] if 'footer' not in _args else _args['footer'] + def on(self,**_args): + return {'load':{}} if 'on' not in _args else _args['on'] + def order(self,**_args): + return {'menu':[]} if 'order' not in _args else _args['order'] + # def menu (self,**_args): + # return {'menu':[]} if 'menu' not in _args else _args['menu'] + def overwrite(self,**_args): + return {} + def header (self,**_args): + _data = {"title":"QCMS Project", "subtitle":"Powered by Python Flask"} + return self.update(_data,**_args) + \ No newline at end of file diff --git a/cms/engine/plugins/__init__.py b/cms/engine/plugins/__init__.py new file mode 100644 index 0000000..6ada10b --- /dev/null +++ b/cms/engine/plugins/__init__.py @@ -0,0 +1,42 @@ +import json +import pandas as pd +import importlib +import importlib.util +import os + +def stats (_config) : + """ + Returns the statistics of the plugins + """ + _data = [] + for _name in _config : + _log = {"files":_name,"loaded":len(_config[_name]),"logs":json.dumps(_config[_name])} + _data.append(_log) + return pd.DataFrame(_data) + pass +def load(_path,_filename,_name) : + """ + This function will load external module form a given location and return a pointer to a function in a given module + :path absolute path of the file (considered plugin) to be loaded + :name name of the function to be applied + """ + # _path = _args['path'] #os.sep.join([_args['root'],'plugin']) + + if os.path.isdir(_path): + files = os.listdir(_path) + if files : + files = [name for name in files if name.endswith('.py') and name == _filename] + if files: + _path = os.sep.join([_path,files[0]]) + else: + return None + else: + return None + #-- We have a file ... + # _name = _args['name'] + spec = importlib.util.spec_from_file_location(_filename, _path) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + return getattr(module,_name) if hasattr(module,_name) else None + + diff --git a/cms/engine/project/__init__.py b/cms/engine/project/__init__.py new file mode 100644 index 0000000..a24339f --- /dev/null +++ b/cms/engine/project/__init__.py @@ -0,0 +1,154 @@ +# from cms import _system, _header,_layout +import base64 +import os +import json +import meta +from cms.engine import themes +# def make (**_args): +# """ +# :context +# :port port to be served + +# :title title of the application +# :root folder of the content to be scanned +# """ +# if 'port' in _args : +# _args['app'] = {'port':_args['port']} +# del _args['port'] +# if 'context' not in _args : +# _args['context'] = '' +# _info ={'system': _system(**_args)} +# _hargs = {'title':'QCMS'} if 'title' not in _args else {'title':_args['title']} +# _hargs['subtitle'] = '' if 'subtitle' not in _args else _args['subtitle'] +# _hargs['logo'] = True + + + + +# _info['layout'] = _layout(root=_args['root'],index='index.html') +# _info['layout']['header'] = _header(**_hargs) +# # + + +# if 'footer' in _args : +# _info['layout']['footer'] = [{'text':_args['footer']}] if type(_args['footer']) != list else [{'text':term} for term in _args['footer']] +# else: +# _info['layout']['footer'] = [{'text':'Powered by QCMS'}] + +# return _info +def make_folder (projectFolder, webroot): + """ + This function creates project folders, inside the project folder is the web root folder + """ + + if not os.path.exists(projectFolder): + os.makedirs(projectFolder) + _path = os.sep.join([projectFolder,webroot]) + folders = ['_assets',os.sep.join(['_assets','images']), os.sep.join(['_assets','themes']), '_plugins'] + for folder in folders : + _projectPath = os.sep.join([_path,folder]) + if not os.path.exists(_projectPath) : + os.makedirs(_projectPath) +def _ilogo (_path): + """ + This function creates a default logo in a designated folder, after the project folder has been created + :_path project path + """ + _image = """""" + _image1 = """""" + _image2 = """""" + _index = 0 + for _image in [_image1,_image2] : + _,_image= _image.split(',',1) + _logoPath = os.sep.join([_path,'_assets','images','logo.png']) + if _index > 0 : + _logoPath = _logoPath.replace('.png',f'-{_index}.png') + f = open(_logoPath,'wb') + f.write(base64.b64decode(_image)) + f.close() + _index += 1 + +def _index (_path,root): + """ + Creating a default index.html for the site given the project root location + """ + _html = f""" +
+
Thank you for considering QCMS
+
version {meta.__version__} by {meta.__author__}, {meta.__email__}
+
+

+ +

+ +
+
+
+ +
+ QCMS is powered by Python/Flask +
    + Small footprint & can work as either a portal or a standalone application +
+
    + Simple paradigm that leverages folder structure to generate a site +
+
+ +
+
+ QCMS has built-in support for industry standard frameworks +
+ + +
+ + + +
+
+

+

Learn more about QCMS and at {themes.URL}
+

+ """ + _indexfile = os.sep.join([_path,'index.html']) + if os.path.exists(_indexfile): + # + # In case a project needs to be upgraded ... + f = open(_indexfile,'w') + f.write(_html) + f.close() +def _itheme(_path,_root,_name='default'): + _data = themes.Get(_name)[_name] + _themefolder = os.sep.join([_path,_root,'_assets','themes',_name]) + if not os.path.exists(_themefolder) : + os.makedirs(_themefolder) + + for _name in _data : + f = open(os.sep.join([_themefolder,_name+'.css']),'w') + f.write(_data[_name]) + f.close() + pass +def make (**_args) : + """ + This function create a project folder and within the folder are all the elements needed + """ + _config = _args['config'] + _config['plugins'] = {} + _folder = _args['folder'] + _root = _config['layout']['root'] #-- web root folder + make_folder(_folder,_root) + f = open(os.sep.join([_folder,'qcms-manifest.json']),'w') + f.write( json.dumps(_config)) + f.close() + + _ilogo(os.sep.join([_folder,_root])) + _index(os.sep.join([_folder,_root]),_root) + _itheme(_folder,_root) + \ No newline at end of file diff --git a/cms/engine/themes/__init__.py b/cms/engine/themes/__init__.py new file mode 100644 index 0000000..d3d00b7 --- /dev/null +++ b/cms/engine/themes/__init__.py @@ -0,0 +1,44 @@ +""" +This class implements the base infrastructure to handle themes, the class must leverage +default themes will be stored in layout.root._assets.themes, The expected files in a theme are the following: + - borders + - buttons + - layout + - menu + - header +If the following are not available then we should load the default one. +This would avoid crashes but would come at the expense of a lack of consistent visual layout (alas) +""" +import requests +import os +URL = os.environ['QCMS_HOME_URL'] if 'QCMS_HOME_URL' in os.environ else 'https://dev.the-phi.com/qcms' +def current (_system) : + return _system['theme'] +def List (_url = URL) : + """ + calling qcms to list the available URL + """ + try: + _url = '/'.join([_url,'api','themes','List']) + return requests.get(_url).json() + except Exception as e: + + pass + return [] +def Get(theme,_url= URL) : + """ + This function retrieves a particular theme from a remote source + The name should be in the list provided by the above function + """ + try: + _url = '/'.join([_url,'api','themes','Get']) +f'?theme={theme}' + return requests.get(_url).json() + except Exception as e: + pass + return {} +def installed (path): + return os.listdir(os.sep.join([path,'_assets','themes'])) + +def Set(theme,_system) : + _system['theme'] = theme + return _system diff --git a/cms/static/css/dialog.css b/cms/static/css/dialog.css new file mode 100644 index 0000000..7ec2adf --- /dev/null +++ b/cms/static/css/dialog.css @@ -0,0 +1,10 @@ + +.dialog-title { + background-color:#FF6500;color:#FFFFFF; + text-transform:capitalize; font-weight:bold; align-items:center;display:grid; grid-template-columns:auto 32px; +} +.dialog-button { + display:grid; + grid-template-columns: auto 115px; + gap:4px; +} diff --git a/cms/static/css/source-code.css b/cms/static/css/source-code.css new file mode 100644 index 0000000..e46f72c --- /dev/null +++ b/cms/static/css/source-code.css @@ -0,0 +1,18 @@ +.source-code { + background-color: #000000; COLOR:#ffffff; + font-family: 'Courier New', Courier, monospace; + padding:8px; + padding-left:10px; + text-wrap: wrap; + width:calc(100% - 40px); + border-left:8px solid #CAD5E0; margin-left:10px; font-weight: bold; + font-size:14px; +} + +.editor { + background-color:#f3f3f3; + color:#000000; +} + +.editor .keyword {color: #4682b4; font-weight:bold} +.code-comment { font-style: italic; color:gray; font-size:13px;} \ No newline at end of file