44

I've been playing around with Tornado, and I've written some code that doesn't seem very nice.

I'm writing an app to store recipes as an example. These are my handlers:

handlers = [
    (r"/recipes/", RecipeHandler),
    (r"/recipes", RecipeSearchHandler), #so query params can be used to search
]

This lead me to writing this:

class RecipeHandler(RequestHandler):      
    def get(self):
        self.render('recipes/index.html')

class RecipeSearchHandler(RequestHandler):    
    def get(self):
        try:
            name = self.get_argument('name', True)
            self.write(name)
        # will do some searching
        except AssertionError:
            self.write("no params")
            # will probably redirect to /recipes/

Is there a better way to approach these URLs without a try/except? I'd like /recipes and /recipes/ to show the same thing, whereas /recipes?name=something would do a search, and ideally be a different handler.

4

4 に答える 4

53

GET リクエストにはもっと良い方法があります。ここの github のトルネード ソースにデモがあります。

# url handler
handlers = [(r"/entry/([^/]+)", EntryHandler),]

class EntryHandler(BaseHandler):
    def get(self, slug):
        entry = self.db.get("SELECT * FROM entries WHERE slug = %s", slug)
        if not entry: raise tornado.web.HTTPError(404)
        self.render("entry.html", entry=entry)

正規表現に一致する「テキスト」はすべて EntryHandler の get メソッドにスラッグ引数として渡されます。URL がどのハンドラーとも一致しない場合、ユーザーは 404 エラーを受け取ります。

別のフォールバックを提供したい場合は、パラメーターをオプションにすることができます

(r"/entry/([^/]*)", EntryHandler),

class EntryHandler(BaseHandler):
    def get(self, slug=None):
        pass

アップデート:

リンクの+1。ただし、この URL パターンを拡張して、次のように検索したい場合、より多くのパラメーターを含めることはできますか? /recipes?ingredient=chicken&style=indian – colinjameswebb

はい、そうです。

handlers = [
     (r'/(\d{4})/(\d{2})/(\d{2})/([a-zA-Z\-0-9\.:,_]+)/?', DetailHandler)
]

class DetailHandler(BaseHandler):
    def get(self, year, month, day, slug):
        pass
于 2012-05-23T19:39:10.547 に答える
43

get_argumentデフォルト値を指定できます。

details=self.get_argument("details", None, True)

引数が指定されている場合、引数が指定されていなくても例外は発生しません。

于 2012-12-18T04:01:00.413 に答える
13

トルネードにもget_arguments機能があります。指定された名前の引数のリストを返します。存在しない場合は、空のリスト ( [] ) を返します。try..catchブロックの代わりに Web サービスの入力をサニタイズするこの方法がよりクリーンであることがわかりました。

サンプル:
次の URL ハンドラーがあるとします。

(r"/recipe",GetRecipe)

そしてリクエストハンドラ:

class GetRecipe(RequestHandler):
    def get(self):
        recipe_id = self.get_arguments("rid")
        if recipe_id == []:
            # Handle me
            self.set_status(400)
            return self.finish("Invalid recipe id")
        self.write({"recipe_id":self.get_argument("rid")})


recipe_idself.get_argumentlist も値を保持しますが、この方法で便利に使用できることがわかりました。

結果は次のとおりです。

curl "http://localhost:8890/recipe" -v

*   Trying 127.0.0.1...
* Connected to localhost (127.0.0.1) port 8890 (#0)
> GET /recipe HTTP/1.1
> User-Agent: curl/7.35.0
> Host: localhost:8890
> Accept: */*
> 
< HTTP/1.1 400 Bad Request
< Content-Length: 17
< Content-Type: text/html; charset=UTF-8
* Server TornadoServer/1.1.1 is not blacklisted
< Server: TornadoServer/1.1.1
< 
* Connection #0 to host localhost left intact
Invalid recipe id

curl "http://localhost:8890/recipe?rid=230" -v
*   Trying 127.0.0.1...
* Connected to localhost (127.0.0.1) port 8890 (#0)
> GET /recipe?rid=230 HTTP/1.1
> User-Agent: curl/7.35.0
> Host: localhost:8890
> Accept: */*
> 
< HTTP/1.1 200 OK
< Content-Length: 20
< Etag: "d69ecb9086a20160178ade6b13eb0b3959aa13c6"
< Content-Type: text/javascript; charset=UTF-8
* Server TornadoServer/1.1.1 is not blacklisted
< Server: TornadoServer/1.1.1
< 
* Connection #0 to host localhost left intact
{"recipe_id": "230"}

于 2015-12-22T05:52:36.300 に答える