| 1 | == SahanaPy Framework == |
| 2 | We have built an S3 framework as a higher level of abstraction on top of T2.[[BR]] |
| 3 | This should be used by modules where possible, but if more power is needed then drop down a level of two (to T2, to base Web2Py or to raw Python). |
| 4 | |
| 5 | T2 is used for: |
| 6 | * [wiki:DeveloperGuidelinesAuthenticationAccess AAA] |
| 7 | * simplified [wiki:DeveloperGuidelinesCreateReadUpdateDelete CRUD] |
| 8 | * Conflict Detection |
| 9 | We extend the T2 class in {{{modules/sahana.py}}}.[[BR]] |
| 10 | Patches to the framework should be done here.[[BR]] |
| 11 | If they're generic enough then they can be suggested to Massimo for moving upstream to either Web2Py or T2. |
| 12 | |
| 13 | We extend the CRUD with a [wiki:BluePrintRESTImplementation RESTlike controller].[[BR]] |
| 14 | This provides details on how to configure your Model & Controllers. Views may not be required, other than index.html |
| 15 | |
| 16 | === Mandatory === |
| 17 | Each Controller should start like this (to populate the side navigation Menus): |
| 18 | {{{ |
| 19 | module='module' |
| 20 | # Current Module (for sidebar title) |
| 21 | module_name=db(db.module.name==module).select()[0].name_nice |
| 22 | # List Modules (from which to build Menu of Modules) |
| 23 | modules=db(db.module.enabled=='Yes').select(db.module.ALL,orderby=db.module.menu_priority) |
| 24 | # List Options (from which to build Menu for this Module) |
| 25 | options=db(db['%s_menu_option' % module].enabled=='Yes').select(db['%s_menu_option' % module].ALL,orderby=db['%s_menu_option' % module].priority) |
| 26 | }}} |
| 27 | |
| 28 | Each function needs to return these values to the view: |
| 29 | return dict(module_name=module_name,modules=modules,options=options) |
| 30 | |
| 31 | === Optional === |
| 32 | List output can be made more functional by this .represent 'widget': |
| 33 | {{{ |
| 34 | def shn_list_item(table,resource,action,display='table.name',extra=None): |
| 35 | if extra: |
| 36 | items=DIV(TR(TD(A(eval(display),_href=t2.action(resource,[action,table.id]))),TD(eval(extra)))) |
| 37 | else: |
| 38 | items=DIV(A(eval(display),_href=t2.action(resource,[action,table.id]))) |
| 39 | return DIV(*items) |
| 40 | }}} |
| 41 | You can use it in {{{models/module.py}}} like: |
| 42 | {{{ |
| 43 | db.or_organisation.represent=lambda table:shn_list_item(table,resource='organisation',action='display') |
| 44 | db.person.represent=lambda table:shn_list_item(table,resource='person',action='display',display='table.full_name') |
| 45 | db.gis_projection.represent=lambda table:shn_list_item(table,resource='projection',action='display',extra='table.epsg') |
| 46 | }}} |
| 47 | |
| 48 | Form labels can be set in a translatable manner using: |
| 49 | {{{db.table.field.label=T("label")}}} |
| 50 | |
| 51 | Form field can be made to use a TEXTAREA by marking the field as being type 'text': |
| 52 | {{{SQLField('field','text'),}}} |
| 53 | |
| 54 | Form field can be made to use a SELECT dropdown by setting the field as a lookup to another table...linked to the 'uuid' field to allow [wiki:DeveloperGuidelinesDatabaseSynchronization Database Synchronization], but displaying a more user-friendly field (such as 'name'): |
| 55 | {{{ |
| 56 | SQLField('field',length=64), |
| 57 | |
| 58 | db.table.field.requires=IS_NULL_OR(IS_IN_DB(db,'othertable.uuid','othertable.name')) |
| 59 | }}} |
| 60 | |
| 61 | Form field being required can be marked using: |
| 62 | {{{db.table.field.comment=SPAN("*",_class="req")}}} |
| 63 | |
| 64 | Help for a form field can be set using: |
| 65 | {{{A(SPAN("[Help]"),_class="popupLink",_id="tooltip",_title=T("Help Title|This is what this field is for."))}}} |
| 66 | |
| 67 | Different Flash styles can be set via: |
| 68 | {{{ |
| 69 | session.error=T("Unsupported format!") |
| 70 | redirect(URL(r=request,f=resource)) |
| 71 | }}} |
| 72 | or (in a Multiple Table form.accepts): |
| 73 | {{{ |
| 74 | response.error=T("Form invalid!") |
| 75 | }}} |
| 76 | Supported styles are: |
| 77 | * .warning |
| 78 | * .error |
| 79 | * .information |
| 80 | * .confirmation (Standard T2 Flash messages are usually of this sort so we class them in the same way) |
| 81 | |
| 82 | === Settings === |
| 83 | System-wide settings have their default values set in {{{models/_db.py}}}'s {{{shn_default}}}. |
| 84 | |
| 85 | Upon 1st run of the system, these settings are pulled into the default_setting table in the database from where they can susequently be edited to configure the running instance. |
| 86 | |
| 87 | If these settings are required by all Views then can patch the session to see them in {{{models/_db.py}}}'s {{{shn_sessions()}}}.[[BR]] |
| 88 | e.g. response.debug is made available to the default {{{layout.html}}} this way. |
| 89 | |
| 90 | If you wish to clean out all settings to import afresh, then close down Web2Py & start up in shell mode: |
| 91 | {{{ |
| 92 | python web2py.py -S sahana -M |
| 93 | }}} |
| 94 | Then run: |
| 95 | {{{ |
| 96 | shn_db_clean(db) |
| 97 | }}} |
| 98 | (This function is configured in {{{modules/sahana.py}}}) |
| 99 | |
| 100 | |
| 101 | Modules can set up their own configuration settings tables in a similar way (e.g. GIS does this) |
| 102 | |
| 103 | === jQuery Widgets === |
| 104 | * [wiki:DeveloperGuidelinesDeletableList Deletable List] |
| 105 | |
| 106 | === Conflict Detection === |
| 107 | Sahana is a multi-user system so there is a potential for multiple users to be editing the same record at once.[[BR]] |
| 108 | We use T2 to handle this for us. |
| 109 | |
| 110 | Add this field to each table which needs protecting (in {{{models/db.py}}}): |
| 111 | {{{ |
| 112 | SQLField('modified_on','datetime'), # Used by T2 to do edit conflict-detection |
| 113 | }}} |
| 114 | |
| 115 | This field is also used in [wiki:DeveloperGuidelinesDatabaseSynchronization Database Synchronization] |
| 116 | |
| 117 | ---- |
| 118 | DeveloperGuidelines |