lanes 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +19 -0
- data/.jshintrc +3 -0
- data/Gemfile +9 -0
- data/Guardfile +15 -0
- data/LICENSE-MIT.txt +21 -0
- data/README.md +15 -0
- data/Rakefile +74 -0
- data/bin/lanes +5 -0
- data/client/images/ajax-loader.gif +0 -0
- data/client/images/dataTables/Sorting icons.psd +0 -0
- data/client/images/dataTables/back_disabled.png +0 -0
- data/client/images/dataTables/back_enabled.png +0 -0
- data/client/images/dataTables/back_enabled_hover.png +0 -0
- data/client/images/dataTables/favicon.ico +0 -0
- data/client/images/dataTables/forward_disabled.png +0 -0
- data/client/images/dataTables/forward_enabled.png +0 -0
- data/client/images/dataTables/forward_enabled_hover.png +0 -0
- data/client/images/dataTables/loading-background.png +0 -0
- data/client/images/dataTables/sort_asc.png +0 -0
- data/client/images/dataTables/sort_asc_disabled.png +0 -0
- data/client/images/dataTables/sort_both.png +0 -0
- data/client/images/dataTables/sort_desc.png +0 -0
- data/client/images/dataTables/sort_desc_disabled.png +0 -0
- data/client/images/logo-sm.png +0 -0
- data/client/javascripts/component/Base.coffee +39 -0
- data/client/javascripts/component/ChoicesInput.coffee +47 -0
- data/client/javascripts/component/Grid.coffee +199 -0
- data/client/javascripts/component/ModalDialog.coffee +44 -0
- data/client/javascripts/component/PopOver.coffee +52 -0
- data/client/javascripts/component/RadioGroup.coffee +59 -0
- data/client/javascripts/component/RecordFinder.coffee +143 -0
- data/client/javascripts/component/SelectField.coffee +43 -0
- data/client/javascripts/component/TaggedField.coffee +27 -0
- data/client/javascripts/component/grid/Editor.coffee +65 -0
- data/client/javascripts/component/grid/PopOverEditor.coffee +29 -0
- data/client/javascripts/component/grid/RowEditor.coffee +31 -0
- data/client/javascripts/component/grid/popover-editor.html +18 -0
- data/client/javascripts/component/grid/row-editor.html +16 -0
- data/client/javascripts/component/grid.html +4 -0
- data/client/javascripts/component/index.js +5 -0
- data/client/javascripts/component/modal.html +17 -0
- data/client/javascripts/component/popover.html +5 -0
- data/client/javascripts/component/record-finder/clause.skr +35 -0
- data/client/javascripts/component/record-finder/dialog.skr +4 -0
- data/client/javascripts/component/record-finder/field.skr +8 -0
- data/client/javascripts/data/Bootstrap.coffee +8 -0
- data/client/javascripts/data/ChangeSet.coffee +50 -0
- data/client/javascripts/data/Collection.coffee +111 -0
- data/client/javascripts/data/Config.coffee +15 -0
- data/client/javascripts/data/Model.coffee +269 -0
- data/client/javascripts/data/PubSub.coffee +68 -0
- data/client/javascripts/data/Query.coffee +184 -0
- data/client/javascripts/data/Roles.coffee +91 -0
- data/client/javascripts/data/Screens.coffee +157 -0
- data/client/javascripts/data/Sync.coffee +62 -0
- data/client/javascripts/data/User.coffee +70 -0
- data/client/javascripts/data/index.js +7 -0
- data/client/javascripts/data/mixins/HasCodeField.coffee +13 -0
- data/client/javascripts/extension/Base.coffee +9 -0
- data/client/javascripts/extension/Extensions.coffee +17 -0
- data/client/javascripts/extension/GlAccounts.coffee +9 -0
- data/client/javascripts/extension/index.js +6 -0
- data/client/javascripts/extension/load.js.erb +3 -0
- data/client/javascripts/lanes-complete.js +3 -0
- data/client/javascripts/lanes-workspace.js +1 -0
- data/client/javascripts/lib/MakeBaseClass.coffee +55 -0
- data/client/javascripts/lib/ModuleSupport.coffee +22 -0
- data/client/javascripts/lib/Templates.coffee +47 -0
- data/client/javascripts/lib/create-namespace.js +0 -0
- data/client/javascripts/lib/debounce.coffee +15 -0
- data/client/javascripts/lib/defer.coffee +7 -0
- data/client/javascripts/lib/el.js +115 -0
- data/client/javascripts/lib/index.js +9 -0
- data/client/javascripts/lib/loader.coffee +95 -0
- data/client/javascripts/lib/namespace.coffee +9 -0
- data/client/javascripts/lib/noConflict.coffee +14 -0
- data/client/javascripts/lib/promise_helpers.coffee +4 -0
- data/client/javascripts/lib/results.coffee +15 -0
- data/client/javascripts/lib/underscore.inflection.js +210 -0
- data/client/javascripts/lib/utilFunctions.coffee +51 -0
- data/client/javascripts/plugins/ResizeSensor.js +144 -0
- data/client/javascripts/plugins/index.js +4 -0
- data/client/javascripts/plugins/overlay.coffee +41 -0
- data/client/javascripts/plugins/trigger.coffee +15 -0
- data/client/javascripts/vendor/bootstrap/affix.js +142 -0
- data/client/javascripts/vendor/bootstrap/alert.js +92 -0
- data/client/javascripts/vendor/bootstrap/button.js +110 -0
- data/client/javascripts/vendor/bootstrap/carousel.js +223 -0
- data/client/javascripts/vendor/bootstrap/collapse.js +170 -0
- data/client/javascripts/vendor/bootstrap/dropdown.js +151 -0
- data/client/javascripts/vendor/bootstrap/modal.js +280 -0
- data/client/javascripts/vendor/bootstrap/popover.js +113 -0
- data/client/javascripts/vendor/bootstrap/scrollspy.js +170 -0
- data/client/javascripts/vendor/bootstrap/tab.js +128 -0
- data/client/javascripts/vendor/bootstrap/tooltip.js +457 -0
- data/client/javascripts/vendor/bootstrap/transition.js +59 -0
- data/client/javascripts/vendor/dataTables/dataTables.bootstrap.js +156 -0
- data/client/javascripts/vendor/dataTables/dataTables.scroller.js +1185 -0
- data/client/javascripts/vendor/dataTables/datatables.responsive.js +666 -0
- data/client/javascripts/vendor/dataTables/index.js +2 -0
- data/client/javascripts/vendor/dataTables/jquery.dataTables.js +14380 -0
- data/client/javascripts/vendor/jquery-2.js +9190 -0
- data/client/javascripts/vendor/jquery.tap.js +401 -0
- data/client/javascripts/vendor/magicsuggest.js +1565 -0
- data/client/javascripts/vendor/message-bus.js +285 -0
- data/client/javascripts/vendor/modern-stack.js +14 -0
- data/client/javascripts/vendor/packaged.js +13769 -0
- data/client/javascripts/view/Assets.coffee +9 -0
- data/client/javascripts/view/Base.coffee +231 -0
- data/client/javascripts/view/FormBindings.coffee +98 -0
- data/client/javascripts/view/Functions.coffee +13 -0
- data/client/javascripts/view/Helpers.coffee +77 -0
- data/client/javascripts/view/InterfaceState.coffee +88 -0
- data/client/javascripts/view/Keys.coffee +59 -0
- data/client/javascripts/view/ModelObserver.coffee +31 -0
- data/client/javascripts/view/ModelUpdate.coffee +8 -0
- data/client/javascripts/view/PubSub.coffee +29 -0
- data/client/javascripts/view/RenderContext.coffee +32 -0
- data/client/javascripts/view/SaveNotify.coffee +30 -0
- data/client/javascripts/view/Screen.coffee +30 -0
- data/client/javascripts/view/TimedHighlight.coffee +38 -0
- data/client/javascripts/view/TimedMask.coffee +65 -0
- data/client/javascripts/view/_button.html +3 -0
- data/client/javascripts/view/_toolbar.html +27 -0
- data/client/javascripts/view/index.js +10 -0
- data/client/javascripts/view/mixins/ScreenChangeListener.coffee +43 -0
- data/client/javascripts/view/model-update.html +13 -0
- data/client/javascripts/view/screen-definitions.js.erb +7 -0
- data/client/javascripts/workspace/ActiveScreensSwitcher.coffee +117 -0
- data/client/javascripts/workspace/Instance.es6 +60 -0
- data/client/javascripts/workspace/Layout.coffee +18 -0
- data/client/javascripts/workspace/LoginDialog.coffee +33 -0
- data/client/javascripts/workspace/Navbar.coffee +44 -0
- data/client/javascripts/workspace/Pages.coffee +46 -0
- data/client/javascripts/workspace/ScreensMenu.coffee +126 -0
- data/client/javascripts/workspace/index.js +12 -0
- data/client/javascripts/workspace/layout.html +4 -0
- data/client/javascripts/workspace/login-dialog.html +16 -0
- data/client/javascripts/workspace/menu.html +356 -0
- data/client/javascripts/workspace/menu_toggle.html +9 -0
- data/client/javascripts/workspace/navbar.html +19 -0
- data/client/javascripts/workspace/pages.html +6 -0
- data/client/javascripts/workspace/screens-menu.html +11 -0
- data/client/javascripts/workspace/screens-switcher.html +7 -0
- data/client/javascripts/workspace/tab.html +0 -0
- data/client/screens/user-management/UserEditScreen.coffee +21 -0
- data/client/screens/user-management/UserManagement.coffee +24 -0
- data/client/screens/user-management/grid-popover-editor.html +33 -0
- data/client/screens/user-management/index.css +4 -0
- data/client/screens/user-management/index.js +2 -0
- data/client/screens/user-management/user-management-styles.scss +7 -0
- data/client/screens/user-management/user-management.html +8 -0
- data/client/stylesheets/compoonents/all.scss +6 -0
- data/client/stylesheets/compoonents/changes-notification.scss +44 -0
- data/client/stylesheets/compoonents/grid-editors.scss +65 -0
- data/client/stylesheets/compoonents/grid.scss +301 -0
- data/client/stylesheets/compoonents/modal-dialog.scss +23 -0
- data/client/stylesheets/compoonents/record-finder.scss +71 -0
- data/client/stylesheets/compoonents/suggest.scss +266 -0
- data/client/stylesheets/fonts/icomoon.eot +0 -0
- data/client/stylesheets/fonts/icomoon.svg +160 -0
- data/client/stylesheets/fonts/icomoon.ttf +0 -0
- data/client/stylesheets/fonts/icomoon.woff +0 -0
- data/client/stylesheets/fonts/selection.json +3565 -0
- data/client/stylesheets/fonts/style.css +451 -0
- data/client/stylesheets/fonts.scss +38 -0
- data/client/stylesheets/forms.scss +75 -0
- data/client/stylesheets/index.css +4 -0
- data/client/stylesheets/keybindings.scss +6 -0
- data/client/stylesheets/lanes-workspace.scss +17 -0
- data/client/stylesheets/layout.scss +272 -0
- data/client/stylesheets/plugins/all.scss +2 -0
- data/client/stylesheets/plugins/overlay.scss +63 -0
- data/client/stylesheets/plugins/resize-sensor.scss +24 -0
- data/client/stylesheets/screens.scss +66 -0
- data/client/stylesheets/tabs.scss +148 -0
- data/client/stylesheets/vendor/bootstrap/_alerts.scss +68 -0
- data/client/stylesheets/vendor/bootstrap/_badges.scss +57 -0
- data/client/stylesheets/vendor/bootstrap/_breadcrumbs.scss +26 -0
- data/client/stylesheets/vendor/bootstrap/_button-groups.scss +240 -0
- data/client/stylesheets/vendor/bootstrap/_buttons.scss +157 -0
- data/client/stylesheets/vendor/bootstrap/_carousel.scss +243 -0
- data/client/stylesheets/vendor/bootstrap/_close.scss +35 -0
- data/client/stylesheets/vendor/bootstrap/_code.scss +68 -0
- data/client/stylesheets/vendor/bootstrap/_component-animations.scss +35 -0
- data/client/stylesheets/vendor/bootstrap/_dropdowns.scss +215 -0
- data/client/stylesheets/vendor/bootstrap/_forms.scss +538 -0
- data/client/stylesheets/vendor/bootstrap/_glyphicons.scss +237 -0
- data/client/stylesheets/vendor/bootstrap/_grid.scss +84 -0
- data/client/stylesheets/vendor/bootstrap/_input-groups.scss +166 -0
- data/client/stylesheets/vendor/bootstrap/_jumbotron.scss +48 -0
- data/client/stylesheets/vendor/bootstrap/_labels.scss +66 -0
- data/client/stylesheets/vendor/bootstrap/_list-group.scss +132 -0
- data/client/stylesheets/vendor/bootstrap/_media.scss +56 -0
- data/client/stylesheets/vendor/bootstrap/_mixins.scss +39 -0
- data/client/stylesheets/vendor/bootstrap/_modals.scss +150 -0
- data/client/stylesheets/vendor/bootstrap/_navbar.scss +659 -0
- data/client/stylesheets/vendor/bootstrap/_navs.scss +242 -0
- data/client/stylesheets/vendor/bootstrap/_normalize.scss +425 -0
- data/client/stylesheets/vendor/bootstrap/_pager.scss +55 -0
- data/client/stylesheets/vendor/bootstrap/_pagination.scss +88 -0
- data/client/stylesheets/vendor/bootstrap/_panels.scss +243 -0
- data/client/stylesheets/vendor/bootstrap/_popovers.scss +133 -0
- data/client/stylesheets/vendor/bootstrap/_print.scss +101 -0
- data/client/stylesheets/vendor/bootstrap/_progress-bars.scss +105 -0
- data/client/stylesheets/vendor/bootstrap/_responsive-embed.scss +34 -0
- data/client/stylesheets/vendor/bootstrap/_responsive-utilities.scss +174 -0
- data/client/stylesheets/vendor/bootstrap/_scaffolding.scss +150 -0
- data/client/stylesheets/vendor/bootstrap/_tables.scss +233 -0
- data/client/stylesheets/vendor/bootstrap/_theme.scss +258 -0
- data/client/stylesheets/vendor/bootstrap/_thumbnails.scss +38 -0
- data/client/stylesheets/vendor/bootstrap/_tooltip.scss +95 -0
- data/client/stylesheets/vendor/bootstrap/_type.scss +304 -0
- data/client/stylesheets/vendor/bootstrap/_utilities.scss +57 -0
- data/client/stylesheets/vendor/bootstrap/_variables.scss +850 -0
- data/client/stylesheets/vendor/bootstrap/_wells.scss +29 -0
- data/client/stylesheets/vendor/bootstrap/bootstrap.scss +50 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_alerts.scss +14 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_background-variant.scss +11 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_border-radius.scss +18 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_buttons.scss +50 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_center-block.scss +7 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_clearfix.scss +22 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_forms.scss +84 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_gradients.scss +58 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_grid-framework.scss +81 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_grid.scss +122 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_hide-text.scss +21 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_image.scss +34 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_labels.scss +12 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_list-group.scss +31 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_nav-divider.scss +10 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_nav-vertical-align.scss +9 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_opacity.scss +8 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_pagination.scss +23 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_panels.scss +24 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_progress-bar.scss +10 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_reset-filter.scss +8 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_resize.scss +6 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_responsive-visibility.scss +21 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_size.scss +10 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_tab-focus.scss +9 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_table-row.scss +28 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_text-emphasis.scss +11 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_text-overflow.scss +8 -0
- data/client/stylesheets/vendor/bootstrap/mixins/_vendor-prefixes.scss +219 -0
- data/client/stylesheets/vendor/bootstrap-custom-grid.scss +85 -0
- data/client/stylesheets/vendor/bootstrap-custom-modals.scss +150 -0
- data/client/stylesheets/vendor/bootstrap.scss +69 -0
- data/client/stylesheets/vendor/dataTables.scss +4 -0
- data/config/database.yml +9 -0
- data/config/puma.rb +7 -0
- data/config.ru +4 -0
- data/db/migrate/20140615031600_create_hip_users.rb +17 -0
- data/db/seed.rb +37 -0
- data/foo/Gemfile +5 -0
- data/foo/Guardfile +13 -0
- data/foo/foo/Gemfile +5 -0
- data/foo/foo/lib/foo.rb +7 -0
- data/foo/lib/foo/version.rb +3 -0
- data/foo/lib/foo.rb +8 -0
- data/lanes.gemspec +54 -0
- data/lib/generators/lanes/migrations/install_generator.rb +42 -0
- data/lib/lanes/access/locked_fields.rb +43 -0
- data/lib/lanes/access/role.rb +58 -0
- data/lib/lanes/access/role_collection.rb +75 -0
- data/lib/lanes/access/roles/administrator.rb +25 -0
- data/lib/lanes/access/roles/support.rb +13 -0
- data/lib/lanes/access/user_maint_screen.rb +32 -0
- data/lib/lanes/access.rb +50 -0
- data/lib/lanes/api/asset_pipeline.rb +59 -0
- data/lib/lanes/api/authentication_helper.rb +21 -0
- data/lib/lanes/api/authentication_provider.rb +45 -0
- data/lib/lanes/api/controller.rb +290 -0
- data/lib/lanes/api/default_routes.rb +35 -0
- data/lib/lanes/api/eco.js +516 -0
- data/lib/lanes/api/error_formatter.rb +37 -0
- data/lib/lanes/api/helper_methods.rb +32 -0
- data/lib/lanes/api/javascript_processor.rb +116 -0
- data/lib/lanes/api/pub_sub.rb +33 -0
- data/lib/lanes/api/request_wrapper.rb +42 -0
- data/lib/lanes/api/root.rb +103 -0
- data/lib/lanes/api/skr_templates.rb +60 -0
- data/lib/lanes/api/test_specs.rb +59 -0
- data/lib/lanes/api/updates.rb +38 -0
- data/lib/lanes/api.rb +27 -0
- data/lib/lanes/cli.rb +13 -0
- data/lib/lanes/concerns/all.rb +16 -0
- data/lib/lanes/concerns/api_path.rb +21 -0
- data/lib/lanes/concerns/association_extensions.rb +85 -0
- data/lib/lanes/concerns/attr_accessor_with_default.rb +62 -0
- data/lib/lanes/concerns/code_identifier.rb +43 -0
- data/lib/lanes/concerns/export_associations.rb +52 -0
- data/lib/lanes/concerns/export_join_tables.rb +39 -0
- data/lib/lanes/concerns/export_methods.rb +104 -0
- data/lib/lanes/concerns/export_scope.rb +66 -0
- data/lib/lanes/concerns/exported_limit_evaluator.rb +17 -0
- data/lib/lanes/concerns/immutable_model.rb +32 -0
- data/lib/lanes/concerns/locked_fields.rb +84 -0
- data/lib/lanes/concerns/pub_sub.rb +105 -0
- data/lib/lanes/concerns/queries.rb +20 -0
- data/lib/lanes/concerns/random_hash_code.rb +40 -0
- data/lib/lanes/concerns/sanitize_api_data.rb +15 -0
- data/lib/lanes/concerns/set_attribute_data.rb +154 -0
- data/lib/lanes/concerns/track_modifications.rb +51 -0
- data/lib/lanes/concerns/visible_id_identifier.rb +53 -0
- data/lib/lanes/configuration.rb +85 -0
- data/lib/lanes/db/migration_helpers.rb +178 -0
- data/lib/lanes/db/migrations.rb +13 -0
- data/lib/lanes/db/seed.rb +27 -0
- data/lib/lanes/db.rb +86 -0
- data/lib/lanes/environment.rb +19 -0
- data/lib/lanes/extension.rb +72 -0
- data/lib/lanes/generators/app/Gemfile +5 -0
- data/lib/lanes/generators/app/Guardfile +13 -0
- data/lib/lanes/generators/app/Rakefile +9 -0
- data/lib/lanes/generators/app/config/database.yml +9 -0
- data/lib/lanes/generators/app/config.ru +4 -0
- data/lib/lanes/generators/app/lib/main_class/version.rb +3 -0
- data/lib/lanes/generators/app/lib/main_class.rb +8 -0
- data/lib/lanes/generators/app.rb +36 -0
- data/lib/lanes/guard_tasks.rb +44 -0
- data/lib/lanes/logger.rb +37 -0
- data/lib/lanes/model.rb +26 -0
- data/lib/lanes/numbers.rb +72 -0
- data/lib/lanes/rails_engine.rb +5 -0
- data/lib/lanes/screens.rb +126 -0
- data/lib/lanes/spec_asset_expander.rb +43 -0
- data/lib/lanes/strings.rb +56 -0
- data/lib/lanes/user.rb +127 -0
- data/lib/lanes/validators/all.rb +2 -0
- data/lib/lanes/validators/email.rb +17 -0
- data/lib/lanes/validators/set.rb +18 -0
- data/lib/lanes/version.rb +5 -0
- data/lib/lanes.rb +22 -0
- data/npm-build/README +1 -0
- data/npm-build/compile.coffee +15 -0
- data/npm-build/package.json +59 -0
- data/npm-build/shims/underscore.js +1416 -0
- data/npm-build/template.js +33 -0
- data/public/javascripts/jasmine_examples/Player.js +22 -0
- data/public/javascripts/jasmine_examples/Song.js +7 -0
- data/spec/api/javascript_processor_spec.rb +107 -0
- data/spec/api/user_spec.rb +52 -0
- data/spec/client/component/ChoicesInputSpec.coffee +12 -0
- data/spec/client/component/foo_spec.coffee +4 -0
- data/spec/client/foo_spec.js +0 -0
- data/spec/client/jasmine_examples/PlayerSpec.js +0 -0
- data/spec/client/support/jasmine.yml +128 -0
- data/spec/client/support/jasmine_helper.rb +15 -0
- data/spec/concerns/api_path_spec.rb +14 -0
- data/spec/concerns/association_extensions_spec.rb +30 -0
- data/spec/concerns/attr_accessor_with_default_spec.rb +57 -0
- data/spec/concerns/code_identifier_spec.rb +45 -0
- data/spec/concerns/export_associations_spec.rb +7 -0
- data/spec/concerns/export_methods_spec.rb +43 -0
- data/spec/concerns/export_scope_spec.rb +15 -0
- data/spec/concerns/exported_limits_spec.rb +47 -0
- data/spec/concerns/pub_sub_spec.rb +83 -0
- data/spec/concerns/set_attribute_data_spec.rb +66 -0
- data/spec/configuration_spec.rb +26 -0
- data/spec/fixtures/lanes/users.yml +13 -0
- data/spec/helpers/.gitkeep +0 -0
- data/spec/helpers/SpecHelper.js +18 -0
- data/spec/locked_fields_spec.rb +27 -0
- data/spec/numbers_spec.rb +26 -0
- data/spec/role_collection_spec.rb +19 -0
- data/spec/spec_helper.rb +163 -0
- data/spec/strings_spec.rb +41 -0
- data/spec/testing_models.rb +54 -0
- data/spec/user_role_spec.rb +7 -0
- data/spec/user_spec.rb +53 -0
- data/tasks/migrations.rake +22 -0
- data/tasks/publish.rake +8 -0
- data/views/index.erb +29 -0
- data/views/specs.erb +25 -0
- data/yard_ext/all.rb +9 -0
- data/yard_ext/code_identifier_handler.rb +33 -0
- data/yard_ext/concern_meta_methods.rb +60 -0
- data/yard_ext/config_options.rb +27 -0
- data/yard_ext/exported_scope.rb +4 -0
- data/yard_ext/immutable_handler.rb +17 -0
- data/yard_ext/json_attr_accessor.rb +22 -0
- data/yard_ext/locked_fields_handler.rb +21 -0
- data/yard_ext/templates/default/layout/html/layout.erb +20 -0
- data/yard_ext/templates/default/method_details/html/github_link.erb +1 -0
- data/yard_ext/templates/default/method_details/setup.rb +3 -0
- data/yard_ext/validators.rb +1 -0
- data/yard_ext/visible_id_handler.rb +38 -0
- metadata +772 -0
@@ -0,0 +1,269 @@
|
|
1
|
+
class DataModel
|
2
|
+
isModel: true
|
3
|
+
|
4
|
+
constructor: (attrs,options)->
|
5
|
+
this._unsaved = if attrs then _.keys(attrs) else []
|
6
|
+
super
|
7
|
+
this.on('change', this._recordUnsaved )
|
8
|
+
|
9
|
+
session:
|
10
|
+
errors: 'object'
|
11
|
+
changes: { type: 'collection', setOnce: true }
|
12
|
+
lastServerMessage: { type: 'string' }
|
13
|
+
|
14
|
+
derived:
|
15
|
+
error_message:
|
16
|
+
deps:['errors'], fn: ->
|
17
|
+
if !@errors then ''
|
18
|
+
else if @errors.exception then @errors.exception
|
19
|
+
else _.toSentence( _.map(@errrors, (value,key)-> "#{key}: #{value}" ) )
|
20
|
+
|
21
|
+
dataTypes:
|
22
|
+
bigdec:
|
23
|
+
set: (newVal)->
|
24
|
+
val: new _.bigDecimal(newVal)
|
25
|
+
type: 'bigdec'
|
26
|
+
default: -> new _.bigDecimal(0)
|
27
|
+
integer:
|
28
|
+
set: (newVal)->
|
29
|
+
val: parseInt(newVal)
|
30
|
+
type: 'integer'
|
31
|
+
date:
|
32
|
+
get: (val)-> new Date(val)
|
33
|
+
default: -> return new Date()
|
34
|
+
set: (newVal)->
|
35
|
+
if _.isDate(newVal)
|
36
|
+
newType='date'
|
37
|
+
newVal = newVal.valueOf();
|
38
|
+
else
|
39
|
+
m=Lanes.Vendor.Moment(newVal)
|
40
|
+
if m.isValid()
|
41
|
+
newType='date'
|
42
|
+
newVal=m.toDate()
|
43
|
+
else
|
44
|
+
newType = typeof newVal;
|
45
|
+
return {
|
46
|
+
val: newVal,
|
47
|
+
type: newType
|
48
|
+
}
|
49
|
+
|
50
|
+
modelForAccess: -> this
|
51
|
+
|
52
|
+
isPersistent: ->
|
53
|
+
!!( this.api_path && ! this.isNew() )
|
54
|
+
|
55
|
+
addChangeSet: (change)->
|
56
|
+
this.changes ||= new Lanes.Data.ChangeSetCollection( parent: this )
|
57
|
+
change.record = this
|
58
|
+
change = this.changes.add(change)
|
59
|
+
this.set( change.value() )
|
60
|
+
|
61
|
+
rootRecord: ->
|
62
|
+
record = this.parent
|
63
|
+
while record
|
64
|
+
if record.parent then record = record.parent else break
|
65
|
+
record
|
66
|
+
|
67
|
+
urlRoot: ->
|
68
|
+
Lanes.Data.Config.api_path + '/' + @resultsFor('api_path')
|
69
|
+
|
70
|
+
setupStandardProps: -> Lanes.emptyFn
|
71
|
+
|
72
|
+
isLoaded: ->
|
73
|
+
!_.isEmpty( _.omit(this.attributes,this.idAttribute) )
|
74
|
+
|
75
|
+
withAssociations: (names...,options={})->
|
76
|
+
scope = options.scope || this
|
77
|
+
if _.isString(options)
|
78
|
+
names.push(options); options={}
|
79
|
+
needed = _.filter( names, (name)->
|
80
|
+
this._associations[name] && ! this._associations[name].instance(this).isLoaded()
|
81
|
+
,this)
|
82
|
+
if _.isEmpty( needed )
|
83
|
+
options.success.call(scope, this ) if options.success
|
84
|
+
options.complete.call(scope,this ) if options.complete
|
85
|
+
return Lanes.Promise.resolve(this)
|
86
|
+
else
|
87
|
+
options['include']=needed
|
88
|
+
this.fetch(options)
|
89
|
+
.then (req)-> req.record
|
90
|
+
|
91
|
+
@findOrCreate: (attrs, options={})->
|
92
|
+
if attrs.id && ( record = Lanes.Data.PubSub.instanceFor(this, attrs.id) )
|
93
|
+
record.set(attrs)
|
94
|
+
else
|
95
|
+
new this(attrs,options)
|
96
|
+
|
97
|
+
@fetch: (options)->
|
98
|
+
record = new this()
|
99
|
+
if _.isNumber(options)
|
100
|
+
record.id = options
|
101
|
+
options = {}
|
102
|
+
ret = record.fetch(options)
|
103
|
+
ret
|
104
|
+
|
105
|
+
fetch: (options={})->
|
106
|
+
handlers = wrapRequest(this,options)
|
107
|
+
super(_.extend(options,{limit:1,ignoreUnsaved:true})).then( =>@ )
|
108
|
+
handlers.promise
|
109
|
+
|
110
|
+
parse:(resp)->
|
111
|
+
if resp.data
|
112
|
+
if _.isArray(resp['data']) then resp['data'][0] else resp['data']
|
113
|
+
else
|
114
|
+
resp
|
115
|
+
|
116
|
+
set: (attrs,options)->
|
117
|
+
super
|
118
|
+
if _.isObject(attrs) && ! options?.saving
|
119
|
+
for name, association of this._associations||{}
|
120
|
+
association.instance(this).set(attrs[association.name], options) if attrs[association.name]
|
121
|
+
this
|
122
|
+
|
123
|
+
save: (options={})->
|
124
|
+
options.setFromResponse=true
|
125
|
+
options.saving=true
|
126
|
+
handlers = wrapRequest(this,options)
|
127
|
+
super({}, options)
|
128
|
+
handlers.promise
|
129
|
+
|
130
|
+
dataForSave: (options={})->
|
131
|
+
if options.saveAll
|
132
|
+
data = @attributes
|
133
|
+
else
|
134
|
+
data = this.unsavedData()
|
135
|
+
for name, association of this._associations||{}
|
136
|
+
if association.isCreated(this) && association.instance(this).isDirty()
|
137
|
+
instance = association.instance(this)
|
138
|
+
data[association.name] = if options.saveAll then instance.attributes else instance.unsavedData()
|
139
|
+
data
|
140
|
+
|
141
|
+
destroy: (options={})->
|
142
|
+
handlers = wrapRequest(this,options)
|
143
|
+
super(options)
|
144
|
+
handlers.promise
|
145
|
+
|
146
|
+
unsavedData: ->
|
147
|
+
attrs = if this.isNew() then {} else { id: this.getId() }
|
148
|
+
_.extend(attrs, _.pick( this.attributes, @_unsaved... ) )
|
149
|
+
|
150
|
+
isDirty:->
|
151
|
+
!!@_unsaved.length
|
152
|
+
|
153
|
+
hasAttribute: (name)->
|
154
|
+
!!this._definition[name]
|
155
|
+
|
156
|
+
associationDefinition: (name,options)->
|
157
|
+
_.extend(options, {
|
158
|
+
fk: options.fk || name + "_id"
|
159
|
+
name: name
|
160
|
+
isCreated: (parent)->
|
161
|
+
parent._cache.hasOwnProperty(@name)
|
162
|
+
instance: (parent)-> parent[@name]
|
163
|
+
})
|
164
|
+
|
165
|
+
|
166
|
+
associationDerivedDefinition: (name,definition)->
|
167
|
+
klass = associationLookupClass(definition)
|
168
|
+
{
|
169
|
+
deps: [definition.fk]
|
170
|
+
fn: ->
|
171
|
+
args = {}
|
172
|
+
args['id'] = this.get(definition.fk) if this.get(definition.fk)
|
173
|
+
if definition.defaultValue
|
174
|
+
_.defaults(args, _.evaluateFunction(definition.defaultValue))
|
175
|
+
klass ||= associationLookupClass(definition)
|
176
|
+
record = klass.findOrCreate(args)
|
177
|
+
record.parent = this
|
178
|
+
record
|
179
|
+
}
|
180
|
+
|
181
|
+
validateFieldChange: (name, value)->
|
182
|
+
return '' unless def = this._definition[name]
|
183
|
+
if def.required && _.isEmpty(value)
|
184
|
+
"Cannot be empty"
|
185
|
+
else
|
186
|
+
''
|
187
|
+
|
188
|
+
sync: ->
|
189
|
+
Lanes.Data.Sync.apply(this,arguments)
|
190
|
+
|
191
|
+
_recordUnsaved: (record,options)->
|
192
|
+
attrs = this.changedAttributes()
|
193
|
+
unless options?.silent || options?.ignoreUnsaved
|
194
|
+
for name,val of attrs
|
195
|
+
@_unsaved.push( name ) if -1 == @_unsaved.indexOf(name)
|
196
|
+
this
|
197
|
+
|
198
|
+
|
199
|
+
@extended: (klass)->
|
200
|
+
klass.prototype.props ||= {}
|
201
|
+
klass.prototype.session ||= {}
|
202
|
+
klass.prototype.associations ||= {}
|
203
|
+
(klass.prototype.addStandardProperties || setupStandardProps).call(klass, klass.prototype)
|
204
|
+
|
205
|
+
setupAssociations(klass.prototype) if klass.prototype.associations
|
206
|
+
|
207
|
+
|
208
|
+
Lanes.lib.ModuleSupport.includeInto(@)
|
209
|
+
@include Lanes.lib.results
|
210
|
+
|
211
|
+
|
212
|
+
associationLookupClass = (definition)->
|
213
|
+
object = definition.model || definition.collection
|
214
|
+
if _.isObject(object) then object else Lanes.getPath( object, "Lanes.Data")
|
215
|
+
|
216
|
+
setupAssociations=(klass)->
|
217
|
+
klass.derived ||= {}
|
218
|
+
klass._associations = {}
|
219
|
+
derivedFn = ( klass.associationDerivedDefinition || DataModel.prototype.associationDerivedDefinition )
|
220
|
+
createFn = ( klass.associationDefinition || DataModel.prototype.associationDefinition)
|
221
|
+
for name, options of klass.associations
|
222
|
+
definition = createFn.call(klass, name, options)
|
223
|
+
klass.derived[ name ] = derivedFn.call(klass, name, definition)
|
224
|
+
klass._associations[name] = definition
|
225
|
+
|
226
|
+
setupStandardProps=(klass)->
|
227
|
+
klass.session['created_at'] ||= 'date'
|
228
|
+
klass.session['updated_at'] ||= 'date'
|
229
|
+
klass.associations['created_by'] ||= { model: 'Lanes.Data.User' }
|
230
|
+
klass.associations['updated_by'] ||= { model: 'Lanes.Data.User' }
|
231
|
+
|
232
|
+
copyServerResp = (record,resp)->
|
233
|
+
record.errors = resp.errors
|
234
|
+
record.lastServerMessage = resp.message
|
235
|
+
{ record: record, response: resp }
|
236
|
+
|
237
|
+
wrapRequest = (record, options)->
|
238
|
+
error = options.error
|
239
|
+
success = options.success
|
240
|
+
options.promise = new Lanes.Promise( (resolve,reject)->
|
241
|
+
options.resolvePromise = resolve
|
242
|
+
options.rejectPromise = reject
|
243
|
+
)
|
244
|
+
options.error = (record, resp, req)->
|
245
|
+
options.rejectPromise( copyServerResp(record,resp.responseJSON) )
|
246
|
+
error?.apply(options.scope, arguments)
|
247
|
+
|
248
|
+
options.success = (record,resp,req)->
|
249
|
+
options.resolvePromise( copyServerResp(record,resp) )
|
250
|
+
record._unsaved = []
|
251
|
+
success?.apply(options.scope, arguments)
|
252
|
+
options
|
253
|
+
|
254
|
+
|
255
|
+
Lanes.Data.Model = Lanes.lib.MakeBaseClass( Lanes.Vendor.Ampersand.Model, DataModel )
|
256
|
+
|
257
|
+
|
258
|
+
class BasicModel
|
259
|
+
constructor: -> super
|
260
|
+
isPersistent: -> false
|
261
|
+
isModel: true
|
262
|
+
|
263
|
+
Lanes.Data.BasicModel = Lanes.lib.MakeBaseClass( Lanes.Vendor.Ampersand.Model, BasicModel )
|
264
|
+
|
265
|
+
|
266
|
+
class State
|
267
|
+
constructor: -> super
|
268
|
+
|
269
|
+
Lanes.Data.State = Lanes.lib.MakeBaseClass( Lanes.Vendor.Ampersand.State, State )
|
@@ -0,0 +1,68 @@
|
|
1
|
+
MB = Lanes.Vendor.MessageBus
|
2
|
+
|
3
|
+
class ModelType
|
4
|
+
constructor: ->
|
5
|
+
super
|
6
|
+
@records = {}
|
7
|
+
|
8
|
+
session:
|
9
|
+
id: 'string'
|
10
|
+
records: 'object'
|
11
|
+
|
12
|
+
subscribe: (model)->
|
13
|
+
channel = "/#{model.api_path}/#{model.id}"
|
14
|
+
MB.subscribe(channel,(changes)->
|
15
|
+
model.addChangeSet(changes)
|
16
|
+
)
|
17
|
+
channel
|
18
|
+
|
19
|
+
add: (model)->
|
20
|
+
if (config = @records[model.id])
|
21
|
+
config.model = model
|
22
|
+
else
|
23
|
+
@records[model.id] = { model: model, channel: this.subscribe(model) }
|
24
|
+
|
25
|
+
remove: (model)->
|
26
|
+
if ( config = @records[model.id] )
|
27
|
+
MB.unsubscribe( config.channel )
|
28
|
+
delete @records[model.id]
|
29
|
+
|
30
|
+
|
31
|
+
Lanes.Data.State.extend(ModelType)
|
32
|
+
|
33
|
+
class ModelTypesCollection
|
34
|
+
constructor: -> super
|
35
|
+
model: ModelType
|
36
|
+
|
37
|
+
forModel: (model)->
|
38
|
+
models = this.get(model.api_path) || this.add(id: model.api_path)
|
39
|
+
|
40
|
+
Lanes.Data.BasicCollection.extend(ModelTypesCollection)
|
41
|
+
|
42
|
+
|
43
|
+
|
44
|
+
Lanes.Data.PubSub = {
|
45
|
+
|
46
|
+
types: new ModelTypesCollection
|
47
|
+
|
48
|
+
forModel: (model)->
|
49
|
+
|
50
|
+
add: (model)->
|
51
|
+
return unless model.isPersistent()
|
52
|
+
@types.forModel(model).add(model)
|
53
|
+
|
54
|
+
remove: (model)->
|
55
|
+
return unless model && model.isPersistent()
|
56
|
+
@types.forModel(model).remove(model)
|
57
|
+
|
58
|
+
instanceFor: ( model_klass, id )->
|
59
|
+
@types.get(model_klass.prototype.api_path)?.records[id]?.model
|
60
|
+
|
61
|
+
clear: ->
|
62
|
+
@types = new ModelTypesCollection
|
63
|
+
|
64
|
+
initialize: ->
|
65
|
+
MB.start()
|
66
|
+
MB.callbackInterval = 500
|
67
|
+
|
68
|
+
}
|
@@ -0,0 +1,184 @@
|
|
1
|
+
class Field extends Lanes.Data.Model
|
2
|
+
|
3
|
+
constructor: (attributes)->
|
4
|
+
super( _.defaults( attributes, {
|
5
|
+
title: _.titleize(attributes.field)
|
6
|
+
}))
|
7
|
+
|
8
|
+
session:
|
9
|
+
title: 'string'
|
10
|
+
field: 'string'
|
11
|
+
selected: 'boolean'
|
12
|
+
type:
|
13
|
+
type: 'string'
|
14
|
+
default: -> @model_field?.type
|
15
|
+
|
16
|
+
derived:
|
17
|
+
model_field:
|
18
|
+
deps: ['field'], fn:->
|
19
|
+
this.collection.query.model_class::_definition[@field]
|
20
|
+
|
21
|
+
validValue: (value)->
|
22
|
+
if this.type == 'n'
|
23
|
+
! _.isNaN( parseFloat(value) )
|
24
|
+
else
|
25
|
+
value
|
26
|
+
|
27
|
+
|
28
|
+
class AvailableFields extends Lanes.Data.Collection
|
29
|
+
|
30
|
+
constructor: (models,options)->
|
31
|
+
@query = options.query
|
32
|
+
super
|
33
|
+
model: Field
|
34
|
+
|
35
|
+
|
36
|
+
|
37
|
+
class Operator extends Lanes.Data.Model
|
38
|
+
|
39
|
+
addStandardProperties: Lanes.emptyFn
|
40
|
+
|
41
|
+
session:
|
42
|
+
id: 'string'
|
43
|
+
name: 'string'
|
44
|
+
types: 'array'
|
45
|
+
field: 'object'
|
46
|
+
selected: 'boolean'
|
47
|
+
|
48
|
+
derived:
|
49
|
+
valid:
|
50
|
+
deps: ['types','field']
|
51
|
+
fn: ->
|
52
|
+
!this.types || ( this.field && _.contains(this.types, this.field.type) )
|
53
|
+
|
54
|
+
|
55
|
+
class Operators extends Lanes.Data.Collection
|
56
|
+
model: Operator
|
57
|
+
|
58
|
+
constructor: ->
|
59
|
+
super
|
60
|
+
this.add([
|
61
|
+
{ id: 'like', name: 'Starts With', types:['string'] }
|
62
|
+
{ id: 'eq', name: 'Equals' }
|
63
|
+
{ id: 'lt', name: 'Less Than', types:['integer','bigdec','number'] }
|
64
|
+
{ id: 'gt', name: 'More Than', types:['integer','bigdec','number'] }
|
65
|
+
])
|
66
|
+
|
67
|
+
setField: (field)->
|
68
|
+
this.invoke('set', 'field', field)
|
69
|
+
selected = this.findWhere(selected: true)
|
70
|
+
if selected && !selected.valid
|
71
|
+
selected.selected = false
|
72
|
+
this.findWhere(valid:true).selected = true
|
73
|
+
|
74
|
+
|
75
|
+
class Clause extends Lanes.Data.Model
|
76
|
+
|
77
|
+
session:
|
78
|
+
value : { type: 'string', default: '' }
|
79
|
+
operators : 'collection'
|
80
|
+
field : 'model'
|
81
|
+
operator : 'model'
|
82
|
+
|
83
|
+
derived:
|
84
|
+
description:
|
85
|
+
deps: ['field','operator']
|
86
|
+
fn: -> "#{@field.title} #{@operator.id}"
|
87
|
+
|
88
|
+
isValid:
|
89
|
+
deps: ['field', 'operator', 'value']
|
90
|
+
fn: -> this.field.validValue(@value)
|
91
|
+
|
92
|
+
initialize:(attrs,options)->
|
93
|
+
super
|
94
|
+
@operators = new Operators
|
95
|
+
@fields = @collection.fields
|
96
|
+
@operators.field = @fields.first()
|
97
|
+
@fields.on('change:selected', this.setField, this)
|
98
|
+
@operators.on('change:selected', this.setOperator, this)
|
99
|
+
|
100
|
+
@fields.at(0).selected = true
|
101
|
+
@operators.at(0).selected = true
|
102
|
+
|
103
|
+
window.q = this
|
104
|
+
|
105
|
+
setField: (field)->
|
106
|
+
return unless field.selected
|
107
|
+
@operators.setField( field )
|
108
|
+
this.field = field
|
109
|
+
|
110
|
+
setOperator: (operator)->
|
111
|
+
return unless operator.selected
|
112
|
+
this.operator = operator
|
113
|
+
|
114
|
+
remove: ->
|
115
|
+
@collection.remove(this)
|
116
|
+
|
117
|
+
toParam: ->
|
118
|
+
param = {}
|
119
|
+
op = this.operator.id
|
120
|
+
value = this.get('value')
|
121
|
+
value +='%' if 'like' == op
|
122
|
+
value = parseFloat(value) if @field.type == "n"
|
123
|
+
param[ this.field.field ] = if 'eq' == op then value else { op: op, value: value }
|
124
|
+
param
|
125
|
+
|
126
|
+
|
127
|
+
class Clauses extends Lanes.Data.Collection
|
128
|
+
|
129
|
+
model: Clause
|
130
|
+
|
131
|
+
initialize:(models,options)->
|
132
|
+
super
|
133
|
+
@query = options.query
|
134
|
+
@fields = options.query.fields
|
135
|
+
|
136
|
+
|
137
|
+
class Lanes.Data.Query extends Lanes.Data.Model
|
138
|
+
|
139
|
+
session:
|
140
|
+
fields: 'collection'
|
141
|
+
clauses: 'collection'
|
142
|
+
collection_class: 'function'
|
143
|
+
initial_field: 'string'
|
144
|
+
|
145
|
+
derived:
|
146
|
+
model_class:
|
147
|
+
deps:['collection_class'], fn: -> @collection_class::model
|
148
|
+
url:
|
149
|
+
deps:['collection_class'], fn: -> @collection_class::url()
|
150
|
+
|
151
|
+
constructor: (options={})->
|
152
|
+
super
|
153
|
+
@fields = new AvailableFields(
|
154
|
+
_.map( options.fields, (col)-> if _.isObject(col) then col else { field: col } ),
|
155
|
+
query: this
|
156
|
+
)
|
157
|
+
@clauses = new Clauses([], query: this )
|
158
|
+
this.listenTo(@clauses,'change remove reset', ->
|
159
|
+
this.trigger('change', arguments...)
|
160
|
+
)
|
161
|
+
@initial_field = @fields.first.field
|
162
|
+
this.addNewClause()
|
163
|
+
this
|
164
|
+
|
165
|
+
isValid: ->
|
166
|
+
! @clauses.findWhere( isValid: false )
|
167
|
+
|
168
|
+
loadSingle: (code,options)->
|
169
|
+
options.query = {}
|
170
|
+
options.query[ @initial_field ] = code
|
171
|
+
@collection_class::model.fetch(options)
|
172
|
+
|
173
|
+
defaultField: ->
|
174
|
+
@fields.findWhere( field: @initial_field )
|
175
|
+
|
176
|
+
asParams: ->
|
177
|
+
params = {}
|
178
|
+
@clauses.each (clause)->
|
179
|
+
_.extend( params, clause.toParam() ) if clause.isValid
|
180
|
+
|
181
|
+
params
|
182
|
+
|
183
|
+
addNewClause: ->
|
184
|
+
@clauses.add({ field: @initial_field, operator: 'like' })
|
@@ -0,0 +1,91 @@
|
|
1
|
+
class Role extends Lanes.Data.BasicModel
|
2
|
+
props:
|
3
|
+
id: 'string'
|
4
|
+
name: 'string'
|
5
|
+
member: { type: 'boolean', default: false }
|
6
|
+
|
7
|
+
|
8
|
+
class Lanes.Data.Roles extends Lanes.Data.BasicCollection
|
9
|
+
model: Role
|
10
|
+
comparator: 'name'
|
11
|
+
|
12
|
+
|
13
|
+
class RoleExtension extends Lanes.Extension.Base
|
14
|
+
identifier: 'roles'
|
15
|
+
setBootstrapData: (data)->
|
16
|
+
Lanes.Data.Roles.all = new Lanes.Data.Roles(
|
17
|
+
_.map( data, (role)->
|
18
|
+
{ id: role.toLowerCase(), name: role }
|
19
|
+
)
|
20
|
+
)
|
21
|
+
|
22
|
+
|
23
|
+
class UserRole
|
24
|
+
constructor: (config={})->
|
25
|
+
@type = config.type
|
26
|
+
for method in RWD
|
27
|
+
this[method] = _.map(config[method],klassFor)
|
28
|
+
|
29
|
+
can: (type,model)->
|
30
|
+
-1 != this[type].indexOf(model)
|
31
|
+
|
32
|
+
|
33
|
+
RWD = ['read','write','delete']
|
34
|
+
|
35
|
+
RoleMap = {
|
36
|
+
administrator: Administrator
|
37
|
+
}
|
38
|
+
|
39
|
+
# The admin is special and can do anything
|
40
|
+
class Administrator
|
41
|
+
constructor: ->
|
42
|
+
UserRole.prototype.constructor.apply(this,arguments)
|
43
|
+
can: -> true
|
44
|
+
|
45
|
+
|
46
|
+
|
47
|
+
class Lanes.Data.UserRoleSet
|
48
|
+
|
49
|
+
constructor: (access={})->
|
50
|
+
_.defaults(access, { roles: [], locked_fields: [] })
|
51
|
+
@roles = []
|
52
|
+
for role in access.roles
|
53
|
+
klass = RoleMap[role.type] || UserRole
|
54
|
+
@roles.push( new klass(role) )
|
55
|
+
@locked_fields = {}
|
56
|
+
for lock in access.locked_fields
|
57
|
+
if klass = klassFor(lock.type)
|
58
|
+
@locked_fields[ klass ] = locks = {}
|
59
|
+
locks[field] = grants for field, grants of lock.locks
|
60
|
+
|
61
|
+
can:(method,model,field)->
|
62
|
+
if model instanceof Lanes.Data.Model
|
63
|
+
model = model.constructor
|
64
|
+
|
65
|
+
if field && ( locks = @locked_fields[model] ) && ( grants = locks[field] )
|
66
|
+
for grant in grants
|
67
|
+
if grant.only && grant.only != method
|
68
|
+
return this.testModelAccess(method,model)
|
69
|
+
else
|
70
|
+
for role in @roles
|
71
|
+
return true if role.type == grant.role
|
72
|
+
return false
|
73
|
+
else
|
74
|
+
return this.testModelAccess(method,model)
|
75
|
+
|
76
|
+
testModelAccess:(method,model)->
|
77
|
+
!!_.detect( @roles, (role)->role.can(method,model) )
|
78
|
+
|
79
|
+
canRead:(model,field)->
|
80
|
+
this.can('read',model,field)
|
81
|
+
|
82
|
+
canWrite:(model,field)->
|
83
|
+
this.can('write',model,field)
|
84
|
+
|
85
|
+
canDelete:(model)->
|
86
|
+
this.can('delete',model)
|
87
|
+
|
88
|
+
klassFor = (identifier)->
|
89
|
+
name = _.chain(identifier).titleize().gsub(' ','').value()
|
90
|
+
Lanes.Data[name] ||
|
91
|
+
Lanes.warn("Role Data object not found for #{identifier}")
|