= Release Management =
[[TOC]]
[http://www.informit.com/articles/printerfriendly.aspx?p=1833567 Four Principles of Low-Risk Software Releases]
== Release Management for the Haiti Portal ==
This example can be updated/adapted for future instances.
We have 3 running instances:
* Production: http://haiti.sahanafoundation.org
* UAT: http://haiti-test.sahanafoundation.org - lifeeth
* Development: http://haiti-dev.sahanafoundation.org - lifeeth
The [BugReportingGuidelines Bug Tracker] should have different views to distinguish between bugs being reported against each version
* !ToDo customise the 'Report Bugs' footer URL to link directly to a new ticket against that version?
We have several Code branches:
* Haiti Portal: https://launchpad.net/s3/haiti-quake-2010
* Haiti Offline: ''tbc''
* Trunk: https://launchpad.net/s3/sahanapy
* RMS: https://code.launchpad.net/~uwthw/s3/rms
* VITA: https://code.launchpad.net/~nursix.org/s3/vita
=== Release Process ===
Overall responsible Person: Fran (currently)
1. Where generic, new code is merged 1st into Trunk by Fran
2. Code is then merged into the Haiti branch by Fran
* RMS is currently Haiti-specific so is merged directly to Haiti branch
3. Code is pulled from the Haiti branch to Dev by Fran
* Data is 1st synced from Prod:
{{{
/root/automysqlbackup.sh
scp /backups/daily/haitiprod/haitiprod_2010-01-27_01h24m.Wednesday.sql.gz flavour@haiti-dev.sahanafoundation.org:.
gunzip haitiprod_2010-01-27_01h24m.Wednesday.sql.gz
sed 's/haitiprod/haitidev/g' haitiprod_2010-01-27_01h24m.Wednesday.sql > haitidev_import.sql
vim /etc/crontab
#0-59/1 * * * * www-data cd /home/haiti/web2py/ && python web2py.py -C -D 1 >> /tmp/cron.output 2>&1
ln -sf /etc/apache2/sites-available/dev-maintain /etc/apache2/sites-enabled/dev
/etc/init.d/apache2 force-reload
mysql
drop DATABASE haitidev;
create DATABASE haitidev;
\q
mysql < haitidev_import.sql
vim /home/haiti/dev/models/00_db.py
migrate=True
ln -sf /etc/apache2/sites-available/dev /etc/apache2/sites-enabled/dev
/etc/init.d/apache2 force-reload
vim /etc/crontab
0-59/1 * * * * www-data cd /home/haiti/web2py/ && python web2py.py -C -D 1 >> /tmp/cron.output 2>&1
vim /home/haiti/dev/models/00_db.py
migrate=False
}}}
* A 1st draft of a data migration plan is made
4. Code is pulled from the Haiti branch to UAT by lifeeth or chamindra
* Testers to be informed before this via the group: ''tbc''
* Data is 1st synced from Prod
* A more detailed data migration plan is made
5. Code is pulled from the Haiti branch to Prod by Fran (Volunteers welcomed to take over!)
* Sign-off for this to come from the group: ''tbc''
* A careful data migration plan is made
* Backup
* Disable system crontab & put Website into Maintenance Mode
{{{
vim /etc/crontab
#0-59/1 * * * * www-data cd /home/haiti/web2py/ && python web2py.py -C -D 1 >> /tmp/cron.output 2>&1
vim /home/haiti/prod/models/00_db.py
migrate=True
rm /etc/apache2/sites-enabled/prod-ssl
/etc/init.d/apache2 force-reload
/home/haiti/update-prod
/home/haiti/shell-prod
db.export_to_csv_file(open('db.csv','wb'))
Ctrl+D
}}}
* DataMigration
* {{{cd /home/haiti/web2py ; python web2py.py -S prod -M -N}}}
{{{
Ctrl+D
}}}
* Restore Website & Cron
{{{
ln -s /etc/apache2/sites-available/prod-ssl /etc/apache2/sites-enabled
/etc/init.d/apache2 force-reload
vim /home/haiti/prod/models/00_db.py
migrate=False
vim /etc/crontab
0-59/1 * * * * www-data cd /home/haiti/web2py/ && python web2py.py -C -D 1 >> /tmp/cron.output 2>&1
}}}
* BluePrintSynchronisation
=== Technical Details ===
==== Production ====
* haiti.sahanapy.org VM: 212.23.5.4
* /home/haiti/web2py/applications/prod
* /etc/apache2/sites-enabled/prod
* ~~Sqlite~~ MySQL
* Web2Py r1544
* Sahana Haiti r547
* Customisations:
* models/00_db.py
{{{
#migrate = True
migrate = False
#exec('from applications.%s.modules.sahana import *' % request.application)
# Faster for Production (where app-name won't change):
from applications.prod.modules.sahana import *
#exec('from applications.%s.modules.s3.s3validators import *' % request.application)
# Faster for Production (where app-name won't change):
from applications.prod.modules.validators import *
}}}
* models/00_settings.py
* auth.settings.registration_requires_verification = True
* auth.settings.registration_requires_approval = True
* models/01_modules.py
{{{
s3.menu_modules.append([T('Home'), False, URL(c='default', f='open_module', vars=dict(id=1))])
if auth.has_membership(1):
s3.menu_modules.append([T('Admin'), False, URL(c='default', f='open_module', vars=dict(id=2))])
s3.menu_modules.append([T('Mapping'), False, URL(c='default', f='open_module', vars=dict(id=3))])
module_id = db(db.s3_module.name == 'or').select()[0].id
s3.menu_modules.append([T('Organizations'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
s3.menu_modules.append([T('People'), False, URL(c='default', f='open_module', vars=dict(id=4))])
#dvi_group = db(db[auth.settings.table_group_name] == 'DVI').select()[0].id
#if auth.has_membership(dvi_group):
#module_id = db(db.s3_module.name == 'dvi').select()[0].id
#s3.menu_modules.append([T('Victim Identification'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
module_id = db(db.s3_module.name == 'rms').select()[0].id
s3.menu_modules.append([T('Requests'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
}}}
* cron/crontab
{{{
*/5 * * * * root *applications/prod/cron/rms_entry2record.py
*/5 * * * * root *applications/prod/cron/rms_tweet2request.py
}}}
* views/layout.html
{{{
}}}
* views/ext.html
{{{
}}}
* views/sahana_scripts_min.html
{{{
}}}
* views/gis/index.html
{{{
#LI(
# SPAN(A(T('Bulk Uploader'),
# _href=URL(c='gis', f='bulk_upload')),_style="font-weight:bold;"),
# P(T('Allows authorized users to upload multiple features into the situation map.')),
# _style="margin-top:1em;"),
}}}
* http://haiti-test.sahanafoundation.org/test/gis/apikey/update/1
* ABQIAAAAgB-1pyZu7pKAZrMGv3nksRR1-chREdZE62d-MGawPNPOGidOJBQQCwHsElsSGbD5VMpNj7DBMNxjTg
==== UAT ====
* logistics.sahanapy.org VM: 212.23.5.5
* /home/haiti/web2py-test/applications/test
* /etc/apache2/sites-enabled/haiti-test
* [wiki:InstallationGuidelinesMySQL MySQL] ''hopefully''
* Web2Py r1544
* Sahana Haiti r547
* Customisations:
* models/00_db.py
* migrate = False
* db = DAL('mysql://haititest:password@localhost/haititest', pool_size=10)
* models/00_settings.py
* S3_PUBLIC_URL = 'http://haiti-test.sahanafoundation.org'
* auth.settings.registration_requires_verification = True
* auth.settings.registration_requires_approval = True
* cron/crontab
{{{
*/5 * * * * root *applications/test/cron/rms_entry2record.py
*/5 * * * * root *applications/test/cron/rms_tweet2request.py
}}}
* views/ext.html
{{{
}}}
* views/sahana_scripts_min.html
{{{
}}}
* views/gis/index.html
{{{
#LI(
# SPAN(A(T('Bulk Uploader'),
# _href=URL(c='gis', f='bulk_upload')),_style="font-weight:bold;"),
# P(T('Allows authorized users to upload multiple features into the situation map.')),
# _style="margin-top:1em;"),
}}}
* http://haiti-test.sahanafoundation.org/test/gis/apikey/update/1
* ABQIAAAAgB-1pyZu7pKAZrMGv3nksRR1-chREdZE62d-MGawPNPOGidOJBQQCwHsElsSGbD5VMpNj7DBMNxjTg
==== Dev ====
* logistics.sahanapy.org VM: 212.23.5.5
* /home/haiti/web2py/applications/dev
* /etc/apache2/sites-available/dev
* /etc/apache2/sites-available/dev-maintain
* /etc/apache2/sites-enabled/dev
* [wiki:InstallationGuidelinesMySQL MySQL]
* Web2Py r1544
* Sahana Haiti r547
{{{
cd /home/haiti/web2py/applications
bzr branch lp:s3/haiti-quake-2010 dev
cd dev
chown www-data:www-data -R cache
chown www-data:www-data -R databases
chown www-data:www-data -R errors
chown www-data:www-data -R sessions
chown www-data:www-data -R static/img/markers
chown www-data:www-data -R static/scripts
chown www-data:www-data -R static/styles
chown www-data:www-data -R uploads
}}}
* Customisations:
* models/00_db.py
* migrate = False (after 1st run)
* db = DAL('mysql://haitidev:password@localhost/haitidev', pool_size=10)
* models/01_modules.py
{{{
s3.menu_modules.append([T('Home'), False, URL(c='default', f='open_module', vars=dict(id=1))])
if auth.has_membership(1):
s3.menu_modules.append([T('Admin'), False, URL(c='default', f='open_module', vars=dict(id=2))])
s3.menu_modules.append([T('Mapping'), False, URL(c='default', f='open_module', vars=dict(id=3))])
module_id = db(db.s3_module.name == 'or').select()[0].id
s3.menu_modules.append([T('Organizations'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
module_id = db(db.s3_module.name == 'hms').select()[0].id
s3.menu_modules.append([T('Hospitals'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
s3.menu_modules.append([T('People'), False, URL(c='default', f='open_module', vars=dict(id=4))])
#dvi_group = db(db[auth.settings.table_group_name] == 'DVI').select()[0].id
#if auth.has_membership(dvi_group):
module_id = db(db.s3_module.name == 'dvi').select()[0].id
s3.menu_modules.append([T('Victims'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
module_id = db(db.s3_module.name == 'rms').select()[0].id
s3.menu_modules.append([T('Requests'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
module_id = db(db.s3_module.name == 'vol').select()[0].id
s3.menu_modules.append([T('Volunteers'), False, URL(c='default', f='open_module', vars=dict(id=module_id))])
}}}
* models/00_settings.py
* S3_PUBLIC_URL = 'http://haiti-dev.sahanafoundation.org'
* auth.settings.registration_requires_verification = True
* auth.settings.registration_requires_approval = True
* cron/crontab
{{{
sed '/sahana/ s/sahana/dev/g' /home/haiti/dev/cron/crontab > /tmp/crontab
mv /tmp/crontab /home/haiti/dev/cron/crontab
vim cron/crontab
*/5 * * * * root *applications/dev/cron/rms_sms2record.py
*/5 * * * * root *applications/dev/cron/rms_tweet2request.py
*/5 * * * * root *applications/dev/cron/import_job_do_processing.py
*/5 * * * * root *applications/dev/cron/import_job_do_import.py
}}}
* views/ext.html
{{{
}}}
* views/sahana_scripts_min.html
{{{
}}}
* http://haiti-dev.sahanafoundation.org/dev/gis/apikey/update/1
* ABQIAAAAgB-1pyZu7pKAZrMGv3nksRR1-chREdZE62d-MGawPNPOGidOJBQQCwHsElsSGbD5VMpNj7DBMNxjTg
* static/robots.txt
{{{
sed '/prod/ s/prod/dev/g' /home/haiti/dev/static/robots.txt > /tmp/robots.txt
mv /tmp/robots.txt /home/haiti/dev/static/robots.txt
}}}
Note that UAT & Dev have different Web2Py instances so that:
* We can trial different Web2Py versions on Dev
* Crontab can be separately enabled/disabled
Note that all names/paths include the words 'dev'/'test'/'prod' to minimise chances of confusion as to which site is being altered
----
[wiki:Haiti]