我正在尝试使用 jinja2 提供的 PackageLoader,但我无法让它工作。
在我app.yaml
的声明中,我声明了所需的库:
libraries:
- name: jinja2
version: latest
- name: setuptools
version: latest
我可以创建的最小示例:
import webapp2, sys
from jinja2 import Environment, PackageLoader
sys.path += ['lib/somepackage']
class Test(webapp2.RequestHandler):
def get(self):
env = Environment(loader=PackageLoader('common'))
self.response.write(env.get_template('test.html').render())
routes = [(r"/", Test)]
app = webapp2.WSGIApplication(routes, debug=True)
该包common
存在于目录中lib/somepackage
,并且有一个templates
包含文件的包test.html
。html 文件仅包含文本“测试”。
当PackageLoader
构建它时,它provider
总是NullProvider
在 Google App Engine 中。当我手动请求提供者时,我DefaultProvider
很明显这里出了点问题。
要请求提供者,我确保lib/somepackage
通过sys.path
转到该目录,然后:
>>> import pkg_resources
>>> provider = pkg_resources.get_provider('common')
>>> provider
<pkg_resources.DefaultProvider instance at 0x8490b2c>
当这在 Google App Engine 中失败时,这是回溯的相关部分:
File "/<>/test/main.py", line 7, in get
self.response.write(env.get_template('test.html').render())
File "/<>/google_appengine/lib/jinja2-2.6/jinja2/environment.py", line 719, in get_template
return self._load_template(name, self.make_globals(globals))
File "/<>/google_appengine/lib/jinja2-2.6/jinja2/environment.py", line 693, in _load_template
template = self.loader.load(self, name, globals)
File "/<>/google_appengine/lib/jinja2-2.6/jinja2/loaders.py", line 115, in load
source, filename, uptodate = self.get_source(environment, name)
File "/<>/google_appengine/lib/jinja2-2.6/jinja2/loaders.py", line 226, in get_source
if not self.provider.has_resource(p):
File "/<>/google_appengine/lib/setuptools-0.6c11/pkg_resources.py", line 1170, in has_resource
return self._has(self._fn(self.module_path, resource_name))
File "/<>/google_appengine/lib/setuptools-0.6c11/pkg_resources.py", line 1218, in _has
"Can't perform this operation for unregistered loader type"
NotImplementedError: Can't perform this operation for unregistered loader type
当我使用 aFileSystemLoader
代替它时,它可以工作,但是我猜这在 zip 中不起作用:
import common.templates
env = Environment(loader=FileSystemLoader(common.templates.__path__))
我已经看到了这个 SO question但这不是问题。
更新:
我刚刚在 1.7.6 SDK 中测试了该示例并且可以正常工作,但是我希望看到它可以在较旧的 SDK(或 old_dev_appserver)上运行。