Commits

codernity committed 8a37cc6

Some changes:
- IndexCreator in interface / app
- HTTP server updated a bit

Comments (0)

Files changed (9)

CodernityDBHTTP/app.py

 else:
     has_json = True
 
-from misc import with_basic_auth, default_method, static_file, auth_static_file, tuple_funct, fixed_user_pass, make_one_item_iter, values_to_base, handle_exception
+from misc import with_basic_auth, default_method, static_file, auth_static_file, tuple_funct, fixed_user_pass, make_one_item_iter, values_to_base, handle_exception, open_with_autosave
 from all_exceptions import *
 from CodernityDB.database import header_for_indexes
 from CodernityDB import __version__ as __db__version__
     from __init__ import __version__ as __srv__version__
 
 
+MAX_POST_SIZE = 1024 * 1024
+
+
 content_types = {
     'application/json': (json.loads, make_one_item_iter(json.dumps)),
 }
                 except ValueError:
                     return handle_exception(start_response, output_f, inc_type, 400, 'You have to specify body')
                 else:
+                    if l > self.MAX_POST_SIZE:
+                        return handle_exception(start_response, output_f, inc_type, 400, 'Too big body max allowed %d bytes' % self.MAX_POST_SIZE)
                     if l != 0:
                         inp = env.get('wsgi.input')
                         try:
                 return handle_exception(start_response, output_f, inc_type)
 
 
-def get_app(db_class, db_path, cfg):
+def get_app(db_class, db_path, cfg, cfg_path):
 
     def auth_function(target, realm="Admin"):
         return with_basic_auth(target, realm, fixed_user_pass('admin', cfg.get('admin_pass')))
         'compact_index': auth_function(db.compact_index),
         'reindex': auth_function(db.reindex),
         'reindex_index': auth_function(db.reindex_index),
-        'open': auth_function(db.open),
+        'open': auth_function(open_with_autosave(db, cfg_path, cfg)),
         'opened': auth_function(lambda: db.opened),
         'destroy': auth_function(db.destroy),
         'exists': auth_function(db.exists),
         '_ui': auth_static_file(os.path.abspath(_ui_path), static_file)
     }
 
-    return CodernityDBApp(routes, static_routes).codernitydb_app
+    app = CodernityDBApp(routes, static_routes)
+    app.MAX_POST_SIZE = cfg.get('max_post_size', MAX_POST_SIZE)
+    return app.codernitydb_app

CodernityDBHTTP/config.json

 {
-    "server": "auto",
-    "port": 9876,
-    "bind": "0.0.0.0",
-    "db_path": null,
-    "admin_pass": "password",
-    "backlog": 50
+    "admin_pass": "password", 
+    "auto_save": 0, 
+    "backlog": 50, 
+    "bind": "0.0.0.0", 
+    "db_path": null, 
+    "port": 9876, 
+    "server": "auto"
 }

CodernityDBHTTP/interface/index.html

             <div data-bind="if: dbInit">
                 <h2>Create or open database</h2>
                 <div class="alert alert-info">
-                    <strong>Note:</strong> If database from provided path is exist, system will try to open it instead create.
+                    <strong>Note:</strong> If there will be database in location that you enter, it will be opened instead of created.
                 </div>
                 <form class="form-horizontal" method="post" data-bind="submit: db_init">
                     <div class="control-group">
                 </div>
             </div>
             <!-- modal -->
-            <div class="modal hide" id="destroy_db_modal">
+            <div class="modal hide" tabindex="-1" id="destroy_db_modal">
                 <div class="modal-header">
                     <a class="close" data-dismiss="modal">x</a>
                     <h3>Are you sure to destroy?</h3>
             <!-- newIndex -->
             <div data-bind="with: newIndex">
                 <!-- modal -->
-                <div class="modal hide fade" id="save_plain_modal">
+                <div class="modal hide fade" tabindex="-1" id="save_plain_modal">
                     <div class="modal-header">
                         <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
                         <h3>Save index</h3>
                     <fieldset>
                         <div class="control-group">
                             <div class="controls">
-                                <textarea style="width: 100%; height: 200px;" data-bind="codeMirror: index_code, codeMirrorOptions: {mode: {name: 'python'}, lineNumbers: true }" type="text" class="input-xlarge" id="index_code"></textarea>
+                            <textarea style="width: 100%; height: 200px;" data-bind="codeMirror: index_code, codeMirrorOptions: {mode: {name: 'python'}, lineNumbers: true }" type="text" class="input-xlarge" id="index_code"></textarea>
                             </div>
                         </div>
                     </fieldset>
                 </form>
                 <div class="form-actions">
                     <button data-bind='click: open_save_modal, enable: index_code().length' class="btn btn-primary"><i class="icon-ok icon-white"></i> Save</button>
-                    <button class="btn" data-bind="click: insert_plain"><i class="icon-plus"></i> Insert plain</button>
+                    <button class="btn" data-bind="click: insert_plain, visible: is_normal_code"><i class="icon-plus"></i> Insert plain</button>
+                    
+                    <!-- ko ifnot: is_normal_code() -->
+                    <button class="btn" id="btn_switch_new_code" data-bind="click: switch_new_code"><i class="icon-share-alt"></i> Python code</button>
+                    <a class="btn" href="http://labs.codernity.com/codernitydb/database_indexes.html#easier-way-of-creating-indexes" target="_blank"><i class="icon-question-sign"></i> Syntax Help</a>
+                    <!-- /ko -->
+                    <!-- ko if: is_normal_code() -->
+                    <button class="btn" id="btn_switch_new_code" data-bind="click: switch_new_code"><i class="icon-share-alt"></i> Simplified code</button>
+                    <!-- /ko -->
                 </div>
             </div>
             <!-- newIndex -->
             <!-- singleIndexDetails -->
             <div data-bind="with: singleIndexDetails">
                 <!-- modal -->
-                <div class="modal hide fade" id="save_edit_plain_modal">
+                <div class="modal hide fade" tabindex="-1" id="save_edit_plain_modal">
                     <div class="modal-header">
                         <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
                         <h3>Save index</h3>
                     <div class="btn-group left">
                         <button class="btn" data-bind="click: view"><i class="icon-zoom-in"></i> View</button>
                         <button class="btn" data-bind="click: toggle_code_details"><i class="icon-eye-open"></i> <span data-bind="text: code_details_text()"></span></button>
+                        <!-- ko ifnot: show_code() -->
                         <button class="btn" data-bind="click: compact"><i class="icon-repeat"></i> Compact</button>
                         <button class="btn" data-bind="click: count_all"><i class="icon-info-sign"></i> Count</button>
+                        <!-- /ko -->
+                        <!-- ko if: show_code() -->
+                            <!-- ko ifnot: is_normal_code() -->
+                                <button class="btn" id="btn_switch_code" data-bind="click: switch_code"><i class="icon-share-alt"></i> Python code</button>
+                                <a class="btn" href="http://labs.codernity.com/codernitydb/database_indexes.html#easier-way-of-creating-indexes" target="_blank"><i class="icon-question-sign"></i> Syntax Help</a>
+                            <!-- /ko -->
+                            <!-- ko if: is_normal_code() -->
+                                <button class="btn" id="btn_switch_code" data-bind="click: switch_code"><i class="icon-share-alt"></i> Simplified code</button>
+                            <!-- /ko -->
+                        <!-- /ko -->
                     </div>
                     <div class="btn-group right" data-bind="if: name != 'id'">
                         <!-- ko if: show_code() -->
                     <li class="active" data-bind="text: _id"></li>
                 </ul>
                 <!-- /ko -->
-                <div class="modal hide" id="delete_doc_modal">
+                <div class="modal hide" tabindex="-1" id="delete_doc_modal">
                     <div class="modal-header">
                         <a class="close" data-dismiss="modal">x</a>
                         <h3>Are you sure to delete?</h3>
                     <fieldset>
                         <div class="control-group">
                             <div class="controls">
-                                <textarea style="width: 100%; height: 200px;" data-bind="codeMirror: document_data, codeMirrorOptions: {mode: {name: 'javascript', json: true}, lineNumbers: true }" type="text" class="input-xlarge" id="document_content"></textarea>
+                            <textarea style="width: 100%; height: 200px;" data-bind="codeMirror: document_data, codeMirrorOptions: {mode: {name: 'javascript', json: true}, lineNumbers: true }" type="text" class="input-xlarge" id="document_content"></textarea>
                             </div>
                         </div>
                     </fieldset>
             </footer>
         </div>
 
+        <div id="notifier" tabindex="-1" class="modal hide fade">
+            <div class="modal-header">
+                <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
+                <h3></h3>
+            </div>
+            <div class="modal-body">
+                <pre></pre>
+            </div>
+            <div class="modal-footer">
+                <a class="btn" data-dismiss="modal">Cancel</a>
+            </div>
+        </div>
+
         <script src="/_ui/scripts/lib/jquery-1.7.2.min.js" type="text/javascript"></script>
         <script src="/_ui/bootstrap/js/bootstrap-dropdown.js"></script>
         <script src="/_ui/bootstrap/js/bootstrap-modal.js"></script>

CodernityDBHTTP/interface/scripts/app.js

 
 (function() {
 
+    var my_cms = {};
+    var error_marker = null;
     $.ajaxSetup({
         "contentType": "application/json",
     })
         console.log(event)
         console.log('ajaxSettings:')
         console.log(ajaxSettings)
+        
+        if(jqXHR.status == 400 && (ajaxSettings['url'] == '/add_index' || ajaxSettings['url'] == '/edit_index')) {
+            var ls = $.parseJSON(jqXHR.responseText)['reason']
+            ls = RegExp('\\(in line: ([0-9]+)\\)').exec(ls)
+            if (error_marker) {
+                error_marker.clean()
+            }
+            if (ls){
+                error_marker = my_cms['index_code'].markText({line : ls[1]-1,ch : 0},{line : ls[1]-1,ch : 99999},"cm-error")
+            }
+        }
     })
 
     $('#indexDetailsNavigation').live('click', function(){
     });
 
     ko.bindingHandlers.codeMirror = function() {
-        var my_cm = null;
         return {
             init: function(element, valueAccessor, allBindingsAccessor, context) {
-                var options = allBindingsAccessor().codeMirrorOptions || {}
-                var modelValue = valueAccessor()
-                var value = ko.utils.unwrapObservable(valueAccessor())
-                var el = $(element)
+              var options = allBindingsAccessor().codeMirrorOptions || {}
+              var modelValue = valueAccessor()
+              var value = ko.utils.unwrapObservable(valueAccessor())
+              var el = $(element)
 
-                setTimeout(function() {
-                    options.onChange = function(from, to, text, next) {
-                        if (ko.isWriteableObservable(modelValue)) {
-                            modelValue(my_cm.getValue())
-                        }
-                    }
-                    my_cm = CodeMirror.fromTextArea(element, options)
-                    if(value) {
-                        my_cm.setValue(value)
-                    }
-                }, 0)
+              setTimeout(function() {
+                      options.onChange = function(from, to, text, next) {
+                      if (ko.isWriteableObservable(modelValue)) {
+                      modelValue(my_cms[el.attr('id')].getValue())
+                      }
+                      }
+                      my_cms[el.attr('id')] = CodeMirror.fromTextArea(element, options)
 
-                ko.utils.domNodeDisposal.addDisposeCallback(element, function () {
-                    setTimeout(function(){
-                        my_cm.toTextArea()
-                    }, 0)
-                });
+                      if(value) {
+                      my_cms[el.attr('id')].setValue(value)
+                      }
+                      }, 0)
 
-            }
-            , update: function(element, valueAccessor) {
-                var el = $(element)
-                var value = ko.utils.unwrapObservable(valueAccessor())
-                var id = el.attr('id')
-                var update = function(val) {
-                    var update_tries = 0
-                    function later() {
-                        setTimeout(function() {
-                            updater()
-                        }, 5)
-                    }
-                    function updater() {
-                        if(!my_cm) {
-                            update_tries++
-                            if (update_tries < 5) {
-                                later()
-                            } else {
-                                console.log("Failed to activate codemirror")
-                            }
-                        } else {
-                            if (val) {
-                                if(val != my_cm.getValue()) {
-                                    my_cm.setValue(val)
-                                }
-                            }
-                        }
-                    }
-                    return updater
-                }
-                if (id !== undefined && id !== '') {
-                    var upd = update(value)
-                    upd()
-                }
-            }
+          ko.utils.domNodeDisposal.addDisposeCallback(element, function () {
+                  setTimeout(function(){
+                      my_cms[el.attr('id')].toTextArea()
+                      }, 0)
+                  });
+
+      }
+      , update: function(element, valueAccessor) {
+          var el = $(element)
+              var value = ko.utils.unwrapObservable(valueAccessor())
+              var id = el.attr('id')
+              var update = function(val) {
+                  var update_tries = 0
+                      function later() {
+                          setTimeout(function() {
+                                  updater()
+                                  }, 5)
+                      }
+                  function updater() {
+                      if(!my_cms[el.attr('id')]) {
+                          update_tries++
+                              if (update_tries < 5) {
+                                  later()
+                              } else {
+                                  console.log("Failed to activate codemirror")
+                              }
+                      } else {
+                          if (val != null) {
+                              if(val != my_cms[el.attr('id')].getValue()) {
+                                  my_cms[el.attr('id')].setValue(val)
+                              }
+                          }
+                          if (error_marker) {
+                              error_marker.clear()
+                              error_marker = null
+                          }
+                      }
+                  }
+                  return updater
+              }
+          if (id !== undefined && id !== '') {
+              var upd = update(value)
+                  upd()
+          }
+      }
         }
     }()
 
         self.show_code = ko.observable(null)
         self.class_names = ko.observableArray([])
         self.class_name = ko.observable('')
+        self.is_normal_code = ko.observable(true)
+         
+        if (name == null) {
+            self.simplified_code = ""
+            self.python_code = ""
+        }
 
         self.validation = new kk.Validation( self, {
             form_name : [   new kk.validators.Required("Enter name"),
 
         self.open_save_modal = function() {
             // look for classes in index code
-            var classes = self.index_code().match(/^class\s+[a-zA-Z_]\w*[\(|:]/gm)
+            if (self.is_normal_code()) {
+                var classes = self.index_code().match(/^class\s+[a-zA-Z_]\w*[\(|:]/gm)
 
-            if(!classes) {
-                application.add_error("Can't find any index class in your code.")
-                return
+                    if(!classes) {
+                        application.add_error("Can't find any index class in your code.")
+                            return
+                    }
+
+                // ['class Klass(', ['class Klass2:']] => ['Klass', 'Klass2']
+                for (var i = 0; i < classes.length; i++) classes[i] = classes[i].split(' ')[1].slice(0, -1)
+                    classes = ko.utils.arrayGetDistinctValues(classes)
+                        self.class_names(classes)
+                        self.class_name(classes[0])
+
+                $("#save_plain_modal").modal('show')
+            } else {
+                self._create()
             }
-
-            // ['class Klass(', ['class Klass2:']] => ['Klass', 'Klass2']
-            for (var i = 0; i < classes.length; i++) classes[i] = classes[i].split(' ')[1].slice(0, -1)
-            classes = ko.utils.arrayGetDistinctValues(classes)
-            self.class_names(classes)
-            self.class_name(classes[0])
-
-            $("#save_plain_modal").modal('show')
         }
 
 
         self.open_save_edit_modal = function() {
-            // look for classes in index code
-            var classes = self.index_code().match(/^class\s+[a-zA-Z_]\w*[\(|:]/gm)
+            if (self.is_normal_code()) {
+                // look for classes in index code
+                var classes = self.index_code().match(/^class\s+[a-zA-Z_]\w*[\(|:]/gm)
 
-            if(!classes) {
-                application.add_error("Can't find any index class in your code.")
-                return
+                    if(!classes) {
+                        application.add_error("Can't find any index class in your code.")
+                            return
+                    }
+
+                // ['class Klass(', ['class Klass2:']] => ['Klass', 'Klass2']
+                for (var i = 0; i < classes.length; i++) classes[i] = classes[i].split(' ')[1].slice(0, -1)
+                    classes = ko.utils.arrayGetDistinctValues(classes)
+                        self.class_names(classes)
+                        self.class_name(classes[0])
+
+                $("#save_edit_plain_modal").modal('show')
+            } else {
+                self._edit()
             }
+        }
 
-            // ['class Klass(', ['class Klass2:']] => ['Klass', 'Klass2']
-            for (var i = 0; i < classes.length; i++) classes[i] = classes[i].split(' ')[1].slice(0, -1)
-            classes = ko.utils.arrayGetDistinctValues(classes)
-            self.class_names(classes)
-            self.class_name(classes[0])
-
-            $("#save_edit_plain_modal").modal('show')
+        self.switch_new_code = function () {
+            if (self.is_normal_code()) {
+                self.python_code = self.index_code()
+                self.index_code(self.simplified_code)
+            } else {
+                self.simplified_code = self.index_code()
+                self.index_code(self.python_code)
+            }
+            self.is_normal_code(!self.is_normal_code())
         }
 
         self.save = function() {
 
             var code_header = "# " + self.form_name() + "\n"
             code_header += "# " + self.class_name() + '\n'
-            code_header += header_footer + "\n\n"
+            code_header += header_footer + "\n"
             var code = ''
-
             if(self.index_code().split('\n')[2].indexOf(header_footer) == 0)
                 code = self.index_code().split('\n').splice(4).join('\n')
             else
         self.insert_plain = function(){
             $.get('/get_index_header')
             .success(function(data) {
+                    /*console.log(data)*/
                 lines = data.split('\n')
+                /*console.log(self.index_code())*/
                 self.index_code(lines.slice(3).join('\n'))
             })
         }
             }
         })
 
+        self.is_normal_code.subscribe(function(newValue){
+            if(my_cms['index_code']){
+                my_cms['index_code'].setOption('mode',self.is_normal_code() ? 'python' : 'text')
+            }
+        })
+        
         self.get_index_code = function() {
-            $.get('/get_index_code/' + self.name, {},
-                function(data) {
+            $.get('/get_index_code/' + self.name ,{code_switch : JSON.stringify("S")}, function(data) {
+                if(!self.is_normal_code()){
                     self.index_code(data)
-                    if(application.index_changed != null) {
-                        $('button.btn-reindex').popover({"placement": "top", title: application.index_changed}).popover('show')
-                        application.index_changed = null
-                    }
                 }
-            )
+                self.simplified_code = data
+                })
+            $.get('/get_index_code/' + self.name,{code_switch : JSON.stringify("P")}, function(data) {
+                if (self.is_normal_code()){
+                    self.index_code(data)
+                }
+                self.python_code = data
+                })
+
+            if(application.index_changed != null) {
+                $('button.btn-reindex').popover({"placement": "top", title: application.index_changed}).popover('show')
+                application.index_changed = null
+            }
         }
 
         self.get_index_template = function() {
                 self.show_code(!self.show_code())
                 if(!self.show_code()) {
                     self.get_details()
+                } else { // XXX: fix it
+                    setTimeout(function() {
+                        my_cms['index_code'].setOption('mode',self.is_normal_code() ? 'python' : 'text')}, 0)
                 }
             }
         }
+        
+
+        self.switch_code = function () {
+            if (self.is_normal_code()) {
+                $.get('/get_index_code/' + self.name ,{code_switch : JSON.stringify("S")}, function(data) {
+                    self.simplified_code = data 
+                    self.index_code(data)
+                    })
+            } else {
+                $.get('/get_index_code/' + self.name ,{code_switch : JSON.stringify("P")}, function(data) {
+                    self.python_code = data 
+                    self.index_code(data)
+                    })
+            }
+            self.is_normal_code(!self.is_normal_code())
+        }
 
         self.code_details_text = function() {
             if (!self.show_code()) {
                 this.before({}, function() {
                     // Dropdown clearMenus() is not called when we using Sammy.js...
                     $('[data-toggle="dropdown"]').parent().removeClass('open')
+                    $('div.popover').remove()
                     self.has_indexes_names()
                     self.clean_iface()
                 })
     application = new DBViewModel()
 
     ko.applyBindings(application)
-})()
+})()

CodernityDBHTTP/interface/scripts/lib/notifier.js

+// https://github.com/Srirangan/notifer.js/
+// for author and details
+// this version is a bit changed for CodernityDB-HTTP needs
+
 (function($){
 
   var config = window.NotifierjsConfig = {
     Notifier.notify(message, title, "%3D");
   };
   Notifier.error = function(message, title) {
-    Notifier.notify(message, title, "%3D%3D");
+      title = title ? title : 'An error occurred';
+      $('#notifier h3').html(title);
+      $('#notifier pre').html(message);
+      $('#notifier').modal('show');
   };
   Notifier.success = function(message, title) {
     Notifier.notify(message, title, "%3D");

CodernityDBHTTP/misc.py

 from os import path as os_path
 from os import access as os_access
 from os import R_OK
+from os import stat
 import sys
 from all_exceptions import DBHTTPException, HTTPDatabaseException, HTTPNotFoundException, HTTPConflictException, DBIsNotOpened
 from types import FunctionType, MethodType, GeneratorType
 from CodernityDB.database import RevConflict, RecordDeleted, RecordNotFound
 import httplib
 
+try:
+    import ujson as json
+except ImportError:
+    try:
+        import simplejson as json
+    except ImportError:
+        try:
+            import json
+        except ImportError:
+            # rly ?
+            has_json = False
+    else:
+        has_json = True
+else:
+    has_json = True
 
 import traceback
 
     'DatabasePathException': 409,
     'DatabaseIsNotOpened': 409,
     'TryReindexException': 409,
-    'ReindexException': 409
+    'ReindexException': 409,
+    'IndexCreatorValueException': 400,
+    'IndexCreatorFunctionException': 400
 }
 
 
                     '.', 1)[1], content_types['default'])
             except:
                 content_type = content_types['default']
+            f_size = stat(f_path).st_size
             with open(f_path, 'rb') as fp:
-                start_response("200 OK", [('Content-Type', content_type)])
+                start_response("200 OK", [('Content-Type', content_type),
+                                          ('Content-Length', str(f_size)),
+                                          ('Content-Encoding', 'chunked')])
                 chunk = fp.read(4096)
                 while chunk:
                     yield chunk
     else:
         error = "%s %s" % (500, httplib.responses[500])
         with_traceback = True
+
     exception_output = {'exception': exc_type.__name__,
                         'reason': reason,
-                        'error': error}
+                        'error': error
+                        }
     if with_traceback:
+
+
+def open_with_autosave(db, cfg_path, cfg):
+    def _inner(*args, **kwargs):
+        db.open(*args, **kwargs)
+        if cfg.get('auto_save') and cfg.get('db_path') != db.path:
+            cfg['db_path'] = db.path
+            with open(cfg_path, 'w') as f:
+                f.write(json.dumps(cfg, sort_keys=True, indent=4))
+    return _inner
         tb = traceback.format_exception(exc_type, exc_value, exc_traceback)
         exception_output['traceback'] = ''.join(tb)
         sys.stderr.writelines(tb)  # print the traceback to console

CodernityDBHTTP/server.py

         from server_wsgiref import create_server
     else:
         raise NotImplemented("Not supported server method")
-    s = create_server(bind, port, db_path, cfg)
+    s = create_server(bind, port, db_path, cfg, path)
     del create_server
     cfg['port'] = s.server_port
     return s, cfg

CodernityDBHTTP/server_gevent.py

         super(CodernityDBGeventWSGIServer, self).__init__(*args, **kwargs)
 
 
-def create_server(bind, port, db_path, cfg, **kwargs):
+def create_server(bind, port, db_path, cfg, cfg_path, **kwargs):
     from gevent import socket
     sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
     sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
     backlog = kwargs.get('backlog', 50)
     sock.listen(backlog)
     s = CodernityDBGeventWSGIServer(
-        sock, get_app(HTTPGeventDatabase, db_path, cfg))
+        sock, get_app(HTTPGeventDatabase, db_path, cfg, cfg_path))
     return s

CodernityDBHTTP/server_wsgiref.py

 
 #         #self.base_env['SERVER_SOFTWARE'] = 'CodernityDB(%s) @ wsgiref' % __db__version__
 
-def create_server(bind, port, db_path, cfg, **kwargs):
-    app = get_app(HTTPThreadSafeDatabase, db_path, cfg)
+def create_server(bind, port, db_path, cfg, cfg_path, **kwargs):
+    app = get_app(HTTPThreadSafeDatabase, db_path, cfg, cfg_path)
     s = make_server(bind, port, app)
     return s