mickeybob
mickeybob

Reputation: 457

ImportError: Cannot re-init internal module __main__, when using Pyramid and Mako with Google App Engine

My Pyramid app works fine when run inside the paste httpserver, but throws an exception in google app engine.

The /hello/{name} url works fine in both paste server and in app engine

The /test url works fine in paste server, but in app engine I get the Traceback below.

Any ideas?

from pyramid.config import Configurator
from pyramid.response import Response

def hello_world(request):
    return Response('Hello %(name)s!' % request.matchdict)

def test(request):
    return dict()

settings = {'mako.directories':'mako_templates'}
config = Configurator(settings=settings)
config.add_route('hello', '/hello/{name}')
config.add_route('test', '/test', view=test, renderer='test.mako')
config.add_view(hello_world, route_name='hello')

wsgi_app = config.make_wsgi_app()

Traceback from google app engine:

ERROR    2013-01-17 02:48:08,893 wsgi.py:235] 
Traceback (most recent call last):
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/runtime/wsgi.py", line 223, in Handle
    result = handler(dict(self._environ), self._StartResponse)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/router.py", line 251, in __call__
    response = self.invoke_subrequest(request, use_tweens=True)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/router.py", line 227, in invoke_subrequest
    response = handle_request(request)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/tweens.py", line 21, in excview_tween
    response = handler(request)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/router.py", line 161, in handle_request
    response = view_callable(context, request)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/config/views.py", line 367, in rendered_view
    context)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/renderers.py", line 531, in render_view
    return self.render_to_response(response, system, request=request)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/renderers.py", line 561, in render_to_response
    result = self.render(value, system_values, request=request)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/renderers.py", line 534, in render
    renderer = self.renderer
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/decorator.py", line 39, in __get__
    val = self.wrapped(inst)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/renderers.py", line 518, in renderer
    return factory(self)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/mako_templating.py", line 129, in __call__
    directories = [ abspath_from_asset_spec(d) for d in directories ]
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pyramid/asset.py", line 43, in abspath_from_asset_spec
    return pkg_resources.resource_filename(pname, filename)
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pkg_resources.py", line 840, in resource_filename
    return get_provider(package_or_requirement).get_resource_filename(
  File "/Users/mnaber/Documents/workspace/UnifiedClassified/site-packages/pkg_resources.py", line 160, in get_provider
    __import__(moduleOrReq)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 692, in Decorate
    return func(self, *args, **kwargs)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 1766, in load_module
    return self.FindAndLoadModule(submodule, fullname, search_path)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 692, in Decorate
    return func(self, *args, **kwargs)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 1630, in FindAndLoadModule
    description)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 692, in Decorate
    return func(self, *args, **kwargs)
  File "/Applications/GoogleAppEngineLauncher.app/Contents/Resources/GoogleAppEngine-default.bundle/Contents/Resources/google_appengine/google/appengine/tools/dev_appserver_import_hook.py", line 1577, in LoadModuleRestricted
    description)
ImportError: Cannot re-init internal module __main__
INFO     2013-01-17 02:48:08,909 dev_appserver.py:3103] "GET /test HTTP/1.1" 500 -

Upvotes: 1

Views: 968

Answers (2)

mickeybob
mickeybob

Reputation: 457

My background on this stuff is pretty shallow, so this may be wrong.

Here's my take: To use mako with pyramid you need to give the Configurator a settings dictionary having a key called 'mako.directories' that points to your mako templates directory. This key can be either a relative path or an absolute path.

The pyramid/mako_templating.py file converts relative paths to absolute paths in this line:

directories = [ abspath_from_asset_spec(d) for d in directories ]

This conversion isn't working in gae (see stack trace). To get around this, I used the Pyramid AssetResolver, like this:

from pyramid.path import AssetResolver
r = AssetResolver('appname')
full_mako_path = r.resolve('http/mako_templates').abspath()

Then, when you create your Pyramid Configurator, you do:

settings = {'mako.directories':full_mako_path}
config = Configurator(settings=settings)

In this case my directory structure is:

/serve.py (contains wsgi app)
/app.yaml
/appname
/appname/http/mako_templates
...

Upvotes: 1

doru
doru

Reputation: 9110

Try this:

Put your Pyramid and Mako folders in a directory named framework in your app directory. Then add these lines of code to your main .py file under the imports:

import sys, os
framework_dir = "framework"
framework_dir_path = os.path.join(os.path.dirname(__file__), framework_dir)
sys.path.insert(0, framework_dir_path)

Upvotes: 0

Related Questions