Skip to content

Content types, workflows, and utilities for the Office of International Education at the University of Wisconsin Oshkosh (http://uwosh.edu/oie)

Notifications You must be signed in to change notification settings

uwosh/uwosh.oie.studyabroadstudent

Repository files navigation

uwosh.oie.studyabroadstudent

Content types, workflows, and utilities for the Office of International Education at the University of Wisconsin Oshkosh (http://uwosh.edu/oie).

Features

  • supports the submission, approval, publication, and management of study away programs
  • supports the submission, approval, publication, and management of student applications for programs
  • scripts to export legacy data from the first OIE workflow site (circa 2006, Archetypes-based, Plone 2.5) and import them to Plone 5 (Dexterity-based)

Documentation

Full documentation for end users can be found in the "docs" folder.

Translations

None (yet)

Installation

Install uwosh.oie.studyabroadstudent by adding it to your buildout:

[buildout]

...

eggs =
    uwosh.oie.studyabroadstudent

and then running bin/buildout

Export / Import

This explains how to export data from the legacy site then import it into a new site.

One-Time Install on Legacy Site

Install these on the legacy site (you need to do this only once):

  • Using the Management Interface, e.g. http://localhost:8080/OIE/manage_main, navigate to portal_skins/custom and create a Script (Python) called listApplicationIDs containing the code in the file src/uwosh/oie/studyabroadstudent/scripts/listApplicationIDs.py. This will be used to output the IDs of all applications on the site.
  • Using the Management Interface, create a Script (Python) in portal_skins/custom called extractApplication containing the code in the file src/uwosh/oie/studyabroadstudent/scripts/extractApplication.py and add id to the Script (Python)'s parameter list. This will be used to export the application for a given ID.

Get the List of IDs to Export

Each time you want to run the export, do this on the legacy site:

  • Run the Script (Python) listApplicationIDs by clicking on the Test tab (you may first have to click on listApplicationIDs in the breadcrumbs before clicking on the Test tab). This outputs Python code in your browser window. This code is an array of application IDs.

  • Save the output to a local file with the .py Python filename extension, e.g. listApplicationIDsoutput.py

  • Copy listApplicationIDsoutput.py to the new server, using scp.

  • On the new server, move listApplicationIDsoutput.py to src/uwosh/oie/studyabroadstudent/scripts/listApplicationIDsoutput.py so that the list of IDs can be loaded in the export script. You can do both like this:

    scp listApplicationIDsoutput.py newserversshuser@newserver:uwosh.oie.studyabroadstudent-zeo/zeocluster/src/uwosh.oie.studyabroadstudent/uwosh/oie/studyabroadstudent/
    

Choosing to Use a Public URL or an SSH Tunnel

The extract script uses XML-RPC to have the legacy site export each application, so you must ensure that there is an internet connection from the new site server to the legacy site.

You can specify the URL of the legacy site as HTTP or HTTPS.

You can choose to use the legacy site's public URL or you can ssh tunnel to the legacy site server and specify a URL such as http://localhost:8084:OIE. The advantage of using an ssh tunnel is that you can start a reserved ZEO client on the legacy server and use it to do the export, rather than using the legacy site's public URL and using a public ZEO client, which would affect any users of the legacy site.

If you choose to use an ssh tunnel (recommended!), you must start the ssh tunnel before you can proceed with the export steps below:

ssh mylegacyserversshuser@mylegacyserver -L8084:localhost:8084

This opens an SSH tunnel on port 8084 so you can start a ZEO client that listens on port 8084:

cd /opt/Plone/zeocluster/
bin/client_reserved start

You must keep this SSH tunnel open for the duration of the export process.

Export the Applications

On the new site server:

  • Change directory to the Plone installation:

    cd /opt/Plone/zeocluster
    
  • Run the script extractApplicationsRemotely.py from a command line, saving its output to a file:

    bin/instance run extractApplicationsRemotely.py > extractoutput.out
    

    This will take hours (to extract 8600 student applications it took over 3.5 hours over the internet).

  • You can override default values with --http-ok=yes, --remote-user, --remote-password, --remote-server, or you can use environment variables REMOTEUSER, REMOTEPASSWD, REMOTESERVER to specify those values:

    export REMOTEUSER=mylegacyploneuser
    export REMOTEPASSWD=mylegacyplonepassword
    export REMOTESERVER=localhost:8084/OIE
    bin/instance run ./src/uwosh/oie/studyabroadstudent/scripts/extractApplicationsRemotely.py --http-ok=yes > extractoutput.out
    

    If you use an SSH tunnel, be sure to specify --http-ok=yes, otherwise the script defaults to assuming HTTPS.

  • If extractApplicationsRemotely.py did not finish running and extracted only a subset of the IDs, you can rerun it to skip the IDs it had previously read.

    First, you must generate the list of IDs that have already been exported, then you must move it to where the code expects to find it:

    grep '^[[]' extractoutput.out | awk -F \' 'BEGIN {print "already_read = ["} {printf "\"%s\",\n", $2} END { print "]\n" }' > already_read.py
    mv already_ready.py src/uwosh/oie/studentstudyabroad/
    

    Then you are ready to rerun the script:

    bin/instance run ./src/uwosh/oie/studyabroadstudent/scripts/extractApplicationsRemotely.py --http-ok=yes --skip-ids=yes --id-file=extractoutput.out > newextractoutput.out
    
  • If you ran the script more than once and have more than one output file, combine all the output files into one file:

    cat extractoutput.out newextractoutput.out > combinedextractoutput.out
    

Convert the Export File to Python

  • Massage the output file to turn it into a Python file:

    src/uwosh/oie/studyabroadstudent/scripts/pythonify_extract.sh extractoutput.out > extractoutput.py
    

Import the Applications

  • Run the script src/uwosh/oie/studyabroadstudent/scripts/importApplications.py to import the applications locally into the site, where the uwosh.oie.studentstudyabroad add-on must already be installed. By default, the import will try to create applications in the folder applications of the site OIE but you can override those values with --site-id=Plone and --folder-id=another-folder-id:

    bin/instance run src/uwosh/oie/studyabroadstudent/scripts/importApplications.py
    

    This took almost one hour on a MacBook Air.

Handling Data Problems

If there is a problem with the data, such as a bad birth date year, you will be dropped into a PDB prompt where you can inspect the values and make any changes needed before continuing:

(Pdb) l
329                 try:
330                     dateOfBirth=date(int((DateOfBirth_year is not None) and DateOfBirth_year or '1900'), month_values[((DateOfBirth_month is not None) and (DateOfBirth_month != '-- choose one --')) and DateOfBirth_month or 'January'], int(((DateOfBirth_day is not None) and (DateOfBirth_day != '-- choose one --') and DateOfBirth_day or 1)))
331                 except:
332                     import pdb;pdb.set_trace()
333
334  ->             try:
335                     passportExpDate=date(int((PassportExpDate_year is not None and PassportExpDate_year is not '') and PassportExpDate_year or '1900'), month_values[(PassportExpDate_month == '-- choose one --' or PassportExpDate_month == '') and 'January' or PassportExpDate_month], int((PassportExpDate_day == '-- choose one --' or PassportExpDate_day == '') and 1 or PassportExpDate_day))
336                 except:
337                     import pdb;pdb.set_trace()
338
339                 obj = api.content.create(
(Pdb) DateOfBirth_year
19991
(Pdb) DateOfBirth_month
'January'
(Pdb) DateOfBirth_day
'8'
(Pdb) DateOfBirth_year=1991
(Pdb) dateOfBirth=date(int((DateOfBirth_year is not None) and DateOfBirth_year or '1900'), month_values[((DateOfBirth_month is not None) and (DateOfBirth_month != '-- choose one --')) and DateOfBirth_month or 'January'], int(((DateOfBirth_day is not None) and (DateOfBirth_day != '-- choose one --') and DateOfBirth_day or 1)))
(Pdb) dateOfBirth
datetime.date(1991, 1, 8)
(Pdb) id
'oiestuapp_howlettj1287342001'
(Pdb) c

Using the ID of the application that had the error, you can go to the site, browse to that application (it will be at a URL like https://app.oie.uwosh.edu/Members/howlettj12/oiestuapp_howlettj1287342001) and edit and save it to correct the data error and prevent this particular data error in future extracts and imports.

Rerunning the Import Multiple Times

The argument --skip-existing tells the script to check first if there is already an existing application object with the same ID before creating one.

Verifying the Import

To verify the import, on the new site:

  • Using the Management Interface, create a Script (Python) in portal_skins/custom called extractApplication containing the code in the file src/uwosh/oie/studyabroadstudent/scripts/extractApplication.py and add id to the parameter list

  • Create and run the script extractApplicationsLocally.py and save its output to a file, like this:

    bin/instance run extractApplicationsLocally.py > extractlocallyoutput.out
    
  • Compare the contents of that file to that of the one you created remotely before:

    diff extractoutput.out extractlocallyoutput.out
    

Legacy Time Zones

Since Plone 2.5, the time zones database has changed: the time zones GMT-5 and GMT-6 have since been renamed Etc/GMT+5 and Etc/GMT+6 (see https://community.plone.org/t/unknowntimezoneerror-pytz-quirks/4255/4 for why the sign change). Some DateTime values in legacy data use the old time zone designations, which causes an error in the unpickler() method in the file tzinfo.py, part of the pytz-2015.7-py2.7.egg. This error prevents viewing of legacy (migrated) OIEStudentApplication objects.

For the moment, the only way to get around this error is to patch the unpickler() method in tzinfo.py and add the following lines right after line 525 (the prophetic comment Raises a KeyError if zone no longer exists, which should never happen and would be a bug.):

# Raises a KeyError if zone no longer exists, which should never happen
# and would be a bug.
newzone = zone
if zone.find('GMT-') != -1:
    newzone = zone.replace('GMT-','Etc/GMT+')
if zone.find('GMT+') != -1:
    import pdb;pdb.set_trace()
    newzone = zone.replace('GMT+','Etc/GMT-')
if zone == 'GMT':
    newzone = 'Etc/GMT'
if zone != newzone:
    logger.warn('fixing nonexistent timezone %s to %s' % (zone, newzone))
    zone = newzone
tz = pytz.timezone(zone)

Contribute

Credits

The project was paid for by the Office of International Education at the University of Wisconsin Oshkosh (http://uwosh.edu/oie).

Implementation by T. Kim Nguyen at Wildcard Corp. (https://wildcardcorp.com).

Support

If you are having issues, please let us know via the GitHub issue tracker.

License

The project is licensed under the GPLv2.

About

Content types, workflows, and utilities for the Office of International Education at the University of Wisconsin Oshkosh (http://uwosh.edu/oie)

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •