Vert.x Java開發(fā)指南——第七章 公開Web API

感興趣的朋友爽雄,可以關注微信服務號“猿學堂社區(qū)”蝠检,或加入“猿學堂社區(qū)”微信交流群

版權聲明:本文由作者自行翻譯,未經作者授權挚瘟,不得隨意轉發(fā)叹谁。

使用我們已經講到的vertx-web模塊公開Web HTTP/JSON API非常簡單饲梭。我們將使用以下URL方案公開Web API:

  1. GET /api/pages 給出一個包含所有wiki頁面名稱和標識的文檔
  2. POST /api/pages 從一個文檔創(chuàng)建新的wiki頁
  3. PUT /api/pages/:id 從一個文檔更新wiki頁面
  4. DELETE /api/pages/:id 刪除一個wiki頁面

下面是使用HTTPie命令行工具與這些API交互的截圖:

在這里插入圖片描述

7.1 Web子路由器

我們需要添加新的路由處理器到HttpServerVerticle類。雖然我們可以直接向現(xiàn)有的路由器添加處理程序焰檩,但我們也可以利用子路由器的優(yōu)勢來處理憔涉。它們允許將一個路由器掛載為另一個路由器的子路由器,這對組織和(或)重用handler非常有用析苫。

此處是API路由器的代碼:

Router apiRouter = Router.router(vertx);
apiRouter.get("/pages").handler(this::apiRoot);
apiRouter.get("/pages/:id").handler(this::apiGetPage);
apiRouter.post().handler(BodyHandler.create());
apiRouter.post("/pages").handler(this::apiCreatePage);
apiRouter.put().handler(BodyHandler.create());
apiRouter.put("/pages/:id").handler(this::apiUpdatePage);
apiRouter.delete("/pages/:id").handler(this::apiDeletePage);
router.mountSubRouter("/api", apiRouter); ①

① 這是我們掛載API路由器的位置兜叨,因此請求以/api開始的路徑將定向到apiRouter。

7.2 處理器

接下來是不同的API路由器處理器代碼衩侥。

7.2.1 根資源

private void apiRoot(RoutingContext context) {
    dbService.fetchAllPagesData(reply -> {
        JsonObject response = new JsonObject();
        if (reply.succeeded()) {
            List<JsonObject> pages = reply.result()
                .stream()
                .map(obj -> new JsonObject()
                    .put("id", obj.getInteger("ID")) ①
                    .put("name", obj.getString("NAME")))
                .collect(Collectors.toList());
                response.put("success", true)
                    .put("pages", pages); ②
                context.response().setStatusCode(200);
                context.response().putHeader("Content-Type", "application/json");
                context.response().end(response.encode()); ③
        } else {
            response.put("success", false)
                    .put("error", reply.cause().getMessage());
            context.response().setStatusCode(500);
            context.response().putHeader("Content-Type", "application/json");
            context.response().end(response.encode());
        }
    });
}

① 我們只是在頁面信息記錄對象中重新映射數(shù)據(jù)庫記錄国旷。

② 在響應載荷中,結果JSON數(shù)組成為pages鍵的值顿乒。

③ JsonObject#encode()給出了JSON數(shù)據(jù)的一個緊湊的String展現(xiàn)议街。

7.2.2 得到一個頁面

private void apiGetPage(RoutingContext context) {
    int id = Integer.valueOf(context.request().getParam("id"));
    dbService.fetchPageById(
            id,
            reply -> {
                JsonObject response = new JsonObject();
                if (reply.succeeded()) {
                    JsonObject dbObject = reply.result();
                    if (dbObject.getBoolean("found")) {
                        JsonObject payload = new JsonObject()
                                .put("name", dbObject.getString("name"))
                                .put("id", dbObject.getInteger("id"))
                                .put("markdown",dbObject.getString("content"))
                                .put("html",Processor.process(dbObject.getString("content")));
                        response.put("success", true).put("page", payload);
                        context.response().setStatusCode(200);
                    } else {
                        context.response().setStatusCode(404);
                        response.put("success", false).put("error","There is no page with ID " + id);
                    }
                } else {
                    response.put("success", false).put("error",reply.cause().getMessage());
                    context.response().setStatusCode(500);
                }
                context.response().putHeader("Content-Type",
                        "application/json");
                context.response().end(response.encode());
            });
}

7.2.3 創(chuàng)建一個頁面

private void apiCreatePage(RoutingContext context) {
    JsonObject page = context.getBodyAsJson();
    if (!validateJsonPageDocument(context, page, "name", "markdown")) {
        return;
    }
    dbService.createPage(
            page.getString("name"),
            page.getString("markdown"),
            reply -> {
                if (reply.succeeded()) {
                    context.response().setStatusCode(201);
                    context.response().putHeader("Content-Type","application/json");
                    context.response().end(new JsonObject().put("success", true).encode());
                } else {
                    context.response().setStatusCode(500);
                    context.response().putHeader("Content-Type","application/json");
                    context.response().end(new JsonObject()
                            .put("success", false)
                            .put("error",reply.cause().getMessage()).encode());
                }
            }
    );
}

這個處理器和其它處理器都需要處理輸入的JSON文檔。下面的validateJsonPageDocument方法是一個驗證并在早期報告錯誤的助手璧榄,因此處理的剩余部分假定存在某些JSON條目特漩。

private boolean validateJsonPageDocument(RoutingContext context, JsonObject page, String... expectedKeys) {
    if (!Arrays.stream(expectedKeys).allMatch(page::containsKey)) {
        LOGGER.error("Bad page creation JSON payload: " + page.encodePrettily() + " from " + context.request().
                remoteAddress());
        context.response().setStatusCode(400);
        context.response().putHeader("Content-Type", "application/json");
        context.response().end(new JsonObject()
                .put("success", false)
                .put("error", "Bad request payload").encode());
        return false;
    }
    return true;
}

7.2.4 更新一個頁面

private void apiUpdatePage(RoutingContext context) {
    int id = Integer.valueOf(context.request().getParam("id"));
    JsonObject page = context.getBodyAsJson();
    if (!validateJsonPageDocument(context, page, "markdown")) {
        return;
    }
    dbService.savePage(id, page.getString("markdown"), reply -> {
        handleSimpleDbReply(context, reply);
    });
}

handleSimpleDbReply方法是一個助手,用于完成請求處理:

private void handleSimpleDbReply(RoutingContext context, AsyncResult<Void> reply) {
    if (reply.succeeded()) {
        context.response().setStatusCode(200);
        context.response().putHeader("Content-Type", "application/json");
        context.response().end(new JsonObject().put("success", true).encode());
    } else {
        context.response().setStatusCode(500);
        context.response().putHeader("Content-Type", "application/json");
        context.response().end(new JsonObject()
            .put("success", false)
            .put("error", reply.cause().getMessage()).encode());
    }
}

7.2.5 刪除一個頁面

private void apiDeletePage(RoutingContext context) {
    int id = Integer.valueOf(context.request().getParam("id"));
    dbService.deletePage(id, reply -> {
        handleSimpleDbReply(context, reply);
    });
}

7.3 單元測試API

我們在io.vertx.guides.wiki.http.ApiTest類中編寫一個基礎的測試用例骨杂。

前導(preamble)包括準備測試環(huán)境涂身。HTTP服務器Verticle依賴數(shù)據(jù)庫Verticle,因此我們需要在測試Vert.x上下文中同時部署這兩個Verticle:

@RunWith(VertxUnitRunner.class)
public class ApiTest {
    private Vertx vertx;
    private WebClient webClient;
    @Before
    public void prepare(TestContext context) {
        vertx = Vertx.vertx();
        JsonObject dbConf = new JsonObject()
            .put(WikiDatabaseVerticle.CONFIG_WIKIDB_JDBC_URL,                   "jdbc:hsqldb:mem:testdb;shutdown=true") ①
            .put(WikiDatabaseVerticle.CONFIG_WIKIDB_JDBC_MAX_POOL_SIZE, 4);
        vertx.deployVerticle(new WikiDatabaseVerticle(),
                new DeploymentOptions().setConfig(dbConf), context.asyncAssertSuccess());
        vertx.deployVerticle(new HttpServerVerticle(), context.asyncAssertSuccess());
        webClient = WebClient.create(vertx, new WebClientOptions()
            .setDefaultHost("localhost")
            .setDefaultPort(8080));
    }
    @After
    public void finish(TestContext context) {
        vertx.close(context.asyncAssertSuccess());
    }
    // (...)

① 我們使用了一個不同的JDBC URL搓蚪,以便使用一個內存數(shù)據(jù)庫進行測試蛤售。

正式的測試用例是一個簡單的場景,此處創(chuàng)造了所有類型的請求妒潭。它創(chuàng)建了一個頁面悴能,獲取它,更新它雳灾,然后刪除它:

@Test
public void play_with_api(TestContext context) {
    Async async = context.async();
    JsonObject page = new JsonObject().put("name", "Sample").put(
            "markdown", "# A page");
    Future<JsonObject> postRequest = Future.future();
    webClient.post("/api/pages").as(BodyCodec.jsonObject())
            .sendJsonObject(page, ar -> {
                if (ar.succeeded()) {
                    HttpResponse<JsonObject> postResponse = ar.result();
                    postRequest.complete(postResponse.body());
                } else {
                    context.fail(ar.cause());
                }
            });
    Future<JsonObject> getRequest = Future.future();
    postRequest.compose(h -> {
        webClient.get("/api/pages").as(BodyCodec.jsonObject()).send(ar -> {
            if (ar.succeeded()) {
                HttpResponse<JsonObject> getResponse = ar.result();
                getRequest.complete(getResponse.body());
            } else {
                context.fail(ar.cause());
            }
        });
    }, getRequest);
    Future<JsonObject> putRequest = Future.future();
    getRequest.compose(
            response -> {
                JsonArray array = response.getJsonArray("pages");
                context.assertEquals(1, array.size());
                context.assertEquals(0, array.getJsonObject(0).getInteger("id"));
                webClient.put("/api/pages/0")
                    .as(BodyCodec.jsonObject())
                    .sendJsonObject(new JsonObject().put("id", 0).put("markdown", "Oh Yeah!"),
                                ar -> {
                                    if (ar.succeeded()) {
                                        HttpResponse<JsonObject> putResponse = ar.result();
                                        putRequest.complete(putResponse.body());
                                    } else {
                                        context.fail(ar.cause());
                                    }
                                });
            }, putRequest);
    Future<JsonObject> deleteRequest = Future.future();
    putRequest.compose(
            response -> {
                context.assertTrue(response.getBoolean("success"));
                webClient.delete("/api/pages/0")
                        .as(BodyCodec.jsonObject())
                        .send(ar -> {
                            if (ar.succeeded()) {
                                HttpResponse<JsonObject> delResponse = ar.result();
                                deleteRequest.complete(delResponse.body());
                            } else {
                                context.fail(ar.cause());
                            }
                        });
            }, deleteRequest);
    deleteRequest.compose(response -> {
        context.assertTrue(response.getBoolean("success"));
        async.complete();
    }, Future.failedFuture("Oh?"));
}

這個測試使用了Future對象組合的方式漠酿,而不是嵌入式回調;最后的組合(compose)必須完成這個異步Future(指的是async)或者測試最后超時谎亩。

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末炒嘲,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子匈庭,更是在濱河造成了極大的恐慌夫凸,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件阱持,死亡現(xiàn)場離奇詭異夭拌,居然都是意外死亡,警方通過查閱死者的電腦和手機道逗,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進店門献烦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來滓窍,“玉大人,你說我怎么就攤上這事巩那±艉唬” “怎么了噪生?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長页藻。 經常有香客問我份帐,道長废境,這世上最難降的妖魔是什么噩凹? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任驮宴,我火速辦了婚禮幻赚,結果婚禮上落恼,老公的妹妹穿的比我還像新娘佳谦。我一直安慰自己钻蔑,他們只是感情好,可當我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著窗怒,像睡著了一般扬虚。 火紅的嫁衣襯著肌膚如雪辜昵。 梳的紋絲不亂的頭發(fā)上堪置,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天优构,我揣著相機與錄音钦椭,去河邊找鬼彪腔。 笑死德挣,一個胖子當著我的面吹牛格嗅,可吹牛的內容都是我干的屯掖。 我是一名探鬼主播贴铜,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼徘意,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了邑退?” 一聲冷哼從身側響起地技,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎作谚,沒想到半個月后妹懒,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體眨唬,經...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年好乐,在試婚紗的時候發(fā)現(xiàn)自己被綠了匾竿。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖反璃,靈堂內的尸體忽然破棺而出昵慌,到底是詐尸還是另有隱情斋攀,我是刑警寧澤,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響艰垂,放射性物質發(fā)生泄漏。R本人自食惡果不足惜胰柑,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望爬泥。 院中可真熱鬧柬讨,春花似錦、人聲如沸袍啡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽嗅剖。三九已至蛋逾,卻和暖如春茁瘦,著一層夾襖步出監(jiān)牢的瞬間撒遣,已是汗流浹背送淆。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工猿推, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留破托,地道東北人哼勇。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓契沫,卻偏偏與公主長得像捧毛,于是被迫代替她去往敵國和親欺旧。 傳聞我的和親對象是個殘疾皇子姑丑,可洞房花燭夜當晚...
    茶點故事閱讀 44,577評論 2 353

推薦閱讀更多精彩內容