From 77e1834091bd0b84352e2ed84bd96ac15dd27ff1 Mon Sep 17 00:00:00 2001 From: jangko Date: Wed, 22 Aug 2018 22:37:04 +0700 Subject: [PATCH 01/12] benchmark initial implementation --- .../asyncdispatch2/plaintext.dockerfile | 6 ++ tests/benchmark/asyncdispatch2/server.nim | 44 ++++++++ tests/benchmark/asyncnet/plaintext.dockerfile | 5 + tests/benchmark/asyncnet/server.nim | 47 ++++++++ tests/benchmark/benchmark.sh | 12 +++ tests/benchmark/bot.nim | 100 ++++++++++++++++++ tests/benchmark/mofuw/README.md | 11 ++ tests/benchmark/mofuw/plaintext.dockerfile | 6 ++ tests/benchmark/mofuw/server.nim | 15 +++ tests/benchmark/pipeline.lua | 16 +++ tests/benchmark/readme.md | 53 ++++++++++ 11 files changed, 315 insertions(+) create mode 100644 tests/benchmark/asyncdispatch2/plaintext.dockerfile create mode 100644 tests/benchmark/asyncdispatch2/server.nim create mode 100644 tests/benchmark/asyncnet/plaintext.dockerfile create mode 100644 tests/benchmark/asyncnet/server.nim create mode 100755 tests/benchmark/benchmark.sh create mode 100644 tests/benchmark/bot.nim create mode 100644 tests/benchmark/mofuw/README.md create mode 100644 tests/benchmark/mofuw/plaintext.dockerfile create mode 100644 tests/benchmark/mofuw/server.nim create mode 100644 tests/benchmark/pipeline.lua create mode 100644 tests/benchmark/readme.md diff --git a/tests/benchmark/asyncdispatch2/plaintext.dockerfile b/tests/benchmark/asyncdispatch2/plaintext.dockerfile new file mode 100644 index 000000000..2721b6dad --- /dev/null +++ b/tests/benchmark/asyncdispatch2/plaintext.dockerfile @@ -0,0 +1,6 @@ +FROM statusteam/nim-base +RUN nimble install -y https://github.com/status-im/nim-asyncdispatch2 +WORKDIR /server +COPY server.nim server.nim +RUN nim c -d:release server.nim +CMD ["./server"] diff --git a/tests/benchmark/asyncdispatch2/server.nim b/tests/benchmark/asyncdispatch2/server.nim new file mode 100644 index 000000000..7d9560621 --- /dev/null +++ b/tests/benchmark/asyncdispatch2/server.nim @@ -0,0 +1,44 @@ +import strutils, posix + +import asyncdispatch2 + +const DOUBLECRLF = @['\r'.byte, '\L'.byte, '\r'.byte, '\L'.byte] + +proc handleConnection(ss: StreamServer, st: StreamTransport) {.async.} = + var resp = "HTTP/1.1 200 OK\r\L" & + "Content-Length: 15\r\L\r\L" & + "Hello, World!\r\L" + var readBuff: array[256, byte] + while not st.atEof: + try: + let read = await st.readUntil(addr readBuff[0], len(readBuff), DOUBLECRLF) + except: + break + let w = await st.write(addr resp[0], resp.len) + assert w == resp.len + st.close() + +proc handleBreak(udata: pointer) = + var cdata = cast[ptr CompletionData](udata) + var svr = cast[StreamServer](cdata.udata) + echo "\nCTRL+C pressed, stopping server..." + svr.stop() + svr.close() + +proc serve(onAddress: string) = + let + ta = initTAddress(onAddress) + svr = createStreamServer(ta, handleConnection, {ReuseAddr}, backlog = 128) + when not defined(windows): + discard addSignal(SIGINT, handleBreak, udata = cast[pointer](svr)) + svr.start() + echo "Server started at ", ta + try: + waitFor svr.join() + except: + echo "Error happens: ", getCurrentExceptionMsg() + finally: + echo "bye!" + +when isMainModule: + serve("0.0.0.0:8080") diff --git a/tests/benchmark/asyncnet/plaintext.dockerfile b/tests/benchmark/asyncnet/plaintext.dockerfile new file mode 100644 index 000000000..b5fbd9776 --- /dev/null +++ b/tests/benchmark/asyncnet/plaintext.dockerfile @@ -0,0 +1,5 @@ +FROM statusteam/nim-base +WORKDIR /server +COPY server.nim server.nim +RUN nim c -d:release server.nim +CMD ["./server"] diff --git a/tests/benchmark/asyncnet/server.nim b/tests/benchmark/asyncnet/server.nim new file mode 100644 index 000000000..d74d46a0d --- /dev/null +++ b/tests/benchmark/asyncnet/server.nim @@ -0,0 +1,47 @@ +# Asyncdispatch2 Test Suite +# (c) Copyright 2018 +# Status Research & Development GmbH +# +# Licensed under either of +# Apache License, version 2.0, (LICENSE-APACHEv2) +# MIT license (LICENSE-MIT) + +import asyncnet, asyncdispatch + +type + KeyboardInterruptError = object of Exception + +proc handleCtrlC() {.noconv.} = + raise newException(KeyboardInterruptError, "Keyboard Interrupt") +setControlCHook(handleCtrlC) + +proc handleConnection(client: AsyncSocket) {.async.} = + var resp = + "HTTP/1.1 200 OK\r\L" & + "Content-Length: 15\r\L\r\L" & + "Hello, World!\r\L" + var line = newFutureVar[string]("asyncnet.recvLine") + line.mget() = newString(256) + while true: + line.mget().setLen(0) + await client.recvLineInto(line) + if line.mget().len == 0: + break + if line.mget().len == 2: # \r\L + await client.send(resp) + client.close() + +proc serve() {.async.} = + var server = newAsyncSocket() + server.setSockOpt(OptReuseAddr, true) + server.bindAddr(Port(8080)) + server.listen() + try: + while true: + let client = await server.accept() + asyncCheck handleConnection(client) + finally: + server.close() + +when isMainModule: + waitFor serve() diff --git a/tests/benchmark/benchmark.sh b/tests/benchmark/benchmark.sh new file mode 100755 index 000000000..30f26bc57 --- /dev/null +++ b/tests/benchmark/benchmark.sh @@ -0,0 +1,12 @@ +#!/bin/bash + +mkdir wrksrc +cd wrksrc +if [ ! -f 4.1.0.tar.gz ]; then + wget -nc https://github.com/wg/wrk/archive/4.1.0.tar.gz | tar xz --strip-components=1 + make > /dev/null + cp wrk .. +fi + +cd .. +nim c -d:release -r bot diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim new file mode 100644 index 000000000..2879c0763 --- /dev/null +++ b/tests/benchmark/bot.nim @@ -0,0 +1,100 @@ +import osproc, json, streams, strutils, os + +const + participants = ["mofuw", "asyncnet", "asyncdispatch2"] + +proc execDocker(command: string): JsonNode = + const + options = {poStdErrToStdOut, poUsePath, poEvalCommand} + var p = startProcess(command, args=[], env=nil, options=options) + var outp = outputStream(p) + result = newJArray() + var line = newStringOfCap(120) + while true: + # FIXME: converts CR-LF to LF. + if outp.readLine(line): + let node = parseJson(line) + result.add node + elif not running(p): break + close(p) + +proc buildImages() = + for c in participants: + let cmd = "docker image build -q -t \"bench-$1:latest\" -f $1/plaintext.dockerfile $1/" % [c] + let ret = execCmd(cmd) + if ret != 0: + raise newException(Exception, "cannot build image: " & c) + +proc killAndRemove(id: string) = + var ret = execCmd("docker kill " & id) + if ret != 0: + raise newException(Exception, "cannot kill container: " & id) + ret = execCmd("docker rm " & id) + if ret != 0: + raise newException(Exception, "cannot remove container: " & id) + +const + levels = [256, 1024, 4096, 16384] + maxConcurrency = levels[^1] + duration = 15 + serverHost = "bench-bot" + url = "http://127.0.0.1:8080/" + pipeline = 16 + accept = "text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7" + +proc runTest(name: string): seq[string] = + let maxThreads = countProcessors() + + var cmd = "docker run -d -p 8080:8080 bench-$1" % [name] + var ret = execCmd(cmd) + if ret != 0: + raise newException(Exception, "cannot run image: " & name) + + cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d 5 -c 8 --timeout 8 -t 8 $3" % + [serverHost, accept, url] + echo "Running Primer " & name + echo cmd + ret = execCmd(cmd) + if ret != 0: + raise newException(Exception, "cannot run primer: " & name) + sleep(5000) + + cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6" % + [serverHost, accept, $duration, $maxConcurrency, $maxThreads, url] + echo "Running Warmup " & name + echo cmd + ret = execCmd(cmd) + if ret != 0: + raise newException(Exception, "cannot run warmup: " & name) + sleep(5000) + + let m = execDocker("docker ps -a --format '{{json .}}'") + for x in m: + let ID = x["ID"] + killAndRemove(ID.getStr()) + + result = newSeqOfCap[string](levels.len) + for c in levels: + echo "Running Concurrency $1 for $2" % [$c, name] + let t = max(c, maxThreads) + cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % + [serverHost, accept, $duration, $c, $t, url, $pipeline] + result.add execProcess(cmd) + +proc main() = + buildImages() + + for p in participants: + let res = runTest(p) + + #let cmd = "docker image ls --format '{{json .}}'" + #let res = execDocker(cmd) + #for x in res: + # echo x["Repository"], "/", x["Tag"] + + # docker kill $(docker ps -q) # kill all running containers + # docker rm $(docker ps -a -q) # remove all stopped containers + #var k = execProcess("./wrk -c 8 -t 3 -s pipeline.lua http://127.0.0.1:8080/ -- 16") + #echo k + +main() diff --git a/tests/benchmark/mofuw/README.md b/tests/benchmark/mofuw/README.md new file mode 100644 index 000000000..b7b5540c6 --- /dev/null +++ b/tests/benchmark/mofuw/README.md @@ -0,0 +1,11 @@ +# Nim's Web Framework \"mofuw\" + +> mofuw is \***MO**re\* \***F**\*aster, \***U**ltra\* \***W**\*eb server. + +## Includes tests +- plaintext: `localhost:8080/plaintext` +- json: `localhost:8080/json` + +## Require +- Nim => 0.18.0 +- mofuw => 1.1.8 \ No newline at end of file diff --git a/tests/benchmark/mofuw/plaintext.dockerfile b/tests/benchmark/mofuw/plaintext.dockerfile new file mode 100644 index 000000000..0968d1468 --- /dev/null +++ b/tests/benchmark/mofuw/plaintext.dockerfile @@ -0,0 +1,6 @@ +FROM statusteam/nim-base +RUN nimble install -y mofuw packedjson +WORKDIR /mofuwApp +COPY server.nim server.nim +RUN nim c -d:release --threads:on -d:bufSize:512 server.nim +CMD ["./server"] diff --git a/tests/benchmark/mofuw/server.nim b/tests/benchmark/mofuw/server.nim new file mode 100644 index 000000000..afbfd739e --- /dev/null +++ b/tests/benchmark/mofuw/server.nim @@ -0,0 +1,15 @@ +import mofuw, packedjson + +proc h(ctx: MofuwCtx) {.async.} = + case ctx.getPath + of "/plaintext": + mofuwResp(HTTP200, "text/plain", "Hello, World!") + of "/json": + mofuwResp(HTTP200, "application/json", $(%{"message": %"Hello, World!"})) + else: + mofuwResp(HTTP404, "text/plain", "NOT FOUND") + +newServeCtx( + port = 8080, + handler = h +).serve() diff --git a/tests/benchmark/pipeline.lua b/tests/benchmark/pipeline.lua new file mode 100644 index 000000000..250772a90 --- /dev/null +++ b/tests/benchmark/pipeline.lua @@ -0,0 +1,16 @@ +init = function(args) + local r = {} + local depth = tonumber(args[1]) or 1 + for i=1,depth do + r[i] = wrk.format() + end + req = table.concat(r) +end + +request = function() + return req +end + +done = function(summary, latency, requests) + io.write("OK\n") +end diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md new file mode 100644 index 000000000..2af1a06ef --- /dev/null +++ b/tests/benchmark/readme.md @@ -0,0 +1,53 @@ +# Plaintext Benchmark + +## Overview +Basically this is a mini [TechEmpower](https://www.techempower.com/benchmarks/) +web framework test. It uses the same tools, same techniques, only smaller in scale. + +## Tools +* [Docker](https://www.docker.com/), a container platform +* [wrk](https://github.com/wg/wrk), Modern HTTP benchmarking tool + +## Participants +Participants in this benchmark are: +* __asyncdispatch2__, lang: Nim +* __asyncdispatch__, lang: Nim, from Nim stdlib +* __actix-raw__, lang: Rust, TFB round 16th rank 1 +* __fasthttp__, lang: Go, TFB round 16th rank 2 +* __ulib-plaintext_fit__, lang: C++, TFB round 16th rank 3 + +Participants from TFB round 16th are impressive, how asyncdispatch2 compared to them? This is the goals of this benchmark. +The rank of TFB round 16th is based on plaintext category, running on a physical server. + +## Benchmark Setup + +* Each of the participants will be put into docker container. Each of them will play the role of a server +* The benchmarking tool __wrk__ will also have its own container, it will play the role of a client connecting to server(participant) +* There will be a __bench-bot__ coordinating the benchmark process. + +When the bench-bot started, it will build images of participants and executing them one by one. +After starting a participant container, bench-bot will run wrk, +then move to next participant, run wrk again for the next participant and doing it again until the last participant. +After the last participant benchmarked, bench-bot will write the benchmark result into a file: `benchmark_result.txt`. +Last thing performed by bench-bot is doing some cleanup and remove container images and instances. + +## How each test is run? + +Each test is executed as follows: +* Start the platform and framework using their start-up mechanisms. +* Run a 5-second __primer__ at 8 client-concurrency to verify that the server is in fact running. These results are not captured. +* Run a 15-second __warmup__ at 256 client-concurrency to allow lazy-initialization to execute and just-in-time compilation to run. These results are not captured. +* Run a 15-second __captured test__ for each of the concurrency levels (or iteration counts) exercised by the test type. + The high-concurrency plaintext test type is tested at 256, 1,024, 4,096, and 16,384 client-side concurrency. +* Stop the platform and framework. + +## How to replicate this test locally? +* First, you need to have docker installed, [see here how](https://www.digitalocean.com/community/tutorials/how-to-install-and-use-docker-on-ubuntu-18-04). +* Run the benchmark script `sh tests/benchmark/benchmark.sh` +* Wait until it finished, and get the result in `tests/benchmark/benchmark_result.txt` + +## How to add more participants? + +* Prepare a directory inside `tests/benchmark/` +* inside that directory prepare a `plaintext.dockerfile` and all necessary source code. +* add a entry in `bot.nim` participants constant list with the directory name. From 34c49ef2911910c9eac3f87c3e2035e2e1828e9b Mon Sep 17 00:00:00 2001 From: jangko Date: Fri, 24 Aug 2018 17:36:35 +0700 Subject: [PATCH 02/12] working prototype --- tests/benchmark/benchmark.sh | 1 + tests/benchmark/benchmark_result.txt | 15 +++ tests/benchmark/bot.nim | 137 +++++++++++++++++++-------- tests/benchmark/pipeline.lua | 6 +- tests/benchmark/readme.md | 8 +- 5 files changed, 121 insertions(+), 46 deletions(-) create mode 100644 tests/benchmark/benchmark_result.txt diff --git a/tests/benchmark/benchmark.sh b/tests/benchmark/benchmark.sh index 30f26bc57..d32348ba4 100755 --- a/tests/benchmark/benchmark.sh +++ b/tests/benchmark/benchmark.sh @@ -10,3 +10,4 @@ fi cd .. nim c -d:release -r bot +echo "$(cat benchmark_result.txt)" diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt new file mode 100644 index 000000000..3f3e153f2 --- /dev/null +++ b/tests/benchmark/benchmark_result.txt @@ -0,0 +1,15 @@ +mofuw + concurrency: 128, request/sec: 95682.82, transfer/sec: 12.045MiB + concurrency: 256, request/sec: 90446.30, transfer/sec: 11.386MiB + concurrency: 512, request/sec: 92075.01, transfer/sec: 11.591MiB + concurrency: 1024, failed +asyncnet + concurrency: 128, request/sec: 44196.54, transfer/sec: 2.276MiB + concurrency: 256, request/sec: 29674.21, transfer/sec: 1.528MiB + concurrency: 512, request/sec: 24244.77, transfer/sec: 1.249MiB + concurrency: 1024, failed +asyncdispatch2 + concurrency: 128, request/sec: 46276.68, transfer/sec: 2.383MiB + concurrency: 256, request/sec: 72831.57, transfer/sec: 3.751MiB + concurrency: 512, request/sec: 51450.58, transfer/sec: 2.65MiB + concurrency: 1024, failed diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 2879c0763..f4185a1d9 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -3,7 +3,7 @@ import osproc, json, streams, strutils, os const participants = ["mofuw", "asyncnet", "asyncdispatch2"] -proc execDocker(command: string): JsonNode = +proc execAndGetJson(command: string): JsonNode = const options = {poStdErrToStdOut, poUsePath, poEvalCommand} var p = startProcess(command, args=[], env=nil, options=options) @@ -13,28 +13,55 @@ proc execDocker(command: string): JsonNode = while true: # FIXME: converts CR-LF to LF. if outp.readLine(line): - let node = parseJson(line) - result.add node + if line[0] == '{': + let node = parseJson(line) + result.add node elif not running(p): break close(p) +proc execSilent(command: string): int = + const + options = {poStdErrToStdOut, poUsePath, poEvalCommand} + var p = startProcess(command, args=[], env=nil, options=options) + var outp = outputStream(p) + var line = newStringOfCap(120) + while true: + if outp.readLine(line): discard + elif not running(p): break + result = peekExitCode(p) + close(p) + proc buildImages() = for c in participants: let cmd = "docker image build -q -t \"bench-$1:latest\" -f $1/plaintext.dockerfile $1/" % [c] - let ret = execCmd(cmd) + let ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot build image: " & c) -proc killAndRemove(id: string) = - var ret = execCmd("docker kill " & id) +proc killContainer(id: string) = + var ret = execSilent("docker kill " & id) if ret != 0: raise newException(Exception, "cannot kill container: " & id) - ret = execCmd("docker rm " & id) + +proc removeContainer(id: string) = + var ret = execSilent("docker rm " & id) if ret != 0: raise newException(Exception, "cannot remove container: " & id) +proc killContainers() = + let m = execAndGetJson("docker ps --format '{{json .}}'") + for x in m: + let ID = x["ID"] + killContainer(ID.getStr()) + +proc removeContainers() = + let m = execAndGetJson("docker ps -a --format '{{json .}}'") + for x in m: + let ID = x["ID"] + removeContainer(ID.getStr()) + const - levels = [256, 1024, 4096, 16384] + levels = [128, 256, 512, 1024] maxConcurrency = levels[^1] duration = 15 serverHost = "bench-bot" @@ -42,59 +69,87 @@ const pipeline = 16 accept = "text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7" -proc runTest(name: string): seq[string] = +proc runTest(name: string): JsonNode = let maxThreads = countProcessors() + echo "** ", name, " **" var cmd = "docker run -d -p 8080:8080 bench-$1" % [name] - var ret = execCmd(cmd) + var ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run image: " & name) - cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d 5 -c 8 --timeout 8 -t 8 $3" % + cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d 5 -c 8 --timeout 8 -t 8 $3" % [serverHost, accept, url] - echo "Running Primer " & name - echo cmd - ret = execCmd(cmd) + echo " Running Primer" + ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run primer: " & name) - sleep(5000) + sleep(2000) - cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6" % - [serverHost, accept, $duration, $maxConcurrency, $maxThreads, url] - echo "Running Warmup " & name - echo cmd - ret = execCmd(cmd) + cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6" % + [serverHost, accept, $duration, $(256), $maxThreads, url] + echo " Running Warmup" + ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run warmup: " & name) - sleep(5000) - - let m = execDocker("docker ps -a --format '{{json .}}'") - for x in m: - let ID = x["ID"] - killAndRemove(ID.getStr()) + sleep(2000) - result = newSeqOfCap[string](levels.len) + var jar = newJArray() for c in levels: - echo "Running Concurrency $1 for $2" % [$c, name] + echo " Running Concurrency $1" % [$c] let t = max(c, maxThreads) - cmd = "wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % + let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % [serverHost, accept, $duration, $c, $t, url, $pipeline] - result.add execProcess(cmd) + let res = execAndGetJson(cmd) + if res.len == 1: + var json = res[0] + json.add("level", newJInt(c)) + json.add("success", newJBool(true)) + jar.add json + else: + var json = newJObject() + json.add("level", newJInt(c)) + json.add("success", newJBool(false)) + jar.add json + sleep(2000) + killContainers() + removeContainers() + + result = newJObject() + result.add("result", jar) + result.add("name", newJString(name)) + +proc renderResult(json: JsonNode, s: Stream) = + s.writeLine(json["name"].getStr()) + let bench = json["result"] + for c in bench: + if c["success"].getBool(): + let concurrency = c["level"].getInt() + let duration = c["duration"].getInt() + let requests = c["requests"].getInt() + let bytes = c["bytes"].getInt() + + let sec = duration.float / 1_000_000.0 + let rps = formatFloat(requests.float / sec, ffDecimal, 2) + let size = bytes.float / sec + let tps = formatSize(size.int) + + s.writeLine(" concurrency: $1, request/sec: $2, transfer/sec: $3" % [$concurrency, rps, tps]) + else: + let concurrency = c["level"].getInt() + s.writeLine(" concurrency: $1, failed" % [$concurrency]) proc main() = buildImages() - + + var resList = newSeq[JsonNode]() for p in participants: let res = runTest(p) - - #let cmd = "docker image ls --format '{{json .}}'" - #let res = execDocker(cmd) - #for x in res: - # echo x["Repository"], "/", x["Tag"] - - # docker kill $(docker ps -q) # kill all running containers - # docker rm $(docker ps -a -q) # remove all stopped containers - #var k = execProcess("./wrk -c 8 -t 3 -s pipeline.lua http://127.0.0.1:8080/ -- 16") - #echo k + resList.add res + + var s = newFileStream("benchmark_result.txt", fmWrite) + for res in resList: + res.renderResult(s) + s.close() main() diff --git a/tests/benchmark/pipeline.lua b/tests/benchmark/pipeline.lua index 250772a90..3d7718539 100644 --- a/tests/benchmark/pipeline.lua +++ b/tests/benchmark/pipeline.lua @@ -12,5 +12,9 @@ request = function() end done = function(summary, latency, requests) - io.write("OK\n") + io.write("{") + io.write('"duration":', summary.duration, ',') + io.write('"requests":', summary.requests, ',') + io.write('"bytes":', summary.bytes) + io.write("}") end diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md index 2af1a06ef..99a7eae39 100644 --- a/tests/benchmark/readme.md +++ b/tests/benchmark/readme.md @@ -21,12 +21,12 @@ The rank of TFB round 16th is based on plaintext category, running on a physical ## Benchmark Setup -* Each of the participants will be put into docker container. Each of them will play the role of a server -* The benchmarking tool __wrk__ will also have its own container, it will play the role of a client connecting to server(participant) +* Each of the participants will be put into docker container. Each of them will play the role of a server. +* The benchmarking tool __wrk__ will play the role of a client connecting to server(participant). * There will be a __bench-bot__ coordinating the benchmark process. When the bench-bot started, it will build images of participants and executing them one by one. -After starting a participant container, bench-bot will run wrk, +After starting a participant container, bench-bot will run wrk, grab the result, then move to next participant, run wrk again for the next participant and doing it again until the last participant. After the last participant benchmarked, bench-bot will write the benchmark result into a file: `benchmark_result.txt`. Last thing performed by bench-bot is doing some cleanup and remove container images and instances. @@ -38,7 +38,7 @@ Each test is executed as follows: * Run a 5-second __primer__ at 8 client-concurrency to verify that the server is in fact running. These results are not captured. * Run a 15-second __warmup__ at 256 client-concurrency to allow lazy-initialization to execute and just-in-time compilation to run. These results are not captured. * Run a 15-second __captured test__ for each of the concurrency levels (or iteration counts) exercised by the test type. - The high-concurrency plaintext test type is tested at 256, 1,024, 4,096, and 16,384 client-side concurrency. + The high-concurrency plaintext test type is tested at 128, 256, 512, and 1024 client-side concurrency. * Stop the platform and framework. ## How to replicate this test locally? From a783ff946ac01f397db047ec40a4596789d3d4bf Mon Sep 17 00:00:00 2001 From: jangko Date: Fri, 24 Aug 2018 20:20:50 +0700 Subject: [PATCH 03/12] add fasthttp --- tests/benchmark/benchmark_result.txt | 5 ++ tests/benchmark/bot.nim | 29 ++++++++-- tests/benchmark/fasthttp/plaintext.dockerfile | 16 ++++++ tests/benchmark/fasthttp/src/common/common.go | 56 +++++++++++++++++++ tests/benchmark/fasthttp/src/server/server.go | 36 ++++++++++++ 5 files changed, 136 insertions(+), 6 deletions(-) create mode 100644 tests/benchmark/fasthttp/plaintext.dockerfile create mode 100644 tests/benchmark/fasthttp/src/common/common.go create mode 100644 tests/benchmark/fasthttp/src/server/server.go diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index 3f3e153f2..8f0079ffd 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -13,3 +13,8 @@ asyncdispatch2 concurrency: 256, request/sec: 72831.57, transfer/sec: 3.751MiB concurrency: 512, request/sec: 51450.58, transfer/sec: 2.65MiB concurrency: 1024, failed +fasthttp + concurrency: 128, request/sec: 119381.15, transfer/sec: 17.419MiB + concurrency: 256, request/sec: 125082.98, transfer/sec: 18.251MiB + concurrency: 512, request/sec: 157172.24, transfer/sec: 22.933MiB + concurrency: 1024, failed diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index f4185a1d9..6f794c1e1 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,7 +1,7 @@ import osproc, json, streams, strutils, os const - participants = ["mofuw", "asyncnet", "asyncdispatch2"] + participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp"] proc execAndGetJson(command: string): JsonNode = const @@ -31,12 +31,15 @@ proc execSilent(command: string): int = result = peekExitCode(p) close(p) +proc buildImage(name: string) = + let cmd = "docker image build -t \"bench-$1:latest\" -f $1/plaintext.dockerfile $1/" % [name] + let ret = execCmd(cmd) + if ret != 0: + raise newException(Exception, "cannot build image: " & name) + proc buildImages() = for c in participants: - let cmd = "docker image build -q -t \"bench-$1:latest\" -f $1/plaintext.dockerfile $1/" % [c] - let ret = execSilent(cmd) - if ret != 0: - raise newException(Exception, "cannot build image: " & c) + buildImage(c) proc killContainer(id: string) = var ret = execSilent("docker kill " & id) @@ -139,7 +142,7 @@ proc renderResult(json: JsonNode, s: Stream) = let concurrency = c["level"].getInt() s.writeLine(" concurrency: $1, failed" % [$concurrency]) -proc main() = +proc runAllTest() = buildImages() var resList = newSeq[JsonNode]() @@ -152,4 +155,18 @@ proc main() = res.renderResult(s) s.close() +proc main() = + if paramCount() > 0: + let name = paramStr(1) + if name notin participants: + echo name & " is not a registered participant" + return + buildImage(name) + let res = runTest(name) + var s = newStringStream() + res.renderResult(s) + echo s.data + else: + runAllTest() + main() diff --git a/tests/benchmark/fasthttp/plaintext.dockerfile b/tests/benchmark/fasthttp/plaintext.dockerfile new file mode 100644 index 000000000..042a44f8b --- /dev/null +++ b/tests/benchmark/fasthttp/plaintext.dockerfile @@ -0,0 +1,16 @@ +FROM golang:1.10.1 + +ADD ./ /fasthttp +WORKDIR /fasthttp + +RUN mkdir bin +ENV GOPATH /fasthttp +ENV PATH ${GOPATH}/bin:${PATH} + +RUN rm -rf ./pkg/* +RUN go get -d -u github.com/valyala/fasthttp/... + +RUN rm -f ./server-mysql +RUN go build -gcflags='-l=4' server + +CMD ./server diff --git a/tests/benchmark/fasthttp/src/common/common.go b/tests/benchmark/fasthttp/src/common/common.go new file mode 100644 index 000000000..86a0ec1af --- /dev/null +++ b/tests/benchmark/fasthttp/src/common/common.go @@ -0,0 +1,56 @@ +package common + +import ( + "encoding/json" + "flag" + "log" + "net" + "sync" + "github.com/valyala/fasthttp" +) + +const worldRowCount = 10000 + +type JSONResponse struct { + Message string `json:"message"` +} + +type World struct { + Id int32 `json:"id"` + RandomNumber int32 `json:"randomNumber"` +} + +var listenAddr = flag.String("listenAddr", ":8080", "Address to listen to") + +func JSONHandler(ctx *fasthttp.RequestCtx) { + r := jsonResponsePool.Get().(*JSONResponse) + r.Message = "Hello, World!" + JSONMarshal(ctx, r) + jsonResponsePool.Put(r) +} + +var jsonResponsePool = &sync.Pool{ + New: func() interface{} { + return &JSONResponse{} + }, +} + +func PlaintextHandler(ctx *fasthttp.RequestCtx) { + ctx.SetContentType("text/plain") + ctx.WriteString("Hello, World!") +} + +func JSONMarshal(ctx *fasthttp.RequestCtx, v interface{}) { + ctx.SetContentType("application/json") + if err := json.NewEncoder(ctx).Encode(v); err != nil { + log.Fatalf("error in json.Encoder.Encode: %s", err) + } +} + +func GetListener() net.Listener { + ln, err := net.Listen("tcp4", *listenAddr) + if err != nil { + log.Fatal(err) + } + return ln +} diff --git a/tests/benchmark/fasthttp/src/server/server.go b/tests/benchmark/fasthttp/src/server/server.go new file mode 100644 index 000000000..c5c1a7a7b --- /dev/null +++ b/tests/benchmark/fasthttp/src/server/server.go @@ -0,0 +1,36 @@ +package main + +import ( + "flag" + "log" + "github.com/valyala/fasthttp" + "common" +) + +func main() { + flag.Parse() + + var err error + + s := &fasthttp.Server{ + Handler: mainHandler, + Name: "go", + } + ln := common.GetListener() + if err = s.Serve(ln); err != nil { + log.Fatalf("Error when serving incoming connections: %s", err) + } +} + +func mainHandler(ctx *fasthttp.RequestCtx) { + path := ctx.Path() + switch string(path) { + case "/plaintext": + common.PlaintextHandler(ctx) + case "/json": + common.JSONHandler(ctx) + default: + ctx.Error("unexpected path", fasthttp.StatusBadRequest) + } +} + From 93c10e42e76e4592e40778a2803c198001c50b1d Mon Sep 17 00:00:00 2001 From: jangko Date: Mon, 27 Aug 2018 21:23:07 +0700 Subject: [PATCH 04/12] add rust actix --- tests/benchmark/actix-raw/Cargo.lock | 1543 +++++++++++++++++ tests/benchmark/actix-raw/Cargo.toml | 28 + .../benchmark/actix-raw/plaintext.dockerfile | 11 + tests/benchmark/actix-raw/src/main.rs | 102 ++ tests/benchmark/actix-raw/src/utils.rs | 70 + tests/benchmark/benchmark_result.txt | 5 + tests/benchmark/bot.nim | 2 +- 7 files changed, 1760 insertions(+), 1 deletion(-) create mode 100644 tests/benchmark/actix-raw/Cargo.lock create mode 100644 tests/benchmark/actix-raw/Cargo.toml create mode 100644 tests/benchmark/actix-raw/plaintext.dockerfile create mode 100644 tests/benchmark/actix-raw/src/main.rs create mode 100644 tests/benchmark/actix-raw/src/utils.rs diff --git a/tests/benchmark/actix-raw/Cargo.lock b/tests/benchmark/actix-raw/Cargo.lock new file mode 100644 index 000000000..4d6900613 --- /dev/null +++ b/tests/benchmark/actix-raw/Cargo.lock @@ -0,0 +1,1543 @@ +[[package]] +name = "actix" +version = "0.7.1" +dependencies = [ + "actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)", + "actix-web 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)", + "askama 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "diesel 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", + "phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)", + "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "actix" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "actix_derive 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)", + "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "crossbeam-channel 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)", + "failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-signal 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "trust-dns-resolver 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)", + "uuid 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "actix-web" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)", + "base64 0.9.2 (registry+https://github.com/rust-lang/crates.io-index)", + "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "cookie 0.11.0 (registry+https://github.com/rust-lang/crates.io-index)", + "encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", + "failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "futures-cpupool 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "h2 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", + "htmlescape 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)", + "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", + "httparse 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)", + "language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "lazycell 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", + "mime_guess 2.0.0-alpha.6 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", + "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", + "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", + "percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", + "regex 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)", + "serde_urlencoded 0.5.3 (registry+https://github.com/rust-lang/crates.io-index)", + "sha1 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)", + "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", + "time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", + "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "actix_derive" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "aho-corasick" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "arrayvec" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "nodrop 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "askama" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "askama_derive 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", + "askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "askama_derive" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", + "nom 4.0.0 (registry+https://github.com/rust-lang/crates.io-index)", + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "askama_shared" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "toml 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "backtrace" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)", + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "dbghelp-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)", + "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "backtrace" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)", + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "backtrace-sys" +version = "0.1.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cc 1.0.22 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "base64" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "safemem 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "bitflags" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "byteorder" +version = "1.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "bytes" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "cc" +version = "1.0.22" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "cfg-if" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "cloudabi" +version = "0.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "cookie" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", + "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "crossbeam-channel" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", + "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "crossbeam-deque" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "arrayvec 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)", + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "memoffset 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", + "scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "crossbeam-utils" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "dbghelp-sys" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "diesel" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "diesel_derives 1.3.0 (registry+https://github.com/rust-lang/crates.io-index)", + "pq-sys 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "diesel_derives" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.13.11 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "dtoa" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "encoding" +version = "0.2.33" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding-index-japanese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", + "encoding-index-korean 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", + "encoding-index-simpchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", + "encoding-index-singlebyte 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", + "encoding-index-tradchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding-index-japanese" +version = "1.20141219.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding-index-korean" +version = "1.20141219.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding-index-simpchinese" +version = "1.20141219.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding-index-singlebyte" +version = "1.20141219.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding-index-tradchinese" +version = "1.20141219.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "encoding_index_tests" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "error-chain" +version = "0.1.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "backtrace 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "error-chain" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "failure" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", + "failure_derive 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "failure_derive" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", + "synstructure 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "fnv" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "fuchsia-zircon" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", + "fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "fuchsia-zircon-sys" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "futures" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "futures-cpupool" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "h2" +version = "0.1.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", + "indexmap 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", + "string 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "hostname" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "winutil 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "htmlescape" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "http" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", + "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "httparse" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "idna" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "unicode-bidi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)", + "unicode-normalization 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "indexmap" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "iovec" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "ipconfig" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "error-chain 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)", + "socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)", + "widestring 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", + "winreg 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "itoa" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "kernel32-sys" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "language-tags" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "lazy_static" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "lazycell" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "lazycell" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "libc" +version = "0.2.43" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "linked-hash-map" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "lock_api" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "owning_ref 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", + "scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "log" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "lru-cache" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "linked-hash-map 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "matches" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "memchr" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "memoffset" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "mime" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "unicase 2.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "mime_guess" +version = "2.0.0-alpha.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", + "phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", + "phf_codegen 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", + "unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "mio" +version = "0.6.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", + "fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", + "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", + "lazycell 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "miow 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", + "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", + "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "mio-uds" +version = "0.6.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "miow" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", + "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", + "ws2_32-sys 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "net2" +version = "0.2.33" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "nodrop" +version = "0.1.12" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "nom" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num_cpus" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "owning_ref" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "stable_deref_trait 1.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "parking_lot" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "lock_api 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", + "parking_lot_core 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "parking_lot_core" +version = "0.2.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "percent-encoding" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "phf" +version = "0.7.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "phf_codegen" +version = "0.7.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "phf_generator 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", + "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "phf_generator" +version = "0.7.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "phf_shared" +version = "0.7.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "siphasher 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)", + "unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "pq-sys" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "vcpkg 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "proc-macro2" +version = "0.3.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "proc-macro2" +version = "0.4.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "quick-error" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "quote" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "quote" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "rand" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "rand" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cloudabi 0.0.3 (registry+https://github.com/rust-lang/crates.io-index)", + "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "rand_core 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "rand_core" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "redox_syscall" +version = "0.1.40" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "regex" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "aho-corasick 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", + "regex-syntax 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)", + "thread_local 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", + "utf8-ranges 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "regex-syntax" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "ucd-util 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "resolv-conf" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "hostname 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "rustc-demangle" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "ryu" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "safemem" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "scopeguard" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "serde" +version = "1.0.75" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "serde_derive" +version = "1.0.75" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "serde_json" +version = "1.0.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", + "ryu 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "serde_urlencoded" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "dtoa 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", + "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", + "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "sha1" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "siphasher" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "slab" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "smallvec" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "unreachable 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "socket2" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "stable_deref_trait" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "string" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "syn" +version = "0.13.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", + "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "syn" +version = "0.14.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "synstructure" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", + "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", + "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", + "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "thread_local" +version = "0.3.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "time" +version = "0.1.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-current-thread 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-fs 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-uds 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-codec" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-current-thread" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-executor" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-fs" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-io" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-reactor" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", + "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", + "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-signal" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-tcp" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-threadpool" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "crossbeam-deque 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)", + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-timer" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-udp" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "tokio-uds" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", + "mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "toml" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "trust-dns-proto" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", + "socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", + "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "trust-dns-resolver" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", + "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", + "ipconfig 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", + "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", + "lru-cache 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", + "resolv-conf 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)", + "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", + "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "trust-dns-proto 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "ucd-util" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "unicase" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "unicase" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "unicode-bidi" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "unicode-normalization" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "unicode-xid" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "unreachable" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "url" +version = "1.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", + "idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "utf8-ranges" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "uuid" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "vcpkg" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "version_check" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "void" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "widestring" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "winapi" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "winapi" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi-i686-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi-x86_64-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "winapi-build" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" + +[[package]] +name = "winreg" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "winutil" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "ws2_32-sys" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +dependencies = [ + "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", + "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[metadata] +"checksum actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)" = "0470f5163ac229ed0329f6abbd6f775591b06ec5ecbf2f2589c16ab8bd6f76b8" +"checksum actix-web 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)" = "d0eddc44f25a168d776edf60477bd3d46566f89286e66288da8e196072e5a870" +"checksum actix_derive 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "5b9d1525ef45e5e021f0b93dace157dcab5d792acb4cc78f3213787d65e2bb92" +"checksum aho-corasick 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)" = "68f56c7353e5a9547cbd76ed90f7bb5ffc3ba09d4ea9bd1d8c06c8b1142eeb5a" +"checksum arrayvec 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)" = "a1e964f9e24d588183fcb43503abda40d288c8657dfc27311516ce2f05675aef" +"checksum askama 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c796fa924bb44233b38ae23020bfb229244e61910648d5f7bdc3659fcdf0572e" +"checksum askama_derive 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "57968dab3a9512b88af88012896e5e1d353e314afd5d190a98d0d4762c8f0a11" +"checksum askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "a180253955cc7bc732fdd18fad8aa9f04a4891fc7aa9c6ce525c128a1de3a146" +"checksum backtrace 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)" = "346d7644f0b5f9bc73082d3b2236b69a05fd35cce0cfa3724e184e6a5c9e2a2f" +"checksum backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)" = "89a47830402e9981c5c41223151efcced65a0510c13097c769cede7efb34782a" +"checksum backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)" = "c66d56ac8dabd07f6aacdaf633f4b8262f5b3601a810a0dcddffd5c22c69daa0" +"checksum base64 0.9.2 (registry+https://github.com/rust-lang/crates.io-index)" = "85415d2594767338a74a30c1d370b2f3262ec1b4ed2d7bba5b3faf4de40467d9" +"checksum bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)" = "228047a76f468627ca71776ecdebd732a3423081fcf5125585bcd7c49886ce12" +"checksum byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "90492c5858dd7d2e78691cfb89f90d273a2800fc11d98f60786e5d87e2f83781" +"checksum bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)" = "e178b8e0e239e844b083d5a0d4a156b2654e67f9f80144d48398fcd736a24fb8" +"checksum cc 1.0.22 (registry+https://github.com/rust-lang/crates.io-index)" = "4a6007c146fdd28d4512a794b07ffe9d8e89e6bf86e2e0c4ddff2e1fb54a0007" +"checksum cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "0c4e7bb64a8ebb0d856483e1e682ea3422f883c5f5615a90d51a2c82fe87fdd3" +"checksum cloudabi 0.0.3 (registry+https://github.com/rust-lang/crates.io-index)" = "ddfc5b9aa5d4507acaf872de71051dfd0e309860e88966e1051e462a077aac4f" +"checksum cookie 0.11.0 (registry+https://github.com/rust-lang/crates.io-index)" = "1465f8134efa296b4c19db34d909637cb2bf0f7aaf21299e23e18fa29ac557cf" +"checksum crossbeam-channel 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "6c0a94250b0278d7fc5a894c3d276b11ea164edc8bf8feb10ca1ea517b44a649" +"checksum crossbeam-deque 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "3486aefc4c0487b9cb52372c97df0a48b8c249514af1ee99703bf70d2f2ceda1" +"checksum crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "30fecfcac6abfef8771151f8be4abc9e4edc112c2bcb233314cafde2680536e9" +"checksum crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)" = "677d453a17e8bd2b913fa38e8b9cf04bcdbb5be790aa294f2389661d72036015" +"checksum dbghelp-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "97590ba53bcb8ac28279161ca943a924d1fd4a8fb3fa63302591647c4fc5b850" +"checksum diesel 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)" = "e71e7a348ae6064e86c4cf0709f0e4c3ef6f30e8e7d3dc05737164af4ebd3511" +"checksum diesel_derives 1.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "03bcaf77491f53e400d5ee3bdd57142ea4e1c47fe9217b3361ff9a76ca0e3d37" +"checksum dtoa 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)" = "6d301140eb411af13d3115f9a562c85cc6b541ade9dfa314132244aaee7489dd" +"checksum encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)" = "6b0d943856b990d12d3b55b359144ff341533e516d94098b1d3fc1ac666d36ec" +"checksum encoding-index-japanese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "04e8b2ff42e9a05335dbf8b5c6f7567e5591d0d916ccef4e0b1710d32a0d0c91" +"checksum encoding-index-korean 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "4dc33fb8e6bcba213fe2f14275f0963fd16f0a02c878e3095ecfdf5bee529d81" +"checksum encoding-index-simpchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "d87a7194909b9118fc707194baa434a4e3b0fb6a5a757c73c3adb07aa25031f7" +"checksum encoding-index-singlebyte 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "3351d5acffb224af9ca265f435b859c7c01537c0849754d3db3fdf2bfe2ae84a" +"checksum encoding-index-tradchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "fd0e20d5688ce3cab59eb3ef3a2083a5c77bf496cb798dc6fcdb75f323890c18" +"checksum encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "a246d82be1c9d791c5dfde9a2bd045fc3cbba3fa2b11ad558f27d01712f00569" +"checksum error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "faa976b4fd2e4c2b2f3f486874b19e61944d3de3de8b61c9fcf835d583871bcc" +"checksum error-chain 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)" = "6930e04918388a9a2e41d518c25cf679ccafe26733fb4127dbf21993f2575d46" +"checksum failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7efb22686e4a466b1ec1a15c2898f91fa9cb340452496dca654032de20ff95b9" +"checksum failure_derive 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "946d0e98a50d9831f5d589038d2ca7f8f455b1c21028c0db0e84116a12696426" +"checksum fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)" = "2fad85553e09a6f881f739c29f0b00b0f01357c743266d478b68951ce23285f3" +"checksum fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "2e9763c69ebaae630ba35f74888db465e49e259ba1bc0eda7d06f4a067615d82" +"checksum fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "3dcaa9ae7725d12cdb85b3ad99a434db70b468c09ded17e012d86b5c1010f7a7" +"checksum futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)" = "884dbe32a6ae4cd7da5c6db9b78114449df9953b8d490c9d7e1b51720b922c62" +"checksum futures-cpupool 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "ab90cde24b3319636588d0c35fe03b1333857621051837ed769faefb4c2162e4" +"checksum h2 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "a27e7ed946e8335bdf9a191bc1b9b14a03ba822d013d2f58437f4fabcbd7fc2c" +"checksum hostname 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "21ceb46a83a85e824ef93669c8b390009623863b5c195d1ba747292c0c72f94e" +"checksum htmlescape 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)" = "e9025058dae765dee5070ec375f591e2ba14638c63feff74f13805a72e523163" +"checksum http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)" = "dca621d0fa606a5ff2850b6e337b57ad6137ee4d67e940449643ff45af6874c6" +"checksum httparse 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7b6288d7db100340ca12873fd4d08ad1b8f206a9457798dfb17c018a33fee540" +"checksum idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "38f09e0f0b1fb55fdee1f17470ad800da77af5186a1a76c026b679358b7e844e" +"checksum indexmap 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "08173ba1e906efb6538785a8844dd496f5d34f0a2d88038e95195172fc667220" +"checksum iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "dbe6e417e7d0975db6512b90796e8ce223145ac4e33c377e4a42882a0e88bb08" +"checksum ipconfig 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "fccb81dd962b29a25de46c4f46e497b75117aa816468b6fff7a63a598a192394" +"checksum itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "5adb58558dcd1d786b5f0bd15f3226ee23486e24b7b58304b60f64dc68e62606" +"checksum kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7507624b29483431c0ba2d82aece8ca6cdba9382bff4ddd0f7490560c056098d" +"checksum language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a91d884b6667cd606bb5a69aa0c99ba811a115fc68915e7056ec08a46e93199a" +"checksum lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ca488b89a5657b0a2ecd45b95609b3e848cf1755da332a0da46e2b2b1cb371a7" +"checksum lazycell 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)" = "a6f08839bc70ef4a3fe1d566d5350f519c5912ea86be0df1740a7d247c7fc0ef" +"checksum lazycell 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "d33a48d0365c96081958cc663eef834975cb1e8d8bea3378513fc72bdbf11e50" +"checksum libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)" = "76e3a3ef172f1a0b9a9ff0dd1491ae5e6c948b94479a3021819ba7d860c8645d" +"checksum linked-hash-map 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7860ec297f7008ff7a1e3382d7f7e1dcd69efc94751a2284bafc3d013c2aa939" +"checksum lock_api 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "949826a5ccf18c1b3a7c3d57692778d21768b79e46eb9dd07bfc4c2160036c54" +"checksum log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)" = "cba860f648db8e6f269df990180c2217f333472b4a6e901e97446858487971e2" +"checksum lru-cache 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "4d06ff7ff06f729ce5f4e227876cb88d10bc59cd4ae1e09fbb2bde15c850dc21" +"checksum matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "7ffc5c5338469d4d3ea17d269fa8ea3512ad247247c30bd2df69e68309ed0a08" +"checksum memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a3b4142ab8738a78c51896f704f83c11df047ff1bda9a92a661aa6361552d93d" +"checksum memoffset 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "0f9dc261e2b62d7a622bf416ea3c5245cdd5d9a7fcc428c0d06804dfce1775b3" +"checksum mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)" = "4b082692d3f6cf41b453af73839ce3dfc212c4411cbb2441dff80a716e38bd79" +"checksum mime_guess 2.0.0-alpha.6 (registry+https://github.com/rust-lang/crates.io-index)" = "30de2e4613efcba1ec63d8133f344076952090c122992a903359be5a4f99c3ed" +"checksum mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)" = "4fcfcb32d63961fb6f367bfd5d21e4600b92cd310f71f9dca25acae196eb1560" +"checksum mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)" = "84c7b5caa3a118a6e34dbac36504503b1e8dc5835e833306b9d6af0e05929f79" +"checksum miow 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "8c1f2f3b1cf331de6896aabf6e9d55dca90356cc9960cca7eaaf408a355ae919" +"checksum net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)" = "42550d9fb7b6684a6d404d9fa7250c2eb2646df731d1c06afc06dcee9e1bcf88" +"checksum nodrop 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "9a2228dca57108069a5262f2ed8bd2e82496d2e074a06d1ccc7ce1687b6ae0a2" +"checksum nom 4.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "898696750eb5c3ce5eb5afbfbe46e7f7c4e1936e19d3e97be4b7937da7b6d114" +"checksum num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "c51a3322e4bca9d212ad9a158a02abc6934d005490c054a2778df73a70aa0a30" +"checksum owning_ref 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "cdf84f41639e037b484f93433aa3897863b561ed65c6e59c7073d7c561710f37" +"checksum parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)" = "69376b761943787ebd5cc85a5bc95958651a22609c5c1c2b65de21786baec72b" +"checksum parking_lot_core 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)" = "4db1a8ccf734a7bce794cc19b3df06ed87ab2f3907036b693c68f56b4d4537fa" +"checksum percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "31010dd2e1ac33d5b46a5b413495239882813e0369f8ed8a5e266f173602f831" +"checksum phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "cec29da322b242f4c3098852c77a0ca261c9c01b806cae85a5572a1eb94db9a6" +"checksum phf_codegen 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "7d187f00cd98d5afbcd8898f6cf181743a449162aeb329dcd2f3849009e605ad" +"checksum phf_generator 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "03dc191feb9b08b0dc1330d6549b795b9d81aec19efe6b4a45aec8d4caee0c4b" +"checksum phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "b539898d22d4273ded07f64a05737649dc69095d92cb87c7097ec68e3f150b93" +"checksum pq-sys 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "6ac25eee5a0582f45a67e837e350d784e7003bd29a5f460796772061ca49ffda" +"checksum proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)" = "1b06e2f335f48d24442b35a19df506a835fb3547bc3c06ef27340da9acf5cae7" +"checksum proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)" = "ee5697238f0d893c7f0ecc59c0999f18d2af85e424de441178bcacc9f9e6cf67" +"checksum quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9274b940887ce9addde99c4eee6b5c44cc494b182b97e73dc8ffdcb3397fd3f0" +"checksum quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9949cfe66888ffe1d53e6ec9d9f3b70714083854be20fd5e271b232a017401e8" +"checksum quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)" = "dd636425967c33af890042c483632d33fa7a18f19ad1d7ea72e8998c6ef8dea5" +"checksum rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)" = "8356f47b32624fef5b3301c1be97e5944ecdd595409cc5da11d05f211db6cfbd" +"checksum rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)" = "e464cd887e869cddcae8792a4ee31d23c7edd516700695608f5b98c67ee0131c" +"checksum rand_core 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "edecf0f94da5551fc9b492093e30b041a891657db7940ee221f9d2f66e82eef2" +"checksum redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)" = "c214e91d3ecf43e9a4e41e578973adeb14b474f2bee858742d127af75a0112b1" +"checksum regex 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)" = "67d0301b0c6804eca7e3c275119d0b01ff3b7ab9258a65709e608a66312a1025" +"checksum regex-syntax 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)" = "747ba3b235651f6e2f67dfa8bcdcd073ddb7c243cb21c442fc12395dfcac212d" +"checksum resolv-conf 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c62bd95a41841efdf7fca2ae9951e64a8d8eae7e5da196d8ce489a2241491a92" +"checksum rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)" = "bcfe5b13211b4d78e5c2cadfebd7769197d95c639c35a50057eb4c05de811395" +"checksum ryu 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "7153dd96dade874ab973e098cb62fcdbb89a03682e46b144fd09550998d4a4a7" +"checksum safemem 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "e27a8b19b835f7aea908818e871f5cc3a5a186550c30773be987e155e8163d8f" +"checksum scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "94258f53601af11e6a49f722422f6e3425c52b06245a5cf9bc09908b174f5e27" +"checksum serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)" = "22d340507cea0b7e6632900a176101fea959c7065d93ba555072da90aaaafc87" +"checksum serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)" = "234fc8b737737b148ccd625175fc6390f5e4dacfdaa543cb93a3430d984a9119" +"checksum serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)" = "44dd2cfde475037451fa99b7e5df77aa3cfd1536575fa8e7a538ab36dcde49ae" +"checksum serde_urlencoded 0.5.3 (registry+https://github.com/rust-lang/crates.io-index)" = "aaed41d9fb1e2f587201b863356590c90c1157495d811430a0c0325fe8169650" +"checksum sha1 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)" = "2579985fda508104f7587689507983eadd6a6e84dd35d6d115361f530916fa0d" +"checksum siphasher 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)" = "0b8de496cf83d4ed58b6be86c3a275b8602f6ffe98d3024a869e124147a9a3ac" +"checksum slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)" = "5f9776d6b986f77b35c6cf846c11ad986ff128fe0b2b63a3628e3755e8d3102d" +"checksum smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "153ffa32fd170e9944f7e0838edf824a754ec4c1fc64746fcc9fe1f8fa602e5d" +"checksum socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)" = "962a516af4d3a7c272cb3a1d50a8cc4e5b41802e4ad54cfb7bee8ba61d37d703" +"checksum stable_deref_trait 1.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "dba1a27d3efae4351c8051072d619e3ade2820635c3958d826bfea39d59b54c8" +"checksum string 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "00caf261d6f90f588f8450b8e1230fa0d5be49ee6140fdfbcb55335aff350970" +"checksum syn 0.13.11 (registry+https://github.com/rust-lang/crates.io-index)" = "14f9bf6292f3a61d2c716723fdb789a41bbe104168e6f496dc6497e531ea1b9b" +"checksum syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)" = "261ae9ecaa397c42b960649561949d69311f08eeaea86a65696e6e46517cf741" +"checksum synstructure 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "85bb9b7550d063ea184027c9b8c20ac167cd36d3e06b3a40bceb9d746dc1a7b7" +"checksum thread_local 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)" = "c6b53e329000edc2b34dbe8545fd20e55a333362d0a321909685a19bd28c3f1b" +"checksum time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)" = "d825be0eb33fda1a7e68012d51e9c7f451dc1a69391e7fdc197060bb8c56667b" +"checksum tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "fbb6a6e9db2702097bfdfddcb09841211ad423b86c75b5ddaca1d62842ac492c" +"checksum tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "881e9645b81c2ce95fcb799ded2c29ffb9f25ef5bef909089a420e5961dd8ccb" +"checksum tokio-current-thread 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "8fdfb899688ac16f618076bd09215edbfda0fd5dfecb375b6942636cb31fa8a7" +"checksum tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "84823b932d566bc3c6aa644df4ca36cb38593c50b7db06011fd4e12e31e4047e" +"checksum tokio-fs 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "b5cbe4ca6e71cb0b62a66e4e6f53a8c06a6eefe46cc5f665ad6f274c9906f135" +"checksum tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "8d6cc2de7725863c86ac71b0b9068476fec50834f055a243558ef1655bbd34cb" +"checksum tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "df6a7ea7d65e0fc1398de28959de8be96909986a7d2e01d4f86d3433dfb91aed" +"checksum tokio-signal 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "527342552ec4a6049f787ccc9e2d6e0eab77bfe6cb7ec7a05c0391e370f466c3" +"checksum tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "5b4c329b47f071eb8a746040465fa751bd95e4716e98daef6a9b4e434c17d565" +"checksum tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)" = "a5758cecb6e0633cea5d563ac07c975e04961690b946b04fd84e7d6445a8f6af" +"checksum tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "d03fa701f9578a01b7014f106b47f0a363b4727a7f3f75d666e312ab7acbbf1c" +"checksum tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "da941144b816d0dcda4db3a1ba87596e4df5e860a72b70783fe435891f80601c" +"checksum tokio-uds 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "424c1ed15a0132251813ccea50640b224c809d6ceafb88154c1a8775873a0e89" +"checksum toml 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "a0263c6c02c4db6c8f7681f9fd35e90de799ebd4cfdeab77a38f4ff6b3d8c0d9" +"checksum trust-dns-proto 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "32d7c204ee231f802aa821f9dc2195aa0d0269ef7e9f8c844208565c9e3981e4" +"checksum trust-dns-resolver 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)" = "4a821ad51a29816420b8cac4b026756b81c023630b97eaa4c8090637ee3508bd" +"checksum ucd-util 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "fd2be2d6639d0f8fe6cdda291ad456e23629558d466e2789d2c3e9892bda285d" +"checksum unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7f4765f83163b74f957c797ad9253caf97f103fb064d3999aea9568d09fc8a33" +"checksum unicase 2.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "284b6d3db520d67fbe88fd778c21510d1b0ba4a551e5d0fbb023d33405f6de8a" +"checksum unicode-bidi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)" = "49f2bd0c6468a8230e1db229cff8029217cf623c767ea5d60bfbd42729ea54d5" +"checksum unicode-normalization 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "6a0180bc61fc5a987082bfa111f4cc95c4caff7f9799f3e46df09163a937aa25" +"checksum unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "fc72304796d0818e357ead4e000d19c9c174ab23dc11093ac919054d20a6a7fc" +"checksum unreachable 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "382810877fe448991dfc7f0dd6e3ae5d58088fd0ea5e35189655f84e6814fa56" +"checksum url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2a321979c09843d272956e73700d12c4e7d3d92b2ee112b31548aef0d4efc5a6" +"checksum utf8-ranges 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "fd70f467df6810094968e2fce0ee1bd0e87157aceb026a8c083bcf5e25b9efe4" +"checksum uuid 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "e1436e58182935dcd9ce0add9ea0b558e8a87befe01c1a301e6020aeb0876363" +"checksum vcpkg 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "def296d3eb3b12371b2c7d0e83bfe1403e4db2d7a0bba324a12b21c4ee13143d" +"checksum version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "7716c242968ee87e5542f8021178248f267f295a5c4803beae8b8b7fd9bc6051" +"checksum void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "6a02e4885ed3bc0f2de90ea6dd45ebcbb66dacffe03547fadbb0eeae2770887d" +"checksum widestring 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7157704c2e12e3d2189c507b7482c52820a16dfa4465ba91add92f266667cadb" +"checksum winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a" +"checksum winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)" = "773ef9dcc5f24b7d850d0ff101e542ff24c3b090a9768e03ff889fdef41f00fd" +"checksum winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc" +"checksum winapi-i686-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" +"checksum winapi-x86_64-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" +"checksum winreg 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)" = "a27a759395c1195c4cc5cda607ef6f8f6498f64e78f7900f5de0a127a424704a" +"checksum winutil 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "7daf138b6b14196e3830a588acf1e86966c694d3e8fb026fb105b8b5dca07e6e" +"checksum ws2_32-sys 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "d59cefebd0c892fa2dd6de581e937301d8552cb44489cdff035c6187cb63fa5e" diff --git a/tests/benchmark/actix-raw/Cargo.toml b/tests/benchmark/actix-raw/Cargo.toml new file mode 100644 index 000000000..71f8ec13d --- /dev/null +++ b/tests/benchmark/actix-raw/Cargo.toml @@ -0,0 +1,28 @@ +[package] +name = "actix" +version = "0.7.1" + +[dependencies] +askama = "0.7" +serde = "1.0" +serde_json = "1.0" +serde_derive = "1.0" +rand = "0.5" +bytes = "0.4" +futures = "0.1" +http = "0.1" +diesel = { version = "1.2", features = ["postgres"] } +url = { version="1.7", features=["query_encoding"] } + +actix = "0.7" +actix-web = { version="0.7", default-features = false } + +phf = "0.7.22" + +[build-dependencies] +askama = "0.7" + +[profile.release] +lto = true +opt-level = 3 +codegen-units = 1 diff --git a/tests/benchmark/actix-raw/plaintext.dockerfile b/tests/benchmark/actix-raw/plaintext.dockerfile new file mode 100644 index 000000000..d7f8fe4ba --- /dev/null +++ b/tests/benchmark/actix-raw/plaintext.dockerfile @@ -0,0 +1,11 @@ +FROM rust:1.28-slim + +ADD ./ /actix +WORKDIR /actix + +RUN apt update -yqq && \ + apt install -yqq libpq-dev +RUN cargo clean +RUN RUSTFLAGS="-C target-cpu=native" cargo build --release + +CMD ./target/release/actix diff --git a/tests/benchmark/actix-raw/src/main.rs b/tests/benchmark/actix-raw/src/main.rs new file mode 100644 index 000000000..f6f22748e --- /dev/null +++ b/tests/benchmark/actix-raw/src/main.rs @@ -0,0 +1,102 @@ +extern crate actix; +extern crate actix_web; +extern crate bytes; +extern crate futures; +extern crate serde_json; +#[macro_use] +extern crate serde_derive; +extern crate askama; +extern crate rand; +extern crate url; +extern crate diesel; + +use actix::prelude::*; +use actix_web::server::{ + self, HttpHandler, HttpHandlerTask, HttpServer, Request, Writer, +}; +use actix_web::Error; +use futures::{Async, Poll}; + +mod utils; + +use utils::{Message, Writer as JsonWriter}; + +const HTTPOK: &[u8] = b"HTTP/1.1 200 OK\r\n"; +const HDR_SERVER: &[u8] = b"Server: Actix\r\n"; +const HDR_CTPLAIN: &[u8] = b"Content-Type: text/plain"; +const HDR_CTJSON: &[u8] = b"Content-Type: application/json"; +const BODY: &[u8] = b"Hello, World!"; + +struct App { +} + +impl HttpHandler for App { + type Task = Box; + + fn handle(&self, req: Request) -> Result, Request> { + { + let path = req.path(); + match path.len() { + 10 if path == "/plaintext" => return Ok(Box::new(Plaintext)), + 5 if path == "/json" => return Ok(Box::new(Json)), + _ => (), + } + } + Err(req) + } +} + +struct Plaintext; + +impl HttpHandlerTask for Plaintext { + fn poll_io(&mut self, io: &mut Writer) -> Poll { + { + let mut bytes = io.buffer(); + bytes.reserve(196); + bytes.extend_from_slice(HTTPOK); + bytes.extend_from_slice(HDR_SERVER); + bytes.extend_from_slice(HDR_CTPLAIN); + server::write_content_length(13, &mut bytes); + } + io.set_date(); + io.buffer().extend_from_slice(BODY); + Ok(Async::Ready(true)) + } +} + +struct Json; + +impl HttpHandlerTask for Json { + fn poll_io(&mut self, io: &mut Writer) -> Poll { + let message = Message { + message: "Hello, World!", + }; + + { + let mut bytes = io.buffer(); + bytes.reserve(196); + bytes.extend_from_slice(HTTPOK); + bytes.extend_from_slice(HDR_SERVER); + bytes.extend_from_slice(HDR_CTJSON); + server::write_content_length(27, &mut bytes); + } + io.set_date(); + serde_json::to_writer(JsonWriter(io.buffer()), &message).unwrap(); + Ok(Async::Ready(true)) + } +} + +fn main() { + let sys = System::new("techempower"); + + // start http server + HttpServer::new(move || { + vec![App { }] + }).backlog(8192) + .bind("0.0.0.0:8080") + .unwrap() + .start(); + + println!("Started http server: 127.0.0.1:8080"); + let _ = sys.run(); +} diff --git a/tests/benchmark/actix-raw/src/utils.rs b/tests/benchmark/actix-raw/src/utils.rs new file mode 100644 index 000000000..0870abad1 --- /dev/null +++ b/tests/benchmark/actix-raw/src/utils.rs @@ -0,0 +1,70 @@ +#![allow(dead_code)] +use bytes::BytesMut; +use std::{cmp, fmt, io}; + +use actix_web::http::Uri; +use url::form_urlencoded; + +pub const SIZE: usize = 31; + +#[derive(Serialize, Deserialize)] +pub struct Message { + pub message: &'static str, +} + +pub struct Writer<'a>(pub &'a mut BytesMut); + +impl<'a> io::Write for Writer<'a> { + fn write(&mut self, buf: &[u8]) -> io::Result { + self.0.extend_from_slice(buf); + Ok(buf.len()) + } + fn flush(&mut self) -> io::Result<()> { + Ok(()) + } +} + +pub struct StackWriter<'a>(pub &'a mut [u8], pub usize); + +impl<'a> io::Write for StackWriter<'a> { + #[inline] + fn write(&mut self, buf: &[u8]) -> io::Result { + let l = buf.len(); + let new = self.1 + l; + self.0[self.1..new].copy_from_slice(buf); + self.1 = new; + Ok(l) + } + #[inline] + fn flush(&mut self) -> io::Result<()> { + Ok(()) + } +} + +impl<'a> fmt::Write for StackWriter<'a> { + #[inline] + fn write_str(&mut self, s: &str) -> Result<(), fmt::Error> { + if !s.is_empty() { + let b = s.as_bytes(); + let l = b.len(); + let new = self.1 + l; + self.0[self.1..new].copy_from_slice(b); + self.1 = new; + } + Ok(()) + } +} + +pub fn get_query_param(uri: &Uri) -> u16 { + let mut q = None; + let q_str = if let Some(s) = uri.query() { s } else { "" }; + for (key, val) in form_urlencoded::parse(q_str.as_ref()) { + if key == "q" { + q = Some(val); + break; + } + } + + q.map(|q| cmp::min(500, cmp::max(1, q.parse::().ok().unwrap_or(1)))) + .unwrap_or(1) +} diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index 8f0079ffd..cbbeae899 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -18,3 +18,8 @@ fasthttp concurrency: 256, request/sec: 125082.98, transfer/sec: 18.251MiB concurrency: 512, request/sec: 157172.24, transfer/sec: 22.933MiB concurrency: 1024, failed +actix-raw + concurrency: 128, request/sec: 147586.97, transfer/sec: 10.275MiB + concurrency: 256, request/sec: 137716.71, transfer/sec: 9.588MiB + concurrency: 512, request/sec: 162553.94, transfer/sec: 11.317MiB + concurrency: 1024, failed diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 6f794c1e1..d3c7f4e0f 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,7 +1,7 @@ import osproc, json, streams, strutils, os const - participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp"] + participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw"] proc execAndGetJson(command: string): JsonNode = const From 5ced571be842fded00cc2c6ee5c1130d4f3f9365 Mon Sep 17 00:00:00 2001 From: jangko Date: Thu, 30 Aug 2018 11:51:29 +0700 Subject: [PATCH 05/12] single thread mofuw and actix --- tests/benchmark/actix-raw/Cargo.lock | 1543 ------------------------ tests/benchmark/actix-raw/Cargo.toml | 15 - tests/benchmark/actix-raw/src/main.rs | 43 +- tests/benchmark/actix-raw/src/utils.rs | 70 -- tests/benchmark/benchmark.sh | 13 - tests/benchmark/benchmark_result.txt | 40 +- tests/benchmark/bot.nim | 127 +- tests/benchmark/jsonfmt.lua | 7 + tests/benchmark/mofuw/server.nim | 18 +- tests/benchmark/readme.md | 5 +- 10 files changed, 166 insertions(+), 1715 deletions(-) delete mode 100644 tests/benchmark/actix-raw/Cargo.lock delete mode 100644 tests/benchmark/actix-raw/src/utils.rs delete mode 100755 tests/benchmark/benchmark.sh create mode 100644 tests/benchmark/jsonfmt.lua diff --git a/tests/benchmark/actix-raw/Cargo.lock b/tests/benchmark/actix-raw/Cargo.lock deleted file mode 100644 index 4d6900613..000000000 --- a/tests/benchmark/actix-raw/Cargo.lock +++ /dev/null @@ -1,1543 +0,0 @@ -[[package]] -name = "actix" -version = "0.7.1" -dependencies = [ - "actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)", - "actix-web 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)", - "askama 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "diesel 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", - "phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)", - "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "actix" -version = "0.7.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "actix_derive 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)", - "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "crossbeam-channel 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)", - "failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-signal 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "trust-dns-resolver 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)", - "uuid 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "actix-web" -version = "0.7.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)", - "base64 0.9.2 (registry+https://github.com/rust-lang/crates.io-index)", - "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "cookie 0.11.0 (registry+https://github.com/rust-lang/crates.io-index)", - "encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", - "failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "futures-cpupool 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "h2 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", - "htmlescape 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)", - "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", - "httparse 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)", - "language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "lazycell 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", - "mime_guess 2.0.0-alpha.6 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", - "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", - "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", - "percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", - "regex 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)", - "serde_urlencoded 0.5.3 (registry+https://github.com/rust-lang/crates.io-index)", - "sha1 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)", - "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", - "time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", - "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "actix_derive" -version = "0.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "aho-corasick" -version = "0.6.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "arrayvec" -version = "0.4.7" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "nodrop 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "askama" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "askama_derive 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", - "askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "askama_derive" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)", - "nom 4.0.0 (registry+https://github.com/rust-lang/crates.io-index)", - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "askama_shared" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "toml 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "backtrace" -version = "0.2.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)", - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "dbghelp-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)", - "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "backtrace" -version = "0.3.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)", - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "backtrace-sys" -version = "0.1.24" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cc 1.0.22 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "base64" -version = "0.9.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "safemem 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "bitflags" -version = "1.0.4" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "byteorder" -version = "1.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "bytes" -version = "0.4.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "cc" -version = "1.0.22" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "cfg-if" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "cloudabi" -version = "0.0.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "cookie" -version = "0.11.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", - "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "crossbeam-channel" -version = "0.2.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", - "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "crossbeam-deque" -version = "0.6.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "crossbeam-epoch" -version = "0.5.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "arrayvec 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)", - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "memoffset 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", - "scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "crossbeam-utils" -version = "0.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "dbghelp-sys" -version = "0.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "diesel" -version = "1.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "diesel_derives 1.3.0 (registry+https://github.com/rust-lang/crates.io-index)", - "pq-sys 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "diesel_derives" -version = "1.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.13.11 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "dtoa" -version = "0.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "encoding" -version = "0.2.33" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding-index-japanese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", - "encoding-index-korean 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", - "encoding-index-simpchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", - "encoding-index-singlebyte 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", - "encoding-index-tradchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding-index-japanese" -version = "1.20141219.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding-index-korean" -version = "1.20141219.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding-index-simpchinese" -version = "1.20141219.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding-index-singlebyte" -version = "1.20141219.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding-index-tradchinese" -version = "1.20141219.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "encoding_index_tests" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "error-chain" -version = "0.1.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "backtrace 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "error-chain" -version = "0.8.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "failure" -version = "0.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", - "failure_derive 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "failure_derive" -version = "0.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", - "synstructure 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "fnv" -version = "1.0.6" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "fuchsia-zircon" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)", - "fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "fuchsia-zircon-sys" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "futures" -version = "0.1.23" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "futures-cpupool" -version = "0.1.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "h2" -version = "0.1.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)", - "indexmap 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", - "string 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "hostname" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "winutil 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "htmlescape" -version = "0.3.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "http" -version = "0.1.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)", - "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "httparse" -version = "1.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "idna" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "unicode-bidi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)", - "unicode-normalization 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "indexmap" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "iovec" -version = "0.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "ipconfig" -version = "0.1.7" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "error-chain 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)", - "socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)", - "widestring 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", - "winreg 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "itoa" -version = "0.4.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "kernel32-sys" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "language-tags" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "lazy_static" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "lazycell" -version = "0.6.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "lazycell" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "libc" -version = "0.2.43" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "linked-hash-map" -version = "0.4.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "lock_api" -version = "0.1.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "owning_ref 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", - "scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "log" -version = "0.4.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "lru-cache" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "linked-hash-map 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "matches" -version = "0.1.8" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "memchr" -version = "2.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "memoffset" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "mime" -version = "0.3.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "unicase 2.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "mime_guess" -version = "2.0.0-alpha.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)", - "phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", - "phf_codegen 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", - "unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "mio" -version = "0.6.15" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", - "fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", - "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", - "lazycell 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "miow 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", - "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", - "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "mio-uds" -version = "0.6.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "miow" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)", - "net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", - "ws2_32-sys 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "net2" -version = "0.2.33" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "nodrop" -version = "0.1.12" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "nom" -version = "4.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "num_cpus" -version = "1.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "owning_ref" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "stable_deref_trait 1.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "parking_lot" -version = "0.6.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "lock_api 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", - "parking_lot_core 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "parking_lot_core" -version = "0.2.14" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "percent-encoding" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "phf" -version = "0.7.23" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "phf_codegen" -version = "0.7.23" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "phf_generator 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", - "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "phf_generator" -version = "0.7.23" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "phf_shared" -version = "0.7.23" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "siphasher 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)", - "unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "pq-sys" -version = "0.4.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "vcpkg 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "proc-macro2" -version = "0.3.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "proc-macro2" -version = "0.4.13" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "quick-error" -version = "1.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "quote" -version = "0.5.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "quote" -version = "0.6.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "rand" -version = "0.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "rand" -version = "0.5.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cloudabi 0.0.3 (registry+https://github.com/rust-lang/crates.io-index)", - "fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "rand_core 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "rand_core" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "redox_syscall" -version = "0.1.40" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "regex" -version = "1.0.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "aho-corasick 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)", - "regex-syntax 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)", - "thread_local 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)", - "utf8-ranges 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "regex-syntax" -version = "0.6.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "ucd-util 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "resolv-conf" -version = "0.6.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "hostname 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "rustc-demangle" -version = "0.1.9" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "ryu" -version = "0.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "safemem" -version = "0.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "scopeguard" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "serde" -version = "1.0.75" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "serde_derive" -version = "1.0.75" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "serde_json" -version = "1.0.26" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", - "ryu 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "serde_urlencoded" -version = "0.5.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "dtoa 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", - "itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)", - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", - "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "sha1" -version = "0.6.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "siphasher" -version = "0.2.3" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "slab" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "smallvec" -version = "0.6.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "unreachable 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "socket2" -version = "0.3.7" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "stable_deref_trait" -version = "1.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "string" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "syn" -version = "0.13.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", - "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "syn" -version = "0.14.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "synstructure" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)", - "quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)", - "syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)", - "unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "thread_local" -version = "0.3.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "time" -version = "0.1.40" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio" -version = "0.1.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-current-thread 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-fs 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-uds 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-codec" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-current-thread" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-executor" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-fs" -version = "0.1.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-io" -version = "0.1.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-reactor" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", - "parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)", - "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-signal" -version = "0.2.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-tcp" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-threadpool" -version = "0.1.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "crossbeam-deque 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)", - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-timer" -version = "0.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-udp" -version = "0.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "tokio-uds" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)", - "mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "toml" -version = "0.4.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "trust-dns-proto" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", - "socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)", - "url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "trust-dns-resolver" -version = "0.9.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)", - "futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)", - "ipconfig 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)", - "lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)", - "log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)", - "lru-cache 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", - "resolv-conf 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)", - "smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)", - "tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "trust-dns-proto 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "ucd-util" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "unicase" -version = "1.4.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "unicase" -version = "2.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "unicode-bidi" -version = "0.3.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "unicode-normalization" -version = "0.1.7" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "unicode-xid" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "unreachable" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "url" -version = "1.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)", - "idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", - "percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "utf8-ranges" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "uuid" -version = "0.6.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)", - "rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "vcpkg" -version = "0.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "version_check" -version = "0.1.4" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "void" -version = "1.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "widestring" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "winapi" -version = "0.2.8" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "winapi" -version = "0.3.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi-i686-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi-x86_64-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "winapi-build" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "winapi-i686-pc-windows-gnu" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "winapi-x86_64-pc-windows-gnu" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" - -[[package]] -name = "winreg" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "winutil" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[[package]] -name = "ws2_32-sys" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -dependencies = [ - "winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)", - "winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)", -] - -[metadata] -"checksum actix 0.7.3 (registry+https://github.com/rust-lang/crates.io-index)" = "0470f5163ac229ed0329f6abbd6f775591b06ec5ecbf2f2589c16ab8bd6f76b8" -"checksum actix-web 0.7.4 (registry+https://github.com/rust-lang/crates.io-index)" = "d0eddc44f25a168d776edf60477bd3d46566f89286e66288da8e196072e5a870" -"checksum actix_derive 0.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "5b9d1525ef45e5e021f0b93dace157dcab5d792acb4cc78f3213787d65e2bb92" -"checksum aho-corasick 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)" = "68f56c7353e5a9547cbd76ed90f7bb5ffc3ba09d4ea9bd1d8c06c8b1142eeb5a" -"checksum arrayvec 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)" = "a1e964f9e24d588183fcb43503abda40d288c8657dfc27311516ce2f05675aef" -"checksum askama 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c796fa924bb44233b38ae23020bfb229244e61910648d5f7bdc3659fcdf0572e" -"checksum askama_derive 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "57968dab3a9512b88af88012896e5e1d353e314afd5d190a98d0d4762c8f0a11" -"checksum askama_shared 0.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "a180253955cc7bc732fdd18fad8aa9f04a4891fc7aa9c6ce525c128a1de3a146" -"checksum backtrace 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)" = "346d7644f0b5f9bc73082d3b2236b69a05fd35cce0cfa3724e184e6a5c9e2a2f" -"checksum backtrace 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)" = "89a47830402e9981c5c41223151efcced65a0510c13097c769cede7efb34782a" -"checksum backtrace-sys 0.1.24 (registry+https://github.com/rust-lang/crates.io-index)" = "c66d56ac8dabd07f6aacdaf633f4b8262f5b3601a810a0dcddffd5c22c69daa0" -"checksum base64 0.9.2 (registry+https://github.com/rust-lang/crates.io-index)" = "85415d2594767338a74a30c1d370b2f3262ec1b4ed2d7bba5b3faf4de40467d9" -"checksum bitflags 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)" = "228047a76f468627ca71776ecdebd732a3423081fcf5125585bcd7c49886ce12" -"checksum byteorder 1.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "90492c5858dd7d2e78691cfb89f90d273a2800fc11d98f60786e5d87e2f83781" -"checksum bytes 0.4.9 (registry+https://github.com/rust-lang/crates.io-index)" = "e178b8e0e239e844b083d5a0d4a156b2654e67f9f80144d48398fcd736a24fb8" -"checksum cc 1.0.22 (registry+https://github.com/rust-lang/crates.io-index)" = "4a6007c146fdd28d4512a794b07ffe9d8e89e6bf86e2e0c4ddff2e1fb54a0007" -"checksum cfg-if 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "0c4e7bb64a8ebb0d856483e1e682ea3422f883c5f5615a90d51a2c82fe87fdd3" -"checksum cloudabi 0.0.3 (registry+https://github.com/rust-lang/crates.io-index)" = "ddfc5b9aa5d4507acaf872de71051dfd0e309860e88966e1051e462a077aac4f" -"checksum cookie 0.11.0 (registry+https://github.com/rust-lang/crates.io-index)" = "1465f8134efa296b4c19db34d909637cb2bf0f7aaf21299e23e18fa29ac557cf" -"checksum crossbeam-channel 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "6c0a94250b0278d7fc5a894c3d276b11ea164edc8bf8feb10ca1ea517b44a649" -"checksum crossbeam-deque 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "3486aefc4c0487b9cb52372c97df0a48b8c249514af1ee99703bf70d2f2ceda1" -"checksum crossbeam-epoch 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "30fecfcac6abfef8771151f8be4abc9e4edc112c2bcb233314cafde2680536e9" -"checksum crossbeam-utils 0.5.0 (registry+https://github.com/rust-lang/crates.io-index)" = "677d453a17e8bd2b913fa38e8b9cf04bcdbb5be790aa294f2389661d72036015" -"checksum dbghelp-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "97590ba53bcb8ac28279161ca943a924d1fd4a8fb3fa63302591647c4fc5b850" -"checksum diesel 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)" = "e71e7a348ae6064e86c4cf0709f0e4c3ef6f30e8e7d3dc05737164af4ebd3511" -"checksum diesel_derives 1.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "03bcaf77491f53e400d5ee3bdd57142ea4e1c47fe9217b3361ff9a76ca0e3d37" -"checksum dtoa 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)" = "6d301140eb411af13d3115f9a562c85cc6b541ade9dfa314132244aaee7489dd" -"checksum encoding 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)" = "6b0d943856b990d12d3b55b359144ff341533e516d94098b1d3fc1ac666d36ec" -"checksum encoding-index-japanese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "04e8b2ff42e9a05335dbf8b5c6f7567e5591d0d916ccef4e0b1710d32a0d0c91" -"checksum encoding-index-korean 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "4dc33fb8e6bcba213fe2f14275f0963fd16f0a02c878e3095ecfdf5bee529d81" -"checksum encoding-index-simpchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "d87a7194909b9118fc707194baa434a4e3b0fb6a5a757c73c3adb07aa25031f7" -"checksum encoding-index-singlebyte 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "3351d5acffb224af9ca265f435b859c7c01537c0849754d3db3fdf2bfe2ae84a" -"checksum encoding-index-tradchinese 1.20141219.5 (registry+https://github.com/rust-lang/crates.io-index)" = "fd0e20d5688ce3cab59eb3ef3a2083a5c77bf496cb798dc6fcdb75f323890c18" -"checksum encoding_index_tests 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "a246d82be1c9d791c5dfde9a2bd045fc3cbba3fa2b11ad558f27d01712f00569" -"checksum error-chain 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "faa976b4fd2e4c2b2f3f486874b19e61944d3de3de8b61c9fcf835d583871bcc" -"checksum error-chain 0.8.1 (registry+https://github.com/rust-lang/crates.io-index)" = "6930e04918388a9a2e41d518c25cf679ccafe26733fb4127dbf21993f2575d46" -"checksum failure 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7efb22686e4a466b1ec1a15c2898f91fa9cb340452496dca654032de20ff95b9" -"checksum failure_derive 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "946d0e98a50d9831f5d589038d2ca7f8f455b1c21028c0db0e84116a12696426" -"checksum fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)" = "2fad85553e09a6f881f739c29f0b00b0f01357c743266d478b68951ce23285f3" -"checksum fuchsia-zircon 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "2e9763c69ebaae630ba35f74888db465e49e259ba1bc0eda7d06f4a067615d82" -"checksum fuchsia-zircon-sys 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "3dcaa9ae7725d12cdb85b3ad99a434db70b468c09ded17e012d86b5c1010f7a7" -"checksum futures 0.1.23 (registry+https://github.com/rust-lang/crates.io-index)" = "884dbe32a6ae4cd7da5c6db9b78114449df9953b8d490c9d7e1b51720b922c62" -"checksum futures-cpupool 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "ab90cde24b3319636588d0c35fe03b1333857621051837ed769faefb4c2162e4" -"checksum h2 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "a27e7ed946e8335bdf9a191bc1b9b14a03ba822d013d2f58437f4fabcbd7fc2c" -"checksum hostname 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "21ceb46a83a85e824ef93669c8b390009623863b5c195d1ba747292c0c72f94e" -"checksum htmlescape 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)" = "e9025058dae765dee5070ec375f591e2ba14638c63feff74f13805a72e523163" -"checksum http 0.1.10 (registry+https://github.com/rust-lang/crates.io-index)" = "dca621d0fa606a5ff2850b6e337b57ad6137ee4d67e940449643ff45af6874c6" -"checksum httparse 1.3.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7b6288d7db100340ca12873fd4d08ad1b8f206a9457798dfb17c018a33fee540" -"checksum idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "38f09e0f0b1fb55fdee1f17470ad800da77af5186a1a76c026b679358b7e844e" -"checksum indexmap 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "08173ba1e906efb6538785a8844dd496f5d34f0a2d88038e95195172fc667220" -"checksum iovec 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "dbe6e417e7d0975db6512b90796e8ce223145ac4e33c377e4a42882a0e88bb08" -"checksum ipconfig 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "fccb81dd962b29a25de46c4f46e497b75117aa816468b6fff7a63a598a192394" -"checksum itoa 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "5adb58558dcd1d786b5f0bd15f3226ee23486e24b7b58304b60f64dc68e62606" -"checksum kernel32-sys 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7507624b29483431c0ba2d82aece8ca6cdba9382bff4ddd0f7490560c056098d" -"checksum language-tags 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a91d884b6667cd606bb5a69aa0c99ba811a115fc68915e7056ec08a46e93199a" -"checksum lazy_static 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ca488b89a5657b0a2ecd45b95609b3e848cf1755da332a0da46e2b2b1cb371a7" -"checksum lazycell 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)" = "a6f08839bc70ef4a3fe1d566d5350f519c5912ea86be0df1740a7d247c7fc0ef" -"checksum lazycell 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "d33a48d0365c96081958cc663eef834975cb1e8d8bea3378513fc72bdbf11e50" -"checksum libc 0.2.43 (registry+https://github.com/rust-lang/crates.io-index)" = "76e3a3ef172f1a0b9a9ff0dd1491ae5e6c948b94479a3021819ba7d860c8645d" -"checksum linked-hash-map 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7860ec297f7008ff7a1e3382d7f7e1dcd69efc94751a2284bafc3d013c2aa939" -"checksum lock_api 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "949826a5ccf18c1b3a7c3d57692778d21768b79e46eb9dd07bfc4c2160036c54" -"checksum log 0.4.4 (registry+https://github.com/rust-lang/crates.io-index)" = "cba860f648db8e6f269df990180c2217f333472b4a6e901e97446858487971e2" -"checksum lru-cache 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "4d06ff7ff06f729ce5f4e227876cb88d10bc59cd4ae1e09fbb2bde15c850dc21" -"checksum matches 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "7ffc5c5338469d4d3ea17d269fa8ea3512ad247247c30bd2df69e68309ed0a08" -"checksum memchr 2.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "a3b4142ab8738a78c51896f704f83c11df047ff1bda9a92a661aa6361552d93d" -"checksum memoffset 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "0f9dc261e2b62d7a622bf416ea3c5245cdd5d9a7fcc428c0d06804dfce1775b3" -"checksum mime 0.3.9 (registry+https://github.com/rust-lang/crates.io-index)" = "4b082692d3f6cf41b453af73839ce3dfc212c4411cbb2441dff80a716e38bd79" -"checksum mime_guess 2.0.0-alpha.6 (registry+https://github.com/rust-lang/crates.io-index)" = "30de2e4613efcba1ec63d8133f344076952090c122992a903359be5a4f99c3ed" -"checksum mio 0.6.15 (registry+https://github.com/rust-lang/crates.io-index)" = "4fcfcb32d63961fb6f367bfd5d21e4600b92cd310f71f9dca25acae196eb1560" -"checksum mio-uds 0.6.6 (registry+https://github.com/rust-lang/crates.io-index)" = "84c7b5caa3a118a6e34dbac36504503b1e8dc5835e833306b9d6af0e05929f79" -"checksum miow 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "8c1f2f3b1cf331de6896aabf6e9d55dca90356cc9960cca7eaaf408a355ae919" -"checksum net2 0.2.33 (registry+https://github.com/rust-lang/crates.io-index)" = "42550d9fb7b6684a6d404d9fa7250c2eb2646df731d1c06afc06dcee9e1bcf88" -"checksum nodrop 0.1.12 (registry+https://github.com/rust-lang/crates.io-index)" = "9a2228dca57108069a5262f2ed8bd2e82496d2e074a06d1ccc7ce1687b6ae0a2" -"checksum nom 4.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "898696750eb5c3ce5eb5afbfbe46e7f7c4e1936e19d3e97be4b7937da7b6d114" -"checksum num_cpus 1.8.0 (registry+https://github.com/rust-lang/crates.io-index)" = "c51a3322e4bca9d212ad9a158a02abc6934d005490c054a2778df73a70aa0a30" -"checksum owning_ref 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "cdf84f41639e037b484f93433aa3897863b561ed65c6e59c7073d7c561710f37" -"checksum parking_lot 0.6.3 (registry+https://github.com/rust-lang/crates.io-index)" = "69376b761943787ebd5cc85a5bc95958651a22609c5c1c2b65de21786baec72b" -"checksum parking_lot_core 0.2.14 (registry+https://github.com/rust-lang/crates.io-index)" = "4db1a8ccf734a7bce794cc19b3df06ed87ab2f3907036b693c68f56b4d4537fa" -"checksum percent-encoding 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "31010dd2e1ac33d5b46a5b413495239882813e0369f8ed8a5e266f173602f831" -"checksum phf 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "cec29da322b242f4c3098852c77a0ca261c9c01b806cae85a5572a1eb94db9a6" -"checksum phf_codegen 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "7d187f00cd98d5afbcd8898f6cf181743a449162aeb329dcd2f3849009e605ad" -"checksum phf_generator 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "03dc191feb9b08b0dc1330d6549b795b9d81aec19efe6b4a45aec8d4caee0c4b" -"checksum phf_shared 0.7.23 (registry+https://github.com/rust-lang/crates.io-index)" = "b539898d22d4273ded07f64a05737649dc69095d92cb87c7097ec68e3f150b93" -"checksum pq-sys 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "6ac25eee5a0582f45a67e837e350d784e7003bd29a5f460796772061ca49ffda" -"checksum proc-macro2 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)" = "1b06e2f335f48d24442b35a19df506a835fb3547bc3c06ef27340da9acf5cae7" -"checksum proc-macro2 0.4.13 (registry+https://github.com/rust-lang/crates.io-index)" = "ee5697238f0d893c7f0ecc59c0999f18d2af85e424de441178bcacc9f9e6cf67" -"checksum quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9274b940887ce9addde99c4eee6b5c44cc494b182b97e73dc8ffdcb3397fd3f0" -"checksum quote 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9949cfe66888ffe1d53e6ec9d9f3b70714083854be20fd5e271b232a017401e8" -"checksum quote 0.6.8 (registry+https://github.com/rust-lang/crates.io-index)" = "dd636425967c33af890042c483632d33fa7a18f19ad1d7ea72e8998c6ef8dea5" -"checksum rand 0.4.3 (registry+https://github.com/rust-lang/crates.io-index)" = "8356f47b32624fef5b3301c1be97e5944ecdd595409cc5da11d05f211db6cfbd" -"checksum rand 0.5.5 (registry+https://github.com/rust-lang/crates.io-index)" = "e464cd887e869cddcae8792a4ee31d23c7edd516700695608f5b98c67ee0131c" -"checksum rand_core 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "edecf0f94da5551fc9b492093e30b041a891657db7940ee221f9d2f66e82eef2" -"checksum redox_syscall 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)" = "c214e91d3ecf43e9a4e41e578973adeb14b474f2bee858742d127af75a0112b1" -"checksum regex 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)" = "67d0301b0c6804eca7e3c275119d0b01ff3b7ab9258a65709e608a66312a1025" -"checksum regex-syntax 0.6.2 (registry+https://github.com/rust-lang/crates.io-index)" = "747ba3b235651f6e2f67dfa8bcdcd073ddb7c243cb21c442fc12395dfcac212d" -"checksum resolv-conf 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "c62bd95a41841efdf7fca2ae9951e64a8d8eae7e5da196d8ce489a2241491a92" -"checksum rustc-demangle 0.1.9 (registry+https://github.com/rust-lang/crates.io-index)" = "bcfe5b13211b4d78e5c2cadfebd7769197d95c639c35a50057eb4c05de811395" -"checksum ryu 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "7153dd96dade874ab973e098cb62fcdbb89a03682e46b144fd09550998d4a4a7" -"checksum safemem 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "e27a8b19b835f7aea908818e871f5cc3a5a186550c30773be987e155e8163d8f" -"checksum scopeguard 0.3.3 (registry+https://github.com/rust-lang/crates.io-index)" = "94258f53601af11e6a49f722422f6e3425c52b06245a5cf9bc09908b174f5e27" -"checksum serde 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)" = "22d340507cea0b7e6632900a176101fea959c7065d93ba555072da90aaaafc87" -"checksum serde_derive 1.0.75 (registry+https://github.com/rust-lang/crates.io-index)" = "234fc8b737737b148ccd625175fc6390f5e4dacfdaa543cb93a3430d984a9119" -"checksum serde_json 1.0.26 (registry+https://github.com/rust-lang/crates.io-index)" = "44dd2cfde475037451fa99b7e5df77aa3cfd1536575fa8e7a538ab36dcde49ae" -"checksum serde_urlencoded 0.5.3 (registry+https://github.com/rust-lang/crates.io-index)" = "aaed41d9fb1e2f587201b863356590c90c1157495d811430a0c0325fe8169650" -"checksum sha1 0.6.0 (registry+https://github.com/rust-lang/crates.io-index)" = "2579985fda508104f7587689507983eadd6a6e84dd35d6d115361f530916fa0d" -"checksum siphasher 0.2.3 (registry+https://github.com/rust-lang/crates.io-index)" = "0b8de496cf83d4ed58b6be86c3a275b8602f6ffe98d3024a869e124147a9a3ac" -"checksum slab 0.4.1 (registry+https://github.com/rust-lang/crates.io-index)" = "5f9776d6b986f77b35c6cf846c11ad986ff128fe0b2b63a3628e3755e8d3102d" -"checksum smallvec 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "153ffa32fd170e9944f7e0838edf824a754ec4c1fc64746fcc9fe1f8fa602e5d" -"checksum socket2 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)" = "962a516af4d3a7c272cb3a1d50a8cc4e5b41802e4ad54cfb7bee8ba61d37d703" -"checksum stable_deref_trait 1.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "dba1a27d3efae4351c8051072d619e3ade2820635c3958d826bfea39d59b54c8" -"checksum string 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "00caf261d6f90f588f8450b8e1230fa0d5be49ee6140fdfbcb55335aff350970" -"checksum syn 0.13.11 (registry+https://github.com/rust-lang/crates.io-index)" = "14f9bf6292f3a61d2c716723fdb789a41bbe104168e6f496dc6497e531ea1b9b" -"checksum syn 0.14.9 (registry+https://github.com/rust-lang/crates.io-index)" = "261ae9ecaa397c42b960649561949d69311f08eeaea86a65696e6e46517cf741" -"checksum synstructure 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "85bb9b7550d063ea184027c9b8c20ac167cd36d3e06b3a40bceb9d746dc1a7b7" -"checksum thread_local 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)" = "c6b53e329000edc2b34dbe8545fd20e55a333362d0a321909685a19bd28c3f1b" -"checksum time 0.1.40 (registry+https://github.com/rust-lang/crates.io-index)" = "d825be0eb33fda1a7e68012d51e9c7f451dc1a69391e7fdc197060bb8c56667b" -"checksum tokio 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "fbb6a6e9db2702097bfdfddcb09841211ad423b86c75b5ddaca1d62842ac492c" -"checksum tokio-codec 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "881e9645b81c2ce95fcb799ded2c29ffb9f25ef5bef909089a420e5961dd8ccb" -"checksum tokio-current-thread 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "8fdfb899688ac16f618076bd09215edbfda0fd5dfecb375b6942636cb31fa8a7" -"checksum tokio-executor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "84823b932d566bc3c6aa644df4ca36cb38593c50b7db06011fd4e12e31e4047e" -"checksum tokio-fs 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)" = "b5cbe4ca6e71cb0b62a66e4e6f53a8c06a6eefe46cc5f665ad6f274c9906f135" -"checksum tokio-io 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)" = "8d6cc2de7725863c86ac71b0b9068476fec50834f055a243558ef1655bbd34cb" -"checksum tokio-reactor 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "df6a7ea7d65e0fc1398de28959de8be96909986a7d2e01d4f86d3433dfb91aed" -"checksum tokio-signal 0.2.4 (registry+https://github.com/rust-lang/crates.io-index)" = "527342552ec4a6049f787ccc9e2d6e0eab77bfe6cb7ec7a05c0391e370f466c3" -"checksum tokio-tcp 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "5b4c329b47f071eb8a746040465fa751bd95e4716e98daef6a9b4e434c17d565" -"checksum tokio-threadpool 0.1.6 (registry+https://github.com/rust-lang/crates.io-index)" = "a5758cecb6e0633cea5d563ac07c975e04961690b946b04fd84e7d6445a8f6af" -"checksum tokio-timer 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "d03fa701f9578a01b7014f106b47f0a363b4727a7f3f75d666e312ab7acbbf1c" -"checksum tokio-udp 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)" = "da941144b816d0dcda4db3a1ba87596e4df5e860a72b70783fe435891f80601c" -"checksum tokio-uds 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "424c1ed15a0132251813ccea50640b224c809d6ceafb88154c1a8775873a0e89" -"checksum toml 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "a0263c6c02c4db6c8f7681f9fd35e90de799ebd4cfdeab77a38f4ff6b3d8c0d9" -"checksum trust-dns-proto 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "32d7c204ee231f802aa821f9dc2195aa0d0269ef7e9f8c844208565c9e3981e4" -"checksum trust-dns-resolver 0.9.1 (registry+https://github.com/rust-lang/crates.io-index)" = "4a821ad51a29816420b8cac4b026756b81c023630b97eaa4c8090637ee3508bd" -"checksum ucd-util 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "fd2be2d6639d0f8fe6cdda291ad456e23629558d466e2789d2c3e9892bda285d" -"checksum unicase 1.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7f4765f83163b74f957c797ad9253caf97f103fb064d3999aea9568d09fc8a33" -"checksum unicase 2.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "284b6d3db520d67fbe88fd778c21510d1b0ba4a551e5d0fbb023d33405f6de8a" -"checksum unicode-bidi 0.3.4 (registry+https://github.com/rust-lang/crates.io-index)" = "49f2bd0c6468a8230e1db229cff8029217cf623c767ea5d60bfbd42729ea54d5" -"checksum unicode-normalization 0.1.7 (registry+https://github.com/rust-lang/crates.io-index)" = "6a0180bc61fc5a987082bfa111f4cc95c4caff7f9799f3e46df09163a937aa25" -"checksum unicode-xid 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "fc72304796d0818e357ead4e000d19c9c174ab23dc11093ac919054d20a6a7fc" -"checksum unreachable 1.0.0 (registry+https://github.com/rust-lang/crates.io-index)" = "382810877fe448991dfc7f0dd6e3ae5d58088fd0ea5e35189655f84e6814fa56" -"checksum url 1.7.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2a321979c09843d272956e73700d12c4e7d3d92b2ee112b31548aef0d4efc5a6" -"checksum utf8-ranges 1.0.1 (registry+https://github.com/rust-lang/crates.io-index)" = "fd70f467df6810094968e2fce0ee1bd0e87157aceb026a8c083bcf5e25b9efe4" -"checksum uuid 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "e1436e58182935dcd9ce0add9ea0b558e8a87befe01c1a301e6020aeb0876363" -"checksum vcpkg 0.2.6 (registry+https://github.com/rust-lang/crates.io-index)" = "def296d3eb3b12371b2c7d0e83bfe1403e4db2d7a0bba324a12b21c4ee13143d" -"checksum version_check 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)" = "7716c242968ee87e5542f8021178248f267f295a5c4803beae8b8b7fd9bc6051" -"checksum void 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "6a02e4885ed3bc0f2de90ea6dd45ebcbb66dacffe03547fadbb0eeae2770887d" -"checksum widestring 0.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7157704c2e12e3d2189c507b7482c52820a16dfa4465ba91add92f266667cadb" -"checksum winapi 0.2.8 (registry+https://github.com/rust-lang/crates.io-index)" = "167dc9d6949a9b857f3451275e911c3f44255842c1f7a76f33c55103a909087a" -"checksum winapi 0.3.5 (registry+https://github.com/rust-lang/crates.io-index)" = "773ef9dcc5f24b7d850d0ff101e542ff24c3b090a9768e03ff889fdef41f00fd" -"checksum winapi-build 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "2d315eee3b34aca4797b2da6b13ed88266e6d612562a0c46390af8299fc699bc" -"checksum winapi-i686-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" -"checksum winapi-x86_64-pc-windows-gnu 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)" = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" -"checksum winreg 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)" = "a27a759395c1195c4cc5cda607ef6f8f6498f64e78f7900f5de0a127a424704a" -"checksum winutil 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "7daf138b6b14196e3830a588acf1e86966c694d3e8fb026fb105b8b5dca07e6e" -"checksum ws2_32-sys 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)" = "d59cefebd0c892fa2dd6de581e937301d8552cb44489cdff035c6187cb63fa5e" diff --git a/tests/benchmark/actix-raw/Cargo.toml b/tests/benchmark/actix-raw/Cargo.toml index 71f8ec13d..d821c3961 100644 --- a/tests/benchmark/actix-raw/Cargo.toml +++ b/tests/benchmark/actix-raw/Cargo.toml @@ -3,25 +3,10 @@ name = "actix" version = "0.7.1" [dependencies] -askama = "0.7" -serde = "1.0" -serde_json = "1.0" -serde_derive = "1.0" -rand = "0.5" -bytes = "0.4" futures = "0.1" -http = "0.1" -diesel = { version = "1.2", features = ["postgres"] } -url = { version="1.7", features=["query_encoding"] } - actix = "0.7" actix-web = { version="0.7", default-features = false } -phf = "0.7.22" - -[build-dependencies] -askama = "0.7" - [profile.release] lto = true opt-level = 3 diff --git a/tests/benchmark/actix-raw/src/main.rs b/tests/benchmark/actix-raw/src/main.rs index f6f22748e..f8284d015 100644 --- a/tests/benchmark/actix-raw/src/main.rs +++ b/tests/benchmark/actix-raw/src/main.rs @@ -1,14 +1,6 @@ extern crate actix; extern crate actix_web; -extern crate bytes; extern crate futures; -extern crate serde_json; -#[macro_use] -extern crate serde_derive; -extern crate askama; -extern crate rand; -extern crate url; -extern crate diesel; use actix::prelude::*; use actix_web::server::{ @@ -17,14 +9,9 @@ use actix_web::server::{ use actix_web::Error; use futures::{Async, Poll}; -mod utils; - -use utils::{Message, Writer as JsonWriter}; - const HTTPOK: &[u8] = b"HTTP/1.1 200 OK\r\n"; const HDR_SERVER: &[u8] = b"Server: Actix\r\n"; const HDR_CTPLAIN: &[u8] = b"Content-Type: text/plain"; -const HDR_CTJSON: &[u8] = b"Content-Type: application/json"; const BODY: &[u8] = b"Hello, World!"; struct App { @@ -38,7 +25,6 @@ impl HttpHandler for App { let path = req.path(); match path.len() { 10 if path == "/plaintext" => return Ok(Box::new(Plaintext)), - 5 if path == "/json" => return Ok(Box::new(Json)), _ => (), } } @@ -64,28 +50,6 @@ impl HttpHandlerTask for Plaintext { } } -struct Json; - -impl HttpHandlerTask for Json { - fn poll_io(&mut self, io: &mut Writer) -> Poll { - let message = Message { - message: "Hello, World!", - }; - - { - let mut bytes = io.buffer(); - bytes.reserve(196); - bytes.extend_from_slice(HTTPOK); - bytes.extend_from_slice(HDR_SERVER); - bytes.extend_from_slice(HDR_CTJSON); - server::write_content_length(27, &mut bytes); - } - io.set_date(); - serde_json::to_writer(JsonWriter(io.buffer()), &message).unwrap(); - Ok(Async::Ready(true)) - } -} - fn main() { let sys = System::new("techempower"); @@ -93,9 +57,10 @@ fn main() { HttpServer::new(move || { vec![App { }] }).backlog(8192) - .bind("0.0.0.0:8080") - .unwrap() - .start(); + .workers(1) // comment this to enable multithread mode + .bind("0.0.0.0:8080") + .unwrap() + .start(); println!("Started http server: 127.0.0.1:8080"); let _ = sys.run(); diff --git a/tests/benchmark/actix-raw/src/utils.rs b/tests/benchmark/actix-raw/src/utils.rs deleted file mode 100644 index 0870abad1..000000000 --- a/tests/benchmark/actix-raw/src/utils.rs +++ /dev/null @@ -1,70 +0,0 @@ -#![allow(dead_code)] -use bytes::BytesMut; -use std::{cmp, fmt, io}; - -use actix_web::http::Uri; -use url::form_urlencoded; - -pub const SIZE: usize = 31; - -#[derive(Serialize, Deserialize)] -pub struct Message { - pub message: &'static str, -} - -pub struct Writer<'a>(pub &'a mut BytesMut); - -impl<'a> io::Write for Writer<'a> { - fn write(&mut self, buf: &[u8]) -> io::Result { - self.0.extend_from_slice(buf); - Ok(buf.len()) - } - fn flush(&mut self) -> io::Result<()> { - Ok(()) - } -} - -pub struct StackWriter<'a>(pub &'a mut [u8], pub usize); - -impl<'a> io::Write for StackWriter<'a> { - #[inline] - fn write(&mut self, buf: &[u8]) -> io::Result { - let l = buf.len(); - let new = self.1 + l; - self.0[self.1..new].copy_from_slice(buf); - self.1 = new; - Ok(l) - } - #[inline] - fn flush(&mut self) -> io::Result<()> { - Ok(()) - } -} - -impl<'a> fmt::Write for StackWriter<'a> { - #[inline] - fn write_str(&mut self, s: &str) -> Result<(), fmt::Error> { - if !s.is_empty() { - let b = s.as_bytes(); - let l = b.len(); - let new = self.1 + l; - self.0[self.1..new].copy_from_slice(b); - self.1 = new; - } - Ok(()) - } -} - -pub fn get_query_param(uri: &Uri) -> u16 { - let mut q = None; - let q_str = if let Some(s) = uri.query() { s } else { "" }; - for (key, val) in form_urlencoded::parse(q_str.as_ref()) { - if key == "q" { - q = Some(val); - break; - } - } - - q.map(|q| cmp::min(500, cmp::max(1, q.parse::().ok().unwrap_or(1)))) - .unwrap_or(1) -} diff --git a/tests/benchmark/benchmark.sh b/tests/benchmark/benchmark.sh deleted file mode 100755 index d32348ba4..000000000 --- a/tests/benchmark/benchmark.sh +++ /dev/null @@ -1,13 +0,0 @@ -#!/bin/bash - -mkdir wrksrc -cd wrksrc -if [ ! -f 4.1.0.tar.gz ]; then - wget -nc https://github.com/wg/wrk/archive/4.1.0.tar.gz | tar xz --strip-components=1 - make > /dev/null - cp wrk .. -fi - -cd .. -nim c -d:release -r bot -echo "$(cat benchmark_result.txt)" diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index cbbeae899..99715bb0f 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -1,25 +1,33 @@ mofuw - concurrency: 128, request/sec: 95682.82, transfer/sec: 12.045MiB - concurrency: 256, request/sec: 90446.30, transfer/sec: 11.386MiB - concurrency: 512, request/sec: 92075.01, transfer/sec: 11.591MiB - concurrency: 1024, failed + concurrency: 128, request/sec: 9835.15, transfer/sec: 1.219MiB + concurrency: 256, request/sec: 9824.21, transfer/sec: 1.218MiB + concurrency: 512, request/sec: 9775.89, transfer/sec: 1.212MiB + concurrency: 128, request/sec: 93255.38, transfer/sec: 11.562MiB, pipeline: 16 + concurrency: 256, request/sec: 88519.62, transfer/sec: 10.974MiB, pipeline: 16 + concurrency: 512, request/sec: 87012.02, transfer/sec: 10.788MiB, pipeline: 16 asyncnet - concurrency: 128, request/sec: 44196.54, transfer/sec: 2.276MiB - concurrency: 256, request/sec: 29674.21, transfer/sec: 1.528MiB - concurrency: 512, request/sec: 24244.77, transfer/sec: 1.249MiB - concurrency: 1024, failed + concurrency: 128, request/sec: 9544.33, transfer/sec: 503.314KiB + concurrency: 256, request/sec: 9001.25, transfer/sec: 474.675KiB + concurrency: 512, request/sec: 8140.63, transfer/sec: 429.291KiB + concurrency: 128, request/sec: 42758.94, transfer/sec: 2.202MiB, pipeline: 16 + concurrency: 256, request/sec: 27705.72, transfer/sec: 1.427MiB, pipeline: 16 + concurrency: 512, request/sec: 22977.16, transfer/sec: 1.183MiB, pipeline: 16 asyncdispatch2 - concurrency: 128, request/sec: 46276.68, transfer/sec: 2.383MiB - concurrency: 256, request/sec: 72831.57, transfer/sec: 3.751MiB - concurrency: 512, request/sec: 51450.58, transfer/sec: 2.65MiB - concurrency: 1024, failed + concurrency: 128, request/sec: 9771.45, transfer/sec: 515.291KiB + concurrency: 256, request/sec: 9246.41, transfer/sec: 487.604KiB + concurrency: 512, request/sec: 8611.69, transfer/sec: 454.132KiB + concurrency: 128, request/sec: 46380.65, transfer/sec: 2.389MiB, pipeline: 16 + concurrency: 256, request/sec: 66283.44, transfer/sec: 3.413MiB, pipeline: 16 + concurrency: 512, request/sec: 51394.27, transfer/sec: 2.647MiB, pipeline: 16 fasthttp concurrency: 128, request/sec: 119381.15, transfer/sec: 17.419MiB concurrency: 256, request/sec: 125082.98, transfer/sec: 18.251MiB concurrency: 512, request/sec: 157172.24, transfer/sec: 22.933MiB concurrency: 1024, failed actix-raw - concurrency: 128, request/sec: 147586.97, transfer/sec: 10.275MiB - concurrency: 256, request/sec: 137716.71, transfer/sec: 9.588MiB - concurrency: 512, request/sec: 162553.94, transfer/sec: 11.317MiB - concurrency: 1024, failed + concurrency: 128, request/sec: 9314.79, transfer/sec: 1.155MiB + concurrency: 256, request/sec: 8779.18, transfer/sec: 1.088MiB + concurrency: 512, request/sec: 8259.81, transfer/sec: 1.024MiB + concurrency: 128, request/sec: 117186.98, transfer/sec: 14.529MiB, pipeline: 16 + concurrency: 256, request/sec: 121966.19, transfer/sec: 15.121MiB, pipeline: 16 + concurrency: 512, failed diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index d3c7f4e0f..82995a9bc 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,7 +1,7 @@ import osproc, json, streams, strutils, os const - participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw"] + participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw", "ulib"] proc execAndGetJson(command: string): JsonNode = const @@ -64,11 +64,11 @@ proc removeContainers() = removeContainer(ID.getStr()) const - levels = [128, 256, 512, 1024] + levels = [128, 256, 512] maxConcurrency = levels[^1] duration = 15 serverHost = "bench-bot" - url = "http://127.0.0.1:8080/" + url = "http://127.0.0.1:8080/plaintext" pipeline = 16 accept = "text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7" @@ -84,6 +84,7 @@ proc runTest(name: string): JsonNode = cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d 5 -c 8 --timeout 8 -t 8 $3" % [serverHost, accept, url] echo " Running Primer" + ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run primer: " & name) @@ -101,6 +102,25 @@ proc runTest(name: string): JsonNode = for c in levels: echo " Running Concurrency $1" % [$c] let t = max(c, maxThreads) + let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s jsonfmt.lua" % + [serverHost, accept, $duration, $c, $t, url] + let res = execAndGetJson(cmd) + if res.len == 1: + var json = res[0] + json.add("level", newJInt(c)) + json.add("success", newJBool(true)) + json.add("pipeline", newJInt(0)) + jar.add json + else: + var json = newJObject() + json.add("level", newJInt(c)) + json.add("success", newJBool(false)) + json.add("pipeline", newJInt(0)) + jar.add json + sleep(2000) + for c in levels: + echo " Running Concurrency $1 with pipeline $2" % [$c, $pipeline] + let t = max(c, maxThreads) let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % [serverHost, accept, $duration, $c, $t, url, $pipeline] let res = execAndGetJson(cmd) @@ -108,11 +128,13 @@ proc runTest(name: string): JsonNode = var json = res[0] json.add("level", newJInt(c)) json.add("success", newJBool(true)) + json.add("pipeline", newJInt(pipeline)) jar.add json else: var json = newJObject() json.add("level", newJInt(c)) json.add("success", newJBool(false)) + json.add("pipeline", newJInt(pipeline)) jar.add json sleep(2000) killContainers() @@ -131,13 +153,18 @@ proc renderResult(json: JsonNode, s: Stream) = let duration = c["duration"].getInt() let requests = c["requests"].getInt() let bytes = c["bytes"].getInt() + let pipeline = c["pipeline"].getInt() let sec = duration.float / 1_000_000.0 let rps = formatFloat(requests.float / sec, ffDecimal, 2) let size = bytes.float / sec let tps = formatSize(size.int) - s.writeLine(" concurrency: $1, request/sec: $2, transfer/sec: $3" % [$concurrency, rps, tps]) + if pipeline == 0: + s.writeLine(" concurrency: $1, request/sec: $2, transfer/sec: $3" % [$concurrency, rps, tps]) + else: + s.writeLine(" concurrency: $1, request/sec: $2, transfer/sec: $3, pipeline: $4" % [$concurrency, rps, tps, $pipeline]) + else: let concurrency = c["level"].getInt() s.writeLine(" concurrency: $1, failed" % [$concurrency]) @@ -147,26 +174,94 @@ proc runAllTest() = var resList = newSeq[JsonNode]() for p in participants: - let res = runTest(p) - resList.add res + try: + let res = runTest(p) + resList.add res + except Exception as e: + echo e.msg var s = newFileStream("benchmark_result.txt", fmWrite) for res in resList: res.renderResult(s) s.close() + var ss = newStringStream() + for res in resList: + res.renderResult(ss) + echo ss.data + +proc runSingleTest(name: string) = + buildImage(name) + let res = runTest(name) + var s = newStringStream() + res.renderResult(s) + echo s.data + +proc removeImage(id: string) = + var ret = execSilent("docker rmi " & id) + if ret != 0: + raise newException(Exception, "cannot remove image: " & id) + +proc removeDanglingImages() = + let m = execAndGetJson("docker images -f \"dangling=true\" -q --format '{{json .}}'") + for x in m: + let ID = x["ID"] + removeImage(ID.getStr()) + +proc installWrk(forceInstall: bool = false): bool = + result = true + let parentDir = getAppDir() + if fileExists(parentDir & DirSep & "wrk") and not forceInstall: + return true + let curDir = parentDir & DirSep & "wrksrc" + discard existsOrCreateDir(curDir) + setCurrentDir(curDir) + if not fileExists(curDir & DirSep & "4.1.0.tar.gz") or forceInstall: + result = execCmd("wget https://github.com/wg/wrk/archive/4.1.0.tar.gz") == 0 + if result or forceInstall: + result = execCmd("tar xzf 4.1.0.tar.gz --strip-components=1 --skip-old-files") == 0 + if result or forceInstall: + result = execCmd("make > /dev/null") == 0 + if result or forceInstall: + copyFileWithPermissions(curDir & DirSep & "wrk", parentDir & DirSep & "wrk") + setCurrentDir(parentDir) + +proc printHelp() = + echo "clean clean all containers and unused images" + echo "installwrk force install wrk tool" + echo "help print this help" + echo "all run test for all frameworks" + echo "run `bot xxx` with xxx=`framework name` to run test on single framework" + echo "" + echo "available frameworks:" + for c in participants: + echo " ", c + +proc runCommand(cmd: string) = + case cmd + of "clean": + killContainers() + removeContainers() + removeDanglingImages() + of "installwrk": + discard installWrk(true) + of "all": + discard installWrk() + runAllTest() + of "help": + printHelp() + else: + if cmd notin participants: + echo cmd & " is not a registered participant" + return + discard installWrk() + runSingleTest(cmd) + proc main() = if paramCount() > 0: - let name = paramStr(1) - if name notin participants: - echo name & " is not a registered participant" - return - buildImage(name) - let res = runTest(name) - var s = newStringStream() - res.renderResult(s) - echo s.data + let cmd = paramStr(1) + runCommand(cmd) else: - runAllTest() + printHelp() main() diff --git a/tests/benchmark/jsonfmt.lua b/tests/benchmark/jsonfmt.lua new file mode 100644 index 000000000..f2861d21e --- /dev/null +++ b/tests/benchmark/jsonfmt.lua @@ -0,0 +1,7 @@ +done = function(summary, latency, requests) + io.write("{") + io.write('"duration":', summary.duration, ',') + io.write('"requests":', summary.requests, ',') + io.write('"bytes":', summary.bytes) + io.write("}") +end diff --git a/tests/benchmark/mofuw/server.nim b/tests/benchmark/mofuw/server.nim index afbfd739e..3a726ca1c 100644 --- a/tests/benchmark/mofuw/server.nim +++ b/tests/benchmark/mofuw/server.nim @@ -1,4 +1,4 @@ -import mofuw, packedjson +import mofuw, packedjson, threadpool proc h(ctx: MofuwCtx) {.async.} = case ctx.getPath @@ -9,7 +9,21 @@ proc h(ctx: MofuwCtx) {.async.} = else: mofuwResp(HTTP404, "text/plain", "NOT FOUND") +proc serveSingleThread*(ctx: ServeCtx) = + if ctx.handler.isNil: + raise newException(Exception, "Callback is nil. please set callback.") + + echo "SERVER RUNNING" + spawn ctx.runServer(ctx.isSSL) + + when not defined noSync: + sync() + newServeCtx( port = 8080, handler = h -).serve() +).serveSingleThread() + +# use this if you want multithread mode +#).serve() + diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md index 99a7eae39..b51e39d16 100644 --- a/tests/benchmark/readme.md +++ b/tests/benchmark/readme.md @@ -43,7 +43,7 @@ Each test is executed as follows: ## How to replicate this test locally? * First, you need to have docker installed, [see here how](https://www.digitalocean.com/community/tutorials/how-to-install-and-use-docker-on-ubuntu-18-04). -* Run the benchmark script `sh tests/benchmark/benchmark.sh` +* Compile and run the benchmark bot `nim c -r bot all` * Wait until it finished, and get the result in `tests/benchmark/benchmark_result.txt` ## How to add more participants? @@ -51,3 +51,6 @@ Each test is executed as follows: * Prepare a directory inside `tests/benchmark/` * inside that directory prepare a `plaintext.dockerfile` and all necessary source code. * add a entry in `bot.nim` participants constant list with the directory name. + +## How to switch to multi thread mode? +You can find a commented line in the source code of each framework to enable/disable multithread From c2a059da9764644d028db9685ba9039cdf10a120 Mon Sep 17 00:00:00 2001 From: jangko Date: Mon, 3 Sep 2018 11:47:57 +0700 Subject: [PATCH 06/12] asyncnet and asyncdispatch2 support pipeline mode --- tests/benchmark/asyncdispatch2/server.nim | 150 +++++++++++--- tests/benchmark/asyncnet/server.nim | 194 +++++++++++++++--- tests/benchmark/benchmark_result.txt | 24 +-- tests/benchmark/fasthttp/plaintext.dockerfile | 4 +- tests/benchmark/fasthttp/src/server/server.go | 2 + tests/benchmark/readme.md | 2 +- tests/benchmark/ulib/plaintext.dockerfile | 74 +++++++ tests/benchmark/ulib/src/plaintext.usp | 18 ++ 8 files changed, 404 insertions(+), 64 deletions(-) create mode 100644 tests/benchmark/ulib/plaintext.dockerfile create mode 100644 tests/benchmark/ulib/src/plaintext.usp diff --git a/tests/benchmark/asyncdispatch2/server.nim b/tests/benchmark/asyncdispatch2/server.nim index 7d9560621..2ee41c9db 100644 --- a/tests/benchmark/asyncdispatch2/server.nim +++ b/tests/benchmark/asyncdispatch2/server.nim @@ -1,22 +1,113 @@ -import strutils, posix - +import strutils, posix, deques, times, strformat import asyncdispatch2 -const DOUBLECRLF = @['\r'.byte, '\L'.byte, '\r'.byte, '\L'.byte] +proc getFormattedTime*(): string = + result = format(getTime().inZone(utc()), "ddd, dd MMM yyyy hh:mm:ss 'GMT'") + +type + IncomingCtx = ref object + buf: string + bufLen: int + resp: string + respLen: int + + ServerCtx = ref object + ctxQueue: Deque[IncomingCtx] + serverTime: string + +proc newIncomingCtx(readSize: int, writeSize: int): IncomingCtx = + result = IncomingCtx( + buf: newString(readSize), + bufLen: 0, + resp: newString(writeSize), + respLen: 0 + ) + +proc newServerCtx*(readSize, writeSize: int, cap: int): ServerCtx = + new(result) + result.ctxQueue = initDeque[IncomingCtx](cap) + var ctxArray = newSeq[IncomingCtx](cap) + for i in 0 ..< cap: + ctxArray[i] = newIncomingCtx(readSize, writeSize) + GC_ref(ctxArray[i]) + result.ctxQueue.addFirst(ctxArray[i]) + +proc getServerTime*(srv: ServerCtx): string = + result = srv.serverTime + +proc updateServerTime*(srv: ServerCtx) = + srv.serverTime = getFormattedTime() + +proc createCtx*(readSize, writeSize: int): IncomingCtx = + result = newIncomingCtx(readSize, writeSize) + GC_ref(result) + +proc getIncomingCtx*(srv: ServerCtx, readSize, writeSize: int): IncomingCtx = + if srv.ctxQueue.len > 0: + return srv.ctxQueue.popFirst() + else: + return createCtx(readSize, writeSize) + +proc freeCtx*(srv: ServerCtx, ctx: IncomingCtx) = + srv.ctxQueue.addLast(ctx) + +proc resetBuffer(ctx: IncomingCtx) = + ctx.respLen = 0 + ctx.bufLen = 0 + +proc sendMessage(ctx: IncomingCtx, body: string) = + while unlikely ctx.respLen + body.len > ctx.resp.len: + ctx.resp.setLen(ctx.resp.len + ctx.resp.len) + let ol = ctx.respLen + copyMem(addr ctx.resp[ol], unsafeAddr body[0], body.len) + ctx.respLen += body.len + +proc readMessage*(ctx: IncomingCtx, st: StreamTransport): Future[int] {.async.} = + let rcvLimit = + block: + if unlikely(ctx.buf.len - ctx.bufLen == 0): + ctx.buf.setLen(ctx.buf.len + ctx.buf.len) + ctx.buf.len - ctx.bufLen + + let rcv = await st.readOnce(addr ctx.buf[ctx.bufLen], rcvLimit) + ctx.bufLen += rcv + return rcv + +proc makeResp(serverTime: string): string = + result = fmt("HTTP/1.1 200 OK\r\L" & + "Date: {serverTime}\r\l" & + "Server: asyncdispatch2\r\L" & + "Content-Type: text/plain\r\L" & + "Content-Length: 13\r\L\r\L" & + "Hello, World!") + +proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, st: StreamTransport) {.async.} = + + while true: + let rcv = await ctx.readMessage(st) + if rcv == 0: + st.close() + srv.freeCtx(ctx) + return + + var pos = 0 + while (ctx.bufLen - pos) > 3: + if ctx.buf[pos] == '\r': + if ctx.buf[pos+1] == '\L' and + ctx.buf[pos+2] == '\r' and + ctx.buf[pos+3] == '\L': + ctx.sendMessage(makeResp(srv.serverTime)) + inc pos + + let wr = await st.write(ctx.resp[0].addr, ctx.respLen) + assert wr == ctx.respLen + ctx.resetBuffer() proc handleConnection(ss: StreamServer, st: StreamTransport) {.async.} = - var resp = "HTTP/1.1 200 OK\r\L" & - "Content-Length: 15\r\L\r\L" & - "Hello, World!\r\L" - var readBuff: array[256, byte] - while not st.atEof: - try: - let read = await st.readUntil(addr readBuff[0], len(readBuff), DOUBLECRLF) - except: - break - let w = await st.write(addr resp[0], resp.len) - assert w == resp.len - st.close() + var srv = getUserData[ServerCtx](ss) + var ctx = srv.getIncomingCtx(1024, 1024) + ctx.resetBuffer() + asyncCheck handleIncoming(srv, ctx, st) proc handleBreak(udata: pointer) = var cdata = cast[ptr CompletionData](udata) @@ -25,20 +116,31 @@ proc handleBreak(udata: pointer) = svr.stop() svr.close() +proc updateTime(arg: pointer = nil) {.gcsafe.} = + var svr = cast[ServerCtx](arg) + svr.updateServerTime() + proc serve(onAddress: string) = let ta = initTAddress(onAddress) - svr = createStreamServer(ta, handleConnection, {ReuseAddr}, backlog = 128) + ctx = newServerCtx(1024, 1024, 128) + svr = createStreamServer(ta, handleConnection, {ReuseAddr}, backlog = 128, udata = cast[pointer](ctx)) when not defined(windows): discard addSignal(SIGINT, handleBreak, udata = cast[pointer](svr)) - svr.start() - echo "Server started at ", ta - try: - waitFor svr.join() - except: - echo "Error happens: ", getCurrentExceptionMsg() - finally: - echo "bye!" + + ctx.updateServerTime() + addTimer(1000, updateTime, udata = cast[pointer](ctx)) + + var cantAccept = false + while true: + svr.start() + if not cantAccept: + echo "Server started at ", ta + + try: + waitFor svr.join() + except: + cantAccept = true when isMainModule: serve("0.0.0.0:8080") diff --git a/tests/benchmark/asyncnet/server.nim b/tests/benchmark/asyncnet/server.nim index d74d46a0d..02946b3b0 100644 --- a/tests/benchmark/asyncnet/server.nim +++ b/tests/benchmark/asyncnet/server.nim @@ -6,42 +6,186 @@ # Apache License, version 2.0, (LICENSE-APACHEv2) # MIT license (LICENSE-MIT) -import asyncnet, asyncdispatch +import asyncnet, asyncdispatch, nativesockets, net, strformat, times +import deques + +when defined(windows): + from winlean import TCP_NODELAY +else: + from posix import TCP_NODELAY type KeyboardInterruptError = object of Exception proc handleCtrlC() {.noconv.} = raise newException(KeyboardInterruptError, "Keyboard Interrupt") -setControlCHook(handleCtrlC) - -proc handleConnection(client: AsyncSocket) {.async.} = - var resp = - "HTTP/1.1 200 OK\r\L" & - "Content-Length: 15\r\L\r\L" & - "Hello, World!\r\L" - var line = newFutureVar[string]("asyncnet.recvLine") - line.mget() = newString(256) + +proc getFormattedTime*(): string = + result = format(getTime().inZone(utc()), "ddd, dd MMM yyyy hh:mm:ss 'GMT'") + +type + IncomingCtx = ref object + buf: string + bufLen: int + resp: string + respLen: int + + ServerCtx = ref object + ctxQueue: Deque[IncomingCtx] + serverTime: string + +proc newIncomingCtx(readSize: int, writeSize: int): IncomingCtx = + result = IncomingCtx( + buf: newString(readSize), + bufLen: 0, + resp: newString(writeSize), + respLen: 0 + ) + +proc newServerCtx*(readSize, writeSize: int, cap: int): ServerCtx = + new(result) + result.ctxQueue = initDeque[IncomingCtx](cap) + var ctxArray = newSeq[IncomingCtx](cap) + for i in 0 ..< cap: + ctxArray[i] = newIncomingCtx(readSize, writeSize) + GC_ref(ctxArray[i]) + result.ctxQueue.addFirst(ctxArray[i]) + +proc getServerTime*(srv: ServerCtx): string = + result = srv.serverTime + +proc updateServerTime*(srv: ServerCtx) = + srv.serverTime = getFormattedTime() + +proc createCtx*(readSize, writeSize: int): IncomingCtx = + result = newIncomingCtx(readSize, writeSize) + GC_ref(result) + +proc getIncomingCtx*(srv: ServerCtx, readSize, writeSize: int): IncomingCtx = + if srv.ctxQueue.len > 0: + return srv.ctxQueue.popFirst() + else: + return createCtx(readSize, writeSize) + +proc freeCtx*(srv: ServerCtx, ctx: IncomingCtx) = + srv.ctxQueue.addLast(ctx) + +proc resetBuffer(ctx: IncomingCtx) = + ctx.respLen = 0 + ctx.bufLen = 0 + +proc sendMessage(ctx: IncomingCtx, body: string) = + while unlikely ctx.respLen + body.len > ctx.resp.len: + ctx.resp.setLen(ctx.resp.len + ctx.resp.len) + let ol = ctx.respLen + copyMem(addr ctx.resp[ol], unsafeAddr body[0], body.len) + ctx.respLen += body.len + +const timeOut = 30 * 1000 + +proc readMessage*(ctx: IncomingCtx, client: AsyncSocket): Future[int] {.async.} = +#proc readMessage*(ctx: IncomingCtx, client: AsyncFD): Future[int] {.async.} = + let rcvLimit = + block: + if unlikely(ctx.buf.len - ctx.bufLen == 0): + ctx.buf.setLen(ctx.buf.len + ctx.buf.len) + ctx.buf.len - ctx.bufLen + + #let fut = client.recvInto(addr ctx.buf[ctx.bufLen], rcvLimit) + #let isSuccess = await withTimeout(fut, timeOut) + #let rcv = if isSuccess: fut.read else: 0 + let rcv = await client.recvInto(addr ctx.buf[ctx.bufLen], rcvLimit) + ctx.bufLen += rcv + return rcv + +proc makeResp(serverTime: string): string = + result = fmt("HTTP/1.1 200 OK\r\L" & + "Date: {serverTime}\r\l" & + "Server: asyncnet\r\L" & + "Content-Type: text/plain\r\L" & + "Content-Length: 13\r\L\r\L" & + "Hello, World!") + +#proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, client: AsyncFD) {.async.} = +proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, client: AsyncSocket) {.async.} = while true: - line.mget().setLen(0) - await client.recvLineInto(line) - if line.mget().len == 0: - break - if line.mget().len == 2: # \r\L - await client.send(resp) - client.close() + let rcv = await ctx.readMessage(client) + if rcv == 0: + #closeSocket(client) + client.close() + srv.freeCtx(ctx) + return -proc serve() {.async.} = - var server = newAsyncSocket() + var pos = 0 + while (ctx.bufLen - pos) > 3: + if ctx.buf[pos] == '\r': + if ctx.buf[pos+1] == '\L' and + ctx.buf[pos+2] == '\r' and + ctx.buf[pos+3] == '\L': + ctx.sendMessage(makeResp(srv.serverTime)) + inc pos + + let fut = client.send(ctx.resp[0].addr, ctx.respLen) + yield fut + if fut.failed: + #closeSocket(client) + client.close() + srv.freeCtx(ctx) + return + ctx.resetBuffer() + +#proc handleConnection(srv: ServerCtx, client: AsyncFD) {.async.} = +proc handleConnection(srv: ServerCtx, client: AsyncSocket) {.async.} = + var ctx = srv.getIncomingCtx(1024, 1024) + ctx.resetBuffer() + asyncCheck handleIncoming(srv, ctx, client) + +#proc newServerSocket*(port: int): SocketHandle = +# let server = newSocket() +# server.setSockOpt(OptReuseAddr, true) +# server.setSockOpt(OptReusePort, true) +# server.getFd().setSockOptInt(cint(IPPROTO_TCP), TCP_NODELAY, 1) +# server.getFd.setBlocking(false) +# server.bindAddr(Port(port)) +# server.listen() +# return server.getFd() + +proc newServerSocket(port: int): AsyncSocket = + var server = newAsyncSocket(buffered=false) server.setSockOpt(OptReuseAddr, true) - server.bindAddr(Port(8080)) + server.setSockOpt(OptReusePort, true) + server.getFd().setSockOptInt(cint(IPPROTO_TCP), TCP_NODELAY, 1) + server.getFd.setBlocking(false) + server.bindAddr(Port(port)) server.listen() - try: - while true: + result = server + +proc serve() {.async.} = + let + #server = newServerSocket(8080).AsyncFD + server = newServerSocket(8080) + ctx = newServerCtx(1024, 1024, 128) + + #register(server) + proc updateTime(fd: AsyncFD): bool = + ctx.updateServerTime() + + addTimer(1000, false, updateTime) + + var cantAccept = false + while true: + if unlikely cantaccept: + await sleepAsync(1) + cantAccept = true + + try: + #let data = await acceptAddr(server) + #asyncCheck handleConnection(ctx, data.client) let client = await server.accept() - asyncCheck handleConnection(client) - finally: - server.close() + asyncCheck handleConnection(ctx, client) + except: + cantAccept = true when isMainModule: + setControlCHook(handleCtrlC) waitFor serve() diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index 99715bb0f..d02d89376 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -6,19 +6,19 @@ mofuw concurrency: 256, request/sec: 88519.62, transfer/sec: 10.974MiB, pipeline: 16 concurrency: 512, request/sec: 87012.02, transfer/sec: 10.788MiB, pipeline: 16 asyncnet - concurrency: 128, request/sec: 9544.33, transfer/sec: 503.314KiB - concurrency: 256, request/sec: 9001.25, transfer/sec: 474.675KiB - concurrency: 512, request/sec: 8140.63, transfer/sec: 429.291KiB - concurrency: 128, request/sec: 42758.94, transfer/sec: 2.202MiB, pipeline: 16 - concurrency: 256, request/sec: 27705.72, transfer/sec: 1.427MiB, pipeline: 16 - concurrency: 512, request/sec: 22977.16, transfer/sec: 1.183MiB, pipeline: 16 + concurrency: 128, request/sec: 10342.86, transfer/sec: 1.312MiB + concurrency: 256, request/sec: 9355.33, transfer/sec: 1.187MiB + concurrency: 512, failed + concurrency: 128, request/sec: 80298.61, transfer/sec: 10.185MiB, pipeline: 16 + concurrency: 256, request/sec: 74327.86, transfer/sec: 9.428MiB, pipeline: 16 + concurrency: 512, request/sec: 65689.96, transfer/sec: 8.332MiB, pipeline: 16 asyncdispatch2 - concurrency: 128, request/sec: 9771.45, transfer/sec: 515.291KiB - concurrency: 256, request/sec: 9246.41, transfer/sec: 487.604KiB - concurrency: 512, request/sec: 8611.69, transfer/sec: 454.132KiB - concurrency: 128, request/sec: 46380.65, transfer/sec: 2.389MiB, pipeline: 16 - concurrency: 256, request/sec: 66283.44, transfer/sec: 3.413MiB, pipeline: 16 - concurrency: 512, request/sec: 51394.27, transfer/sec: 2.647MiB, pipeline: 16 + concurrency: 128, request/sec: 165609.73, transfer/sec: 21.953MiB + concurrency: 256, request/sec: 135977.47, transfer/sec: 18.025MiB + concurrency: 512, request/sec: 124412.73, transfer/sec: 16.492MiB + concurrency: 128, request/sec: 90894.05, transfer/sec: 12.049MiB, pipeline: 16 + concurrency: 256, request/sec: 44309.85, transfer/sec: 5.874MiB, pipeline: 16 + concurrency: 512, request/sec: 297.15, transfer/sec: 40.336KiB, pipeline: 16 fasthttp concurrency: 128, request/sec: 119381.15, transfer/sec: 17.419MiB concurrency: 256, request/sec: 125082.98, transfer/sec: 18.251MiB diff --git a/tests/benchmark/fasthttp/plaintext.dockerfile b/tests/benchmark/fasthttp/plaintext.dockerfile index 042a44f8b..b87bd2119 100644 --- a/tests/benchmark/fasthttp/plaintext.dockerfile +++ b/tests/benchmark/fasthttp/plaintext.dockerfile @@ -10,7 +10,7 @@ ENV PATH ${GOPATH}/bin:${PATH} RUN rm -rf ./pkg/* RUN go get -d -u github.com/valyala/fasthttp/... -RUN rm -f ./server-mysql +RUN rm -f ./server RUN go build -gcflags='-l=4' server -CMD ./server +CMD GOMAXPROCS=1 ./server diff --git a/tests/benchmark/fasthttp/src/server/server.go b/tests/benchmark/fasthttp/src/server/server.go index c5c1a7a7b..3c8fd580a 100644 --- a/tests/benchmark/fasthttp/src/server/server.go +++ b/tests/benchmark/fasthttp/src/server/server.go @@ -5,9 +5,11 @@ import ( "log" "github.com/valyala/fasthttp" "common" + "runtime" ) func main() { + runtime.GOMAXPROCS(1) flag.Parse() var err error diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md index b51e39d16..c3840a826 100644 --- a/tests/benchmark/readme.md +++ b/tests/benchmark/readme.md @@ -45,7 +45,7 @@ Each test is executed as follows: * First, you need to have docker installed, [see here how](https://www.digitalocean.com/community/tutorials/how-to-install-and-use-docker-on-ubuntu-18-04). * Compile and run the benchmark bot `nim c -r bot all` * Wait until it finished, and get the result in `tests/benchmark/benchmark_result.txt` - +* You can ## How to add more participants? * Prepare a directory inside `tests/benchmark/` diff --git a/tests/benchmark/ulib/plaintext.dockerfile b/tests/benchmark/ulib/plaintext.dockerfile new file mode 100644 index 000000000..d90ecb815 --- /dev/null +++ b/tests/benchmark/ulib/plaintext.dockerfile @@ -0,0 +1,74 @@ +FROM ubuntu:18.04 + +COPY ./ ./ + +RUN apt update -yqq && \ + apt install -yqq software-properties-common build-essential locales wget \ + libreadline6-dev \ + libgdbm-dev ncurses-dev \ + libffi-dev libtool bison libevent-dev \ + liborc-0.4-0 \ + libmcrypt-dev libicu-dev \ + libcap2-bin && \ + add-apt-repository ppa:ubuntu-toolchain-r/test -y && \ + apt update -yqq && \ + apt install -yqq gcc-8 g++-8 + +RUN locale-gen en_US.UTF-8 + +ENV LANG en_US.UTF-8 +ENV LANGUAGE en_US:en +ENV LC_ALL en_US.UTF-8 +ENV DEBIAN_FRONTEND noninteractive + +ENV CC=gcc-8 +ENV CXX=g++-8 +ENV AR=gcc-ar-8 +ENV RANLIB=gcc-ranlib-8 +ENV IROOT=/install +ENV ULIB_ROOT=$IROOT/ULib +ENV ULIB_VERSION=1.4.2 +ENV ULIB_DOCUMENT_ROOT=$ULIB_ROOT/ULIB_DOCUMENT_ROOT + +WORKDIR $IROOT + +RUN mkdir -p $ULIB_DOCUMENT_ROOT +RUN wget -q -O ULib-${ULIB_VERSION}.tar.gz https://github.com/stefanocasazza/ULib/archive/v${ULIB_VERSION}.tar.gz +RUN tar xf ULib-${ULIB_VERSION}.tar.gz + +WORKDIR $IROOT/ULib-$ULIB_VERSION + +# AVOID "configure: error: newly created file is older than distributed files! Check your system clock" +#RUN cp /src/* src/ulib/net/server/plugin/usp +RUN find . -exec touch {} \; + +RUN echo "userver {" >> $ULIB_ROOT/benchmark.cfg +RUN echo "PORT 8080" >> $ULIB_ROOT/benchmark.cfg +RUN echo "PREFORK_CHILD $(nproc)" >> $ULIB_ROOT/benchmark.cfg +RUN echo "TCP_LINGER_SET 0" >> $ULIB_ROOT/benchmark.cfg +RUN echo "LISTEN_BACKLOG 16384" >> $ULIB_ROOT/benchmark.cfg +RUN echo "DOCUMENT_ROOT $ULIB_DOCUMENT_ROOT " >> $ULIB_ROOT/benchmark.cfg +RUN echo "}" >> $ULIB_ROOT/benchmark.cfg + +RUN USP_FLAGS="-DAS_cpoll_cppsp_DO" \ + ./configure --prefix=$ULIB_ROOT \ + --disable-static --disable-examples \ + --without-ssl --without-pcre --without-expat \ + --without-libz --without-libuuid --without-magic --without-libares \ + --enable-static-server-plugin=http + +RUN make install && \ + cd examples/userver && make install && \ + cd ../../src/ulib/net/server/plugin/usp && \ + make plaintext.la && \ + cp .libs/plaintext.so $ULIB_DOCUMENT_ROOT + +ENV PATH=${ULIB_ROOT}/bin:${PATH} + +ADD ./ /ulib +WORKDIR /ulib + +ENV UMEMPOOL="58,0,0,41,16401,-14,-15,11,25" + +CMD setcap cap_sys_nice,cap_sys_resource,cap_net_bind_service,cap_net_raw+eip $IROOT/ULib/bin/userver_tcp && \ + $IROOT/ULib/bin/userver_tcp -c $IROOT/ULib/benchmark.cfg diff --git a/tests/benchmark/ulib/src/plaintext.usp b/tests/benchmark/ulib/src/plaintext.usp new file mode 100644 index 000000000..ba06d3f32 --- /dev/null +++ b/tests/benchmark/ulib/src/plaintext.usp @@ -0,0 +1,18 @@ + + From 9edfb9b7090737a0293cb62d990577174ddb9e4b Mon Sep 17 00:00:00 2001 From: jangko Date: Thu, 6 Sep 2018 19:25:55 +0700 Subject: [PATCH 07/12] fix ad2 benchmark --- .travis.yml | 1 + asyncdispatch2.nimble | 4 +- tests/benchmark/asyncdispatch2/server.nim | 56 ++++++++++------------ tests/benchmark/benchmark_result.txt | 58 ++++++++++++----------- tests/benchmark/bot.nim | 4 +- tests/benchmark/readme.md | 5 +- 6 files changed, 65 insertions(+), 63 deletions(-) diff --git a/.travis.yml b/.travis.yml index 9ea188747..7f7e6ec39 100644 --- a/.travis.yml +++ b/.travis.yml @@ -9,6 +9,7 @@ matrix: script: - docker run statusteam/nim-base nim --version - docker run -v "$(pwd):/project" -w /project statusteam/nim-base sh -c "nimble install -dy && nimble test" + - benchmark/bot all - os: osx before_install: diff --git a/asyncdispatch2.nimble b/asyncdispatch2.nimble index 46d954403..012858d2d 100644 --- a/asyncdispatch2.nimble +++ b/asyncdispatch2.nimble @@ -15,7 +15,6 @@ task test, "Run all tests": exec "nim c -r --gc:markAndSweep tests/testsync" exec "nim c -r -d:release tests/testsync" - exec "nim c -r -d:useSysAssert -d:useGcAssert tests/testsoon" exec "nim c -r tests/testsoon" exec "nim c -r --gc:markAndSweep tests/testsoon" @@ -60,3 +59,6 @@ task test, "Run all tests": exec "nim c -r tests/testbugs" exec "nim c -r --gc:markAndSweep tests/testbugs" exec "nim c -r -d:release tests/testbugs" + +task test, "compile bench-bot": + exec "nim c -d:release benchmark/bot" diff --git a/tests/benchmark/asyncdispatch2/server.nim b/tests/benchmark/asyncdispatch2/server.nim index 2ee41c9db..039183341 100644 --- a/tests/benchmark/asyncdispatch2/server.nim +++ b/tests/benchmark/asyncdispatch2/server.nim @@ -82,26 +82,29 @@ proc makeResp(serverTime: string): string = "Hello, World!") proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, st: StreamTransport) {.async.} = - - while true: - let rcv = await ctx.readMessage(st) - if rcv == 0: - st.close() - srv.freeCtx(ctx) - return - - var pos = 0 - while (ctx.bufLen - pos) > 3: - if ctx.buf[pos] == '\r': - if ctx.buf[pos+1] == '\L' and - ctx.buf[pos+2] == '\r' and - ctx.buf[pos+3] == '\L': - ctx.sendMessage(makeResp(srv.serverTime)) - inc pos - - let wr = await st.write(ctx.resp[0].addr, ctx.respLen) - assert wr == ctx.respLen - ctx.resetBuffer() + try: + while true: + let rcv = await ctx.readMessage(st) + if rcv == 0: + st.close() + srv.freeCtx(ctx) + return + + var pos = 0 + while (ctx.bufLen - pos) > 3: + if ctx.buf[pos] == '\r': + if ctx.buf[pos+1] == '\L' and + ctx.buf[pos+2] == '\r' and + ctx.buf[pos+3] == '\L': + ctx.sendMessage(makeResp(srv.serverTime)) + inc pos + + let wr = await st.write(ctx.resp[0].addr, ctx.respLen) + assert wr == ctx.respLen + ctx.resetBuffer() + except: + st.close() + srv.freeCtx(ctx) proc handleConnection(ss: StreamServer, st: StreamTransport) {.async.} = var srv = getUserData[ServerCtx](ss) @@ -131,16 +134,9 @@ proc serve(onAddress: string) = ctx.updateServerTime() addTimer(1000, updateTime, udata = cast[pointer](ctx)) - var cantAccept = false - while true: - svr.start() - if not cantAccept: - echo "Server started at ", ta - - try: - waitFor svr.join() - except: - cantAccept = true + svr.start() + echo "Server started at ", ta + waitFor svr.join() when isMainModule: serve("0.0.0.0:8080") diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index d02d89376..fb1121ec5 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -1,33 +1,35 @@ mofuw - concurrency: 128, request/sec: 9835.15, transfer/sec: 1.219MiB - concurrency: 256, request/sec: 9824.21, transfer/sec: 1.218MiB - concurrency: 512, request/sec: 9775.89, transfer/sec: 1.212MiB - concurrency: 128, request/sec: 93255.38, transfer/sec: 11.562MiB, pipeline: 16 - concurrency: 256, request/sec: 88519.62, transfer/sec: 10.974MiB, pipeline: 16 - concurrency: 512, request/sec: 87012.02, transfer/sec: 10.788MiB, pipeline: 16 + concurrency: 128, request/sec: 27667.79, transfer/sec: 3.43MiB + concurrency: 256, request/sec: 23469.25, transfer/sec: 2.91MiB + concurrency: 480, request/sec: 19072.19, transfer/sec: 2.365MiB + concurrency: 128, request/sec: 161094.58, transfer/sec: 19.972MiB, pipeline: 16 + concurrency: 256, request/sec: 164338.21, transfer/sec: 20.374MiB, pipeline: 16 + concurrency: 480, request/sec: 159308.03, transfer/sec: 19.751MiB, pipeline: 16 asyncnet - concurrency: 128, request/sec: 10342.86, transfer/sec: 1.312MiB - concurrency: 256, request/sec: 9355.33, transfer/sec: 1.187MiB - concurrency: 512, failed - concurrency: 128, request/sec: 80298.61, transfer/sec: 10.185MiB, pipeline: 16 - concurrency: 256, request/sec: 74327.86, transfer/sec: 9.428MiB, pipeline: 16 - concurrency: 512, request/sec: 65689.96, transfer/sec: 8.332MiB, pipeline: 16 + concurrency: 128, request/sec: 33162.16, transfer/sec: 4.206MiB + concurrency: 256, request/sec: 28858.21, transfer/sec: 3.66MiB + concurrency: 480, request/sec: 24846.94, transfer/sec: 3.152MiB + concurrency: 128, request/sec: 176565.07, transfer/sec: 22.395MiB, pipeline: 16 + concurrency: 256, request/sec: 176510.82, transfer/sec: 22.388MiB, pipeline: 16 + concurrency: 480, request/sec: 166837.28, transfer/sec: 21.161MiB, pipeline: 16 asyncdispatch2 - concurrency: 128, request/sec: 165609.73, transfer/sec: 21.953MiB - concurrency: 256, request/sec: 135977.47, transfer/sec: 18.025MiB - concurrency: 512, request/sec: 124412.73, transfer/sec: 16.492MiB - concurrency: 128, request/sec: 90894.05, transfer/sec: 12.049MiB, pipeline: 16 - concurrency: 256, request/sec: 44309.85, transfer/sec: 5.874MiB, pipeline: 16 - concurrency: 512, request/sec: 297.15, transfer/sec: 40.336KiB, pipeline: 16 + concurrency: 128, request/sec: 328212.10, transfer/sec: 43.508MiB + concurrency: 256, request/sec: 254634.24, transfer/sec: 33.755MiB + concurrency: 480, request/sec: 220645.90, transfer/sec: 29.249MiB + concurrency: 128, request/sec: 358799.70, transfer/sec: 47.563MiB, pipeline: 16 + concurrency: 256, request/sec: 298887.75, transfer/sec: 39.622MiB, pipeline: 16 + concurrency: 480, request/sec: 135642.25, transfer/sec: 17.981MiB, pipeline: 16 fasthttp - concurrency: 128, request/sec: 119381.15, transfer/sec: 17.419MiB - concurrency: 256, request/sec: 125082.98, transfer/sec: 18.251MiB - concurrency: 512, request/sec: 157172.24, transfer/sec: 22.933MiB - concurrency: 1024, failed + concurrency: 128, request/sec: 30920.02, transfer/sec: 3.745MiB + concurrency: 256, request/sec: 27362.95, transfer/sec: 3.314MiB + concurrency: 480, request/sec: 26008.47, transfer/sec: 3.15MiB + concurrency: 128, request/sec: 194787.96, transfer/sec: 23.592MiB, pipeline: 16 + concurrency: 256, request/sec: 191044.91, transfer/sec: 23.139MiB, pipeline: 16 + concurrency: 480, request/sec: 196284.84, transfer/sec: 23.773MiB, pipeline: 16 actix-raw - concurrency: 128, request/sec: 9314.79, transfer/sec: 1.155MiB - concurrency: 256, request/sec: 8779.18, transfer/sec: 1.088MiB - concurrency: 512, request/sec: 8259.81, transfer/sec: 1.024MiB - concurrency: 128, request/sec: 117186.98, transfer/sec: 14.529MiB, pipeline: 16 - concurrency: 256, request/sec: 121966.19, transfer/sec: 15.121MiB, pipeline: 16 - concurrency: 512, failed + concurrency: 128, request/sec: 40068.61, transfer/sec: 4.968MiB + concurrency: 256, request/sec: 33321.05, transfer/sec: 4.131MiB + concurrency: 480, request/sec: 29414.25, transfer/sec: 3.647MiB + concurrency: 128, request/sec: 245921.39, transfer/sec: 30.489MiB, pipeline: 16 + concurrency: 256, request/sec: 237474.19, transfer/sec: 29.441MiB, pipeline: 16 + concurrency: 480, request/sec: 244928.92, transfer/sec: 30.366MiB, pipeline: 16 diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 82995a9bc..6ab4d915c 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,7 +1,7 @@ import osproc, json, streams, strutils, os const - participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw", "ulib"] + participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw"] proc execAndGetJson(command: string): JsonNode = const @@ -64,7 +64,7 @@ proc removeContainers() = removeContainer(ID.getStr()) const - levels = [128, 256, 512] + levels = [128, 256, 480] maxConcurrency = levels[^1] duration = 15 serverHost = "bench-bot" diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md index c3840a826..e61b6b5a1 100644 --- a/tests/benchmark/readme.md +++ b/tests/benchmark/readme.md @@ -38,14 +38,15 @@ Each test is executed as follows: * Run a 5-second __primer__ at 8 client-concurrency to verify that the server is in fact running. These results are not captured. * Run a 15-second __warmup__ at 256 client-concurrency to allow lazy-initialization to execute and just-in-time compilation to run. These results are not captured. * Run a 15-second __captured test__ for each of the concurrency levels (or iteration counts) exercised by the test type. - The high-concurrency plaintext test type is tested at 128, 256, 512, and 1024 client-side concurrency. + The high-concurrency plaintext test type is tested at 128, 256, and 512 client-side concurrency. * Stop the platform and framework. ## How to replicate this test locally? * First, you need to have docker installed, [see here how](https://www.digitalocean.com/community/tutorials/how-to-install-and-use-docker-on-ubuntu-18-04). * Compile and run the benchmark bot `nim c -r bot all` * Wait until it finished, and get the result in `tests/benchmark/benchmark_result.txt` -* You can +* You can see all bench-bot command line options with `./bot` or `./bot help` + ## How to add more participants? * Prepare a directory inside `tests/benchmark/` From 6c2b7f7d9779b2e13145f4e32470a5274c306225 Mon Sep 17 00:00:00 2001 From: andri lim Date: Thu, 6 Sep 2018 19:56:37 +0700 Subject: [PATCH 08/12] add benchmark test to ci --- .travis.yml | 4 +++- asyncdispatch2.nimble | 6 +++--- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/.travis.yml b/.travis.yml index d0b2bd717..5164b6a63 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,4 +1,4 @@ -anguage: c # or other C/C++ variants +language: c # or other C/C++ variants sudo: false @@ -37,3 +37,5 @@ install: script: - nimble install -y - nimble test + - nimble benchmark + diff --git a/asyncdispatch2.nimble b/asyncdispatch2.nimble index 2f350a483..4593ceb30 100644 --- a/asyncdispatch2.nimble +++ b/asyncdispatch2.nimble @@ -60,6 +60,6 @@ task test, "Run all tests": exec "nim c -r --gc:markAndSweep tests/testbugs" exec "nim c -r -d:release tests/testbugs" -task test, "compile bench-bot": - exec "nim c -d:release benchmark/bot" - exec "benchmark/bot all" +task benchmark, "compile bench-bot": + exec "nim c -d:release tests/benchmark/bot" + exec "tests/benchmark/bot all" From 3ad25298a4122da07a251049375f7e11814cca15 Mon Sep 17 00:00:00 2001 From: jangko Date: Tue, 11 Sep 2018 19:48:19 +0700 Subject: [PATCH 09/12] add libreactor and summary --- .travis.yml | 2 - tests/benchmark/benchmark_result.txt | 69 +++++---- tests/benchmark/bot.nim | 112 ++++++++++---- tests/benchmark/fasthttp/src/common/common.go | 56 ------- tests/benchmark/fasthttp/src/server/server.go | 22 ++- tests/benchmark/libreactor/Makefile | 11 ++ .../benchmark/libreactor/plaintext.dockerfile | 34 +++++ tests/benchmark/libreactor/src/main.c | 140 ++++++++++++++++++ tests/benchmark/libreactor/src/setup.c | 107 +++++++++++++ tests/benchmark/libreactor/src/setup.h | 6 + tests/benchmark/mofuw/README.md | 11 -- tests/benchmark/readme.md | 19 ++- tests/benchmark/ulib/plaintext.dockerfile | 74 --------- tests/benchmark/ulib/src/plaintext.usp | 18 --- 14 files changed, 451 insertions(+), 230 deletions(-) delete mode 100644 tests/benchmark/fasthttp/src/common/common.go create mode 100644 tests/benchmark/libreactor/Makefile create mode 100644 tests/benchmark/libreactor/plaintext.dockerfile create mode 100644 tests/benchmark/libreactor/src/main.c create mode 100644 tests/benchmark/libreactor/src/setup.c create mode 100644 tests/benchmark/libreactor/src/setup.h delete mode 100644 tests/benchmark/mofuw/README.md delete mode 100644 tests/benchmark/ulib/plaintext.dockerfile delete mode 100644 tests/benchmark/ulib/src/plaintext.usp diff --git a/.travis.yml b/.travis.yml index 5164b6a63..8e56f7ecd 100644 --- a/.travis.yml +++ b/.travis.yml @@ -37,5 +37,3 @@ install: script: - nimble install -y - nimble test - - nimble benchmark - diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index fb1121ec5..c16859f3a 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -1,35 +1,42 @@ +fasthttp + concurrency: 128, request/sec: 33192.01, transfer/sec: 4.02MiB + concurrency: 256, request/sec: 31513.33, transfer/sec: 3.817MiB + concurrency: 480, request/sec: 29085.05, transfer/sec: 3.523MiB + concurrency: 128, request/sec: 228440.91, transfer/sec: 27.668MiB, pipeline: 16 + concurrency: 128, request/sec: 223112.01, transfer/sec: 27.023MiB, pipeline: 16 + concurrency: 128, request/sec: 230501.58, transfer/sec: 27.918MiB, pipeline: 16 mofuw - concurrency: 128, request/sec: 27667.79, transfer/sec: 3.43MiB - concurrency: 256, request/sec: 23469.25, transfer/sec: 2.91MiB - concurrency: 480, request/sec: 19072.19, transfer/sec: 2.365MiB - concurrency: 128, request/sec: 161094.58, transfer/sec: 19.972MiB, pipeline: 16 - concurrency: 256, request/sec: 164338.21, transfer/sec: 20.374MiB, pipeline: 16 - concurrency: 480, request/sec: 159308.03, transfer/sec: 19.751MiB, pipeline: 16 + concurrency: 128, request/sec: 36500.58, transfer/sec: 4.525MiB + concurrency: 256, request/sec: 27654.28, transfer/sec: 3.429MiB + concurrency: 480, request/sec: 25377.77, transfer/sec: 3.146MiB + concurrency: 128, request/sec: 195180.88, transfer/sec: 24.198MiB, pipeline: 16 + concurrency: 128, request/sec: 189706.99, transfer/sec: 23.519MiB, pipeline: 16 + concurrency: 128, request/sec: 194490.21, transfer/sec: 24.112MiB, pipeline: 16 asyncnet - concurrency: 128, request/sec: 33162.16, transfer/sec: 4.206MiB - concurrency: 256, request/sec: 28858.21, transfer/sec: 3.66MiB - concurrency: 480, request/sec: 24846.94, transfer/sec: 3.152MiB - concurrency: 128, request/sec: 176565.07, transfer/sec: 22.395MiB, pipeline: 16 - concurrency: 256, request/sec: 176510.82, transfer/sec: 22.388MiB, pipeline: 16 - concurrency: 480, request/sec: 166837.28, transfer/sec: 21.161MiB, pipeline: 16 + concurrency: 128, request/sec: 40787.28, transfer/sec: 5.173MiB + concurrency: 256, request/sec: 29591.89, transfer/sec: 3.753MiB + concurrency: 480, request/sec: 27282.40, transfer/sec: 3.46MiB + concurrency: 128, request/sec: 209463.78, transfer/sec: 26.568MiB, pipeline: 16 + concurrency: 128, request/sec: 207632.55, transfer/sec: 26.336MiB, pipeline: 16 + concurrency: 128, request/sec: 191504.55, transfer/sec: 24.29MiB, pipeline: 16 asyncdispatch2 - concurrency: 128, request/sec: 328212.10, transfer/sec: 43.508MiB - concurrency: 256, request/sec: 254634.24, transfer/sec: 33.755MiB - concurrency: 480, request/sec: 220645.90, transfer/sec: 29.249MiB - concurrency: 128, request/sec: 358799.70, transfer/sec: 47.563MiB, pipeline: 16 - concurrency: 256, request/sec: 298887.75, transfer/sec: 39.622MiB, pipeline: 16 - concurrency: 480, request/sec: 135642.25, transfer/sec: 17.981MiB, pipeline: 16 -fasthttp - concurrency: 128, request/sec: 30920.02, transfer/sec: 3.745MiB - concurrency: 256, request/sec: 27362.95, transfer/sec: 3.314MiB - concurrency: 480, request/sec: 26008.47, transfer/sec: 3.15MiB - concurrency: 128, request/sec: 194787.96, transfer/sec: 23.592MiB, pipeline: 16 - concurrency: 256, request/sec: 191044.91, transfer/sec: 23.139MiB, pipeline: 16 - concurrency: 480, request/sec: 196284.84, transfer/sec: 23.773MiB, pipeline: 16 + concurrency: 128, request/sec: 333825.72, transfer/sec: 44.252MiB + concurrency: 256, request/sec: 300071.72, transfer/sec: 39.778MiB + concurrency: 480, request/sec: 264046.75, transfer/sec: 35.002MiB + concurrency: 128, request/sec: 414648.06, transfer/sec: 54.966MiB, pipeline: 16 + concurrency: 128, request/sec: 376728.50, transfer/sec: 49.939MiB, pipeline: 16 + concurrency: 128, request/sec: 278118.96, transfer/sec: 36.868MiB, pipeline: 16 +libreactor + concurrency: 128, request/sec: 51439.25, transfer/sec: 6.623MiB + concurrency: 256, request/sec: 41798.18, transfer/sec: 5.381MiB + concurrency: 480, request/sec: 35357.21, transfer/sec: 4.552MiB + concurrency: 128, request/sec: 430120.80, transfer/sec: 55.376MiB, pipeline: 16 + concurrency: 128, request/sec: 395858.25, transfer/sec: 50.965MiB, pipeline: 16 + concurrency: 128, request/sec: 426958.23, transfer/sec: 54.969MiB, pipeline: 16 actix-raw - concurrency: 128, request/sec: 40068.61, transfer/sec: 4.968MiB - concurrency: 256, request/sec: 33321.05, transfer/sec: 4.131MiB - concurrency: 480, request/sec: 29414.25, transfer/sec: 3.647MiB - concurrency: 128, request/sec: 245921.39, transfer/sec: 30.489MiB, pipeline: 16 - concurrency: 256, request/sec: 237474.19, transfer/sec: 29.441MiB, pipeline: 16 - concurrency: 480, request/sec: 244928.92, transfer/sec: 30.366MiB, pipeline: 16 + concurrency: 128, request/sec: 47562.36, transfer/sec: 5.897MiB + concurrency: 256, request/sec: 33645.82, transfer/sec: 4.171MiB + concurrency: 480, request/sec: 30841.45, transfer/sec: 3.824MiB + concurrency: 128, request/sec: 262156.16, transfer/sec: 32.502MiB, pipeline: 16 + concurrency: 128, request/sec: 282620.14, transfer/sec: 35.039MiB, pipeline: 16 + concurrency: 128, request/sec: 308789.86, transfer/sec: 38.283MiB, pipeline: 16 diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 6ab4d915c..6f61e0afb 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,7 +1,7 @@ import osproc, json, streams, strutils, os const - participants = ["mofuw", "asyncnet", "asyncdispatch2", "fasthttp", "actix-raw"] + participants = ["fasthttp", "mofuw", "asyncnet", "asyncdispatch2", "libreactor", "actix-raw"] proc execAndGetJson(command: string): JsonNode = const @@ -63,47 +63,84 @@ proc removeContainers() = let ID = x["ID"] removeContainer(ID.getStr()) +proc buildExe(name: string) = + let parentDir = getAppDir() + let curDir = parentDir & DirSep & name + setCurrentDir(curDir) + if execCmd("sh build.sh") != 0: + raise newException(Exception, "cannot build executable: " & name) + setCurrentDir(parentDir) + +proc buildExes() = + for c in participants: + buildExe(c) + +proc runServer(id: string): Process = + const options = {poParentStreams} + let parentDir = getAppDir() + let name = parentDir & DirSep & id & DirSep & "server" + result = startProcess(name, args=[], env=nil, options=options) + if not result.running(): + raise newException(Exception, "cannot run server: " & id) + +proc stopServer(p: Process) = + p.terminate() + p.kill() + discard p.waitForExit(1000) + p.close() + const levels = [128, 256, 480] maxConcurrency = levels[^1] - duration = 15 + primer_duration = 5 + warmup_duration = 10 + concurrency_duration = 15 + pipeline_duration = 15 + sleep_duration = 1000 serverHost = "bench-bot" url = "http://127.0.0.1:8080/plaintext" pipeline = 16 accept = "text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7" -proc runTest(name: string): JsonNode = +proc runTest(name: string, noDocker: bool): JsonNode = let maxThreads = countProcessors() + var cmd: string + var ret: int + var server: Process echo "** ", name, " **" - var cmd = "docker run -d -p 8080:8080 bench-$1" % [name] - var ret = execSilent(cmd) - if ret != 0: - raise newException(Exception, "cannot run image: " & name) - cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d 5 -c 8 --timeout 8 -t 8 $3" % - [serverHost, accept, url] + if noDocker: + server = runServer(name) + else: + cmd = "docker run -d -p 8080:8080 bench-$1" % [name] + ret = execSilent(cmd) + if ret != 0: + raise newException(Exception, "cannot run image: " & name & " " & $ret) + + cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c 8 --timeout 8 -t 8 $4" % + [serverHost, accept, $primer_duration, url] echo " Running Primer" ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run primer: " & name) - sleep(2000) + sleep(sleep_duration) cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6" % - [serverHost, accept, $duration, $(256), $maxThreads, url] + [serverHost, accept, $warmup_duration, $(256), $maxThreads, url] echo " Running Warmup" ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run warmup: " & name) - sleep(2000) + sleep(sleep_duration) var jar = newJArray() for c in levels: echo " Running Concurrency $1" % [$c] let t = max(c, maxThreads) let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s jsonfmt.lua" % - [serverHost, accept, $duration, $c, $t, url] + [serverHost, accept, $concurrency_duration, $c, $t, url] let res = execAndGetJson(cmd) if res.len == 1: var json = res[0] @@ -117,12 +154,14 @@ proc runTest(name: string): JsonNode = json.add("success", newJBool(false)) json.add("pipeline", newJInt(0)) jar.add json - sleep(2000) + + sleep(sleep_duration) for c in levels: + let c = 128 echo " Running Concurrency $1 with pipeline $2" % [$c, $pipeline] let t = max(c, maxThreads) let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % - [serverHost, accept, $duration, $c, $t, url, $pipeline] + [serverHost, accept, $pipeline_duration, $c, $t, url, $pipeline] let res = execAndGetJson(cmd) if res.len == 1: var json = res[0] @@ -136,9 +175,13 @@ proc runTest(name: string): JsonNode = json.add("success", newJBool(false)) json.add("pipeline", newJInt(pipeline)) jar.add json - sleep(2000) - killContainers() - removeContainers() + + if noDocker: + stopServer(server) + else: + sleep(sleep_duration) + killContainers() + removeContainers() result = newJObject() result.add("result", jar) @@ -169,13 +212,16 @@ proc renderResult(json: JsonNode, s: Stream) = let concurrency = c["level"].getInt() s.writeLine(" concurrency: $1, failed" % [$concurrency]) -proc runAllTest() = - buildImages() +proc runAllTest(noDocker: bool) = + if noDocker: + buildExes() + else: + buildImages() var resList = newSeq[JsonNode]() for p in participants: try: - let res = runTest(p) + let res = runTest(p, noDocker) resList.add res except Exception as e: echo e.msg @@ -190,9 +236,13 @@ proc runAllTest() = res.renderResult(ss) echo ss.data -proc runSingleTest(name: string) = - buildImage(name) - let res = runTest(name) +proc runSingleTest(name: string, noDocker: bool) = + if noDocker: + buildExe(name) + else: + buildImage(name) + + let res = runTest(name, noDocker) var s = newStringStream() res.renderResult(s) echo s.data @@ -216,8 +266,9 @@ proc installWrk(forceInstall: bool = false): bool = let curDir = parentDir & DirSep & "wrksrc" discard existsOrCreateDir(curDir) setCurrentDir(curDir) + echo "build wrk..." if not fileExists(curDir & DirSep & "4.1.0.tar.gz") or forceInstall: - result = execCmd("wget https://github.com/wg/wrk/archive/4.1.0.tar.gz") == 0 + result = execCmd("wget -q https://github.com/wg/wrk/archive/4.1.0.tar.gz") == 0 if result or forceInstall: result = execCmd("tar xzf 4.1.0.tar.gz --strip-components=1 --skip-old-files") == 0 if result or forceInstall: @@ -237,7 +288,8 @@ proc printHelp() = for c in participants: echo " ", c -proc runCommand(cmd: string) = +proc runCommand(cmd, options: string) = + var noDocker = options == "nodocker" case cmd of "clean": killContainers() @@ -247,7 +299,7 @@ proc runCommand(cmd: string) = discard installWrk(true) of "all": discard installWrk() - runAllTest() + runAllTest(noDocker) of "help": printHelp() else: @@ -255,12 +307,14 @@ proc runCommand(cmd: string) = echo cmd & " is not a registered participant" return discard installWrk() - runSingleTest(cmd) + runSingleTest(cmd, noDocker) proc main() = if paramCount() > 0: let cmd = paramStr(1) - runCommand(cmd) + let options = if paramCount() > 1: + paramStr(2) else: "" + runCommand(cmd, options) else: printHelp() diff --git a/tests/benchmark/fasthttp/src/common/common.go b/tests/benchmark/fasthttp/src/common/common.go deleted file mode 100644 index 86a0ec1af..000000000 --- a/tests/benchmark/fasthttp/src/common/common.go +++ /dev/null @@ -1,56 +0,0 @@ -package common - -import ( - "encoding/json" - "flag" - "log" - "net" - "sync" - "github.com/valyala/fasthttp" -) - -const worldRowCount = 10000 - -type JSONResponse struct { - Message string `json:"message"` -} - -type World struct { - Id int32 `json:"id"` - RandomNumber int32 `json:"randomNumber"` -} - -var listenAddr = flag.String("listenAddr", ":8080", "Address to listen to") - -func JSONHandler(ctx *fasthttp.RequestCtx) { - r := jsonResponsePool.Get().(*JSONResponse) - r.Message = "Hello, World!" - JSONMarshal(ctx, r) - jsonResponsePool.Put(r) -} - -var jsonResponsePool = &sync.Pool{ - New: func() interface{} { - return &JSONResponse{} - }, -} - -func PlaintextHandler(ctx *fasthttp.RequestCtx) { - ctx.SetContentType("text/plain") - ctx.WriteString("Hello, World!") -} - -func JSONMarshal(ctx *fasthttp.RequestCtx, v interface{}) { - ctx.SetContentType("application/json") - if err := json.NewEncoder(ctx).Encode(v); err != nil { - log.Fatalf("error in json.Encoder.Encode: %s", err) - } -} - -func GetListener() net.Listener { - ln, err := net.Listen("tcp4", *listenAddr) - if err != nil { - log.Fatal(err) - } - return ln -} diff --git a/tests/benchmark/fasthttp/src/server/server.go b/tests/benchmark/fasthttp/src/server/server.go index 3c8fd580a..94423e00e 100644 --- a/tests/benchmark/fasthttp/src/server/server.go +++ b/tests/benchmark/fasthttp/src/server/server.go @@ -4,8 +4,8 @@ import ( "flag" "log" "github.com/valyala/fasthttp" - "common" "runtime" + "net" ) func main() { @@ -18,7 +18,7 @@ func main() { Handler: mainHandler, Name: "go", } - ln := common.GetListener() + ln := GetListener() if err = s.Serve(ln); err != nil { log.Fatalf("Error when serving incoming connections: %s", err) } @@ -28,11 +28,23 @@ func mainHandler(ctx *fasthttp.RequestCtx) { path := ctx.Path() switch string(path) { case "/plaintext": - common.PlaintextHandler(ctx) - case "/json": - common.JSONHandler(ctx) + PlaintextHandler(ctx) default: ctx.Error("unexpected path", fasthttp.StatusBadRequest) } } +var listenAddr = flag.String("listenAddr", ":8080", "Address to listen to") + +func PlaintextHandler(ctx *fasthttp.RequestCtx) { + ctx.SetContentType("text/plain") + ctx.WriteString("Hello, World!") +} + +func GetListener() net.Listener { + ln, err := net.Listen("tcp4", *listenAddr) + if err != nil { + log.Fatal(err) + } + return ln +} diff --git a/tests/benchmark/libreactor/Makefile b/tests/benchmark/libreactor/Makefile new file mode 100644 index 000000000..f1e3f2dd7 --- /dev/null +++ b/tests/benchmark/libreactor/Makefile @@ -0,0 +1,11 @@ +CC = gcc-6 +PROG = server +OBJS = src/setup.o src/main.o +CFLAGS = -std=gnu11 -Wall -O3 -march=native -mtune=native -flto -fuse-linker-plugin -Isrc +LDADD = -lreactor -ldynamic -lclo + +$(PROG): $(OBJS) + $(CC) -o $@ $^ $(CFLAGS) $(LDADD) + +clean: + rm -f $(PROG) $(OBJS) diff --git a/tests/benchmark/libreactor/plaintext.dockerfile b/tests/benchmark/libreactor/plaintext.dockerfile new file mode 100644 index 000000000..7332ee459 --- /dev/null +++ b/tests/benchmark/libreactor/plaintext.dockerfile @@ -0,0 +1,34 @@ +FROM ubuntu:16.04 + +RUN apt update -yqq +RUN apt install -yqq software-properties-common python-software-properties wget make + +RUN add-apt-repository ppa:ubuntu-toolchain-r/test -y && \ + apt update -yqq && \ + apt install -yqq gcc-6 g++-6 + +ADD ./ /libreactor +WORKDIR /libreactor + +RUN wget -q https://github.com/fredrikwidlund/libdynamic/releases/download/v1.1.0/libdynamic-1.1.0.tar.gz && \ + tar xfz libdynamic-1.1.0.tar.gz && \ + cd libdynamic-1.1.0 && \ + ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + make && make install + +RUN wget -q https://github.com/fredrikwidlund/libreactor/releases/download/v1.0.0/libreactor-1.0.0.tar.gz && \ + tar xfz libreactor-1.0.0.tar.gz && \ + cd libreactor-1.0.0 && \ + sed -i -e 's,REACTOR_POOL_WORKERS_MAX,1,' src/reactor/reactor_pool.c && \ + ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + make && make install + +RUN wget -q https://github.com/fredrikwidlund/libclo/releases/download/v0.1.0/libclo-0.1.0.tar.gz && \ + tar xfz libclo-0.1.0.tar.gz && \ + cd libclo-0.1.0 && \ + ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + make && make install + +RUN make clean && make + +CMD ["./server"] diff --git a/tests/benchmark/libreactor/src/main.c b/tests/benchmark/libreactor/src/main.c new file mode 100644 index 000000000..05d97bcf0 --- /dev/null +++ b/tests/benchmark/libreactor/src/main.c @@ -0,0 +1,140 @@ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include "setup.h" + +static char date[] = "Thu, 01 Jan 1970 00:00:00 GMT"; + +static int timer_event(void *state, int type, void *data) +{ + (void) state; + (void) data; + if (type != REACTOR_TIMER_EVENT_CALL) + err(1, "timer"); + + reactor_http_date(date); + return REACTOR_OK; +} + +static int plaintext(reactor_http *http) +{ + char content_length[11], *body = "Hello, World!"; + size_t size; + + size = strlen(body); + reactor_util_u32toa(size, content_length); + reactor_http_send_response(http, (reactor_http_response[]){{ + .version = 1, + .status = 200, + .reason = {"OK", 2}, + .headers = 4, + .header[0] = {{"Server", 6}, {"libreactor", 10}}, + .header[1] = {{"Date", 4}, {date, strlen(date)}}, + .header[2] = {{"Content-Type", 12}, {"text/plain", 10}}, + .header[3] = {{"Content-Length", 14}, {content_length, strlen(content_length)}}, + .body = {body, size} + }}); + return REACTOR_OK; +} + +static int json(reactor_http *http) +{ + char body[4096], content_length[11]; + size_t size; + + (void) clo_encode((clo[]) {clo_object({"message", clo_string("Hello, World!")})}, body, sizeof(body)); + size = strlen(body); + reactor_util_u32toa(size, content_length); + reactor_http_send_response(http, (reactor_http_response[]){{ + .version = 1, + .status = 200, + .reason = {"OK", 2}, + .headers = 4, + .header[0] = {{"Server", 6}, {"libreactor", 10}}, + .header[1] = {{"Date", 4}, {date, strlen(date)}}, + .header[2] = {{"Content-Type", 12}, {"application/json", 16}}, + .header[3] = {{"Content-Length", 14}, {content_length, strlen(content_length)}}, + .body = {body, size} + }}); + return REACTOR_OK; +} + +int http_event(void *state, int type, void *data) +{ + reactor_http *http = state; + reactor_http_request *request; + + if (reactor_unlikely(type != REACTOR_HTTP_EVENT_REQUEST)) + { + reactor_http_close(http); + free(http); + return REACTOR_ABORT; + } + request = data; + + if (reactor_http_header_value(&request->path, "/plaintext")) + return plaintext(http); + else if (reactor_http_header_value(&request->path, "/json")) + return json(http); + else + { + reactor_http_send_response(http, (reactor_http_response[]){{ + .version = 1, + .status = 404, + .reason = {"Not Found", 9}, + .headers = 3, + .header[0] = {{"Server", 6}, {"libreactor", 10}}, + .header[1] = {{"Date", 4}, {date, strlen(date)}}, + .header[2] = {{"Content-Length", 14}, {"0", 1}}, + .body = {NULL, 0} + }}); + return REACTOR_OK; + } +} + +static int tcp_event(void *state, int type, void *data) +{ + reactor_http *http; + + if (type != REACTOR_TCP_EVENT_ACCEPT) + err(1, "tcp"); + + http = malloc(sizeof *http); + if (!http) + abort(); + (void) reactor_http_open(http, http_event, http, *(int *) data, REACTOR_HTTP_FLAG_SERVER); + return REACTOR_OK; +} + +int main() +{ + reactor_timer timer; + reactor_tcp tcp; + + setup(1, 0); + (void) reactor_core_construct(); + (void) reactor_timer_open(&timer, timer_event, &timer, 1, 1000000000); + (void) reactor_tcp_open(&tcp, tcp_event, &tcp, "0.0.0.0", "8080", REACTOR_TCP_FLAG_SERVER); + (void) reactor_core_run(); + reactor_core_destruct(); +} diff --git a/tests/benchmark/libreactor/src/setup.c b/tests/benchmark/libreactor/src/setup.c new file mode 100644 index 000000000..82582ecf0 --- /dev/null +++ b/tests/benchmark/libreactor/src/setup.c @@ -0,0 +1,107 @@ +#define _GNU_SOURCE + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +struct cpu +{ + int id; + pid_t pid; + cpu_set_t set; +}; + +int realtime(void) +{ + struct sched_param param; + struct rlimit rlim; + int e; + + e = sched_getparam(0, ¶m); + if (e == -1) + return -1; + + param.sched_priority = sched_get_priority_max(SCHED_FIFO); + rlim = (struct rlimit) {.rlim_cur = param.sched_priority, .rlim_max = param.sched_priority}; + e = prlimit(0, RLIMIT_RTPRIO, &rlim, NULL); + if (e == -1) + return -1; + + e = sched_setscheduler(0, SCHED_FIFO, ¶m); + if (e == -1) + return -1; + + return 0; +} + +static void cpu_list(vector *list, int sib) +{ + struct cpu cpu; + char path[1024], buf[1024]; + size_t i, id; + ssize_t n; + int fd; + + vector_construct(list, sizeof cpu); + for (i = 0;; i ++) + { + snprintf(path, sizeof path, "/sys/devices/system/cpu/cpu%lu/topology/thread_siblings_list", i); + fd = open(path, O_RDONLY); + if (fd == -1) + break; + n = read(fd, buf, sizeof buf - 1); + if (n == -1) + err(1, "read"); + buf[n] = 0; + close(fd); + + id = strtoul(buf, NULL, 0); + if (!sib && id != i) + continue; + + cpu = (struct cpu) {0}; + cpu.id = id; + CPU_ZERO(&cpu.set); + CPU_SET(id, &cpu.set); + vector_push_back(list, &cpu); + } +} + +void setup(size_t skip, int sib) +{ + vector list; + struct cpu *cpu; + size_t i; + int e; + pid_t cpid; + + signal(SIGPIPE, SIG_IGN); + cpu_list(&list, sib); + for (i = 0; i < vector_size(&list); i ++) + { + if (i % skip != 0) + continue; + cpu = vector_at(&list, i); + cpid = fork(); + if (cpid == -1) + err(1, "fork"); + if (cpid == 0) + { + e = sched_setaffinity(0, sizeof cpu->set, &cpu->set); + if (e == -1) + err(1, "sched_setaffinity"); + return; + } + cpu->pid = cpid; + } + wait(NULL); + vector_destruct(&list, NULL); +} diff --git a/tests/benchmark/libreactor/src/setup.h b/tests/benchmark/libreactor/src/setup.h new file mode 100644 index 000000000..be521a78a --- /dev/null +++ b/tests/benchmark/libreactor/src/setup.h @@ -0,0 +1,6 @@ +#ifndef SETUP_H_INCLUDED +#define SETUP_H_INCLUDED + +void setup(size_t, int); + +#endif /* SETUP_H_INCLUDED */ diff --git a/tests/benchmark/mofuw/README.md b/tests/benchmark/mofuw/README.md deleted file mode 100644 index b7b5540c6..000000000 --- a/tests/benchmark/mofuw/README.md +++ /dev/null @@ -1,11 +0,0 @@ -# Nim's Web Framework \"mofuw\" - -> mofuw is \***MO**re\* \***F**\*aster, \***U**ltra\* \***W**\*eb server. - -## Includes tests -- plaintext: `localhost:8080/plaintext` -- json: `localhost:8080/json` - -## Require -- Nim => 0.18.0 -- mofuw => 1.1.8 \ No newline at end of file diff --git a/tests/benchmark/readme.md b/tests/benchmark/readme.md index e61b6b5a1..8948bd2e1 100644 --- a/tests/benchmark/readme.md +++ b/tests/benchmark/readme.md @@ -14,7 +14,7 @@ Participants in this benchmark are: * __asyncdispatch__, lang: Nim, from Nim stdlib * __actix-raw__, lang: Rust, TFB round 16th rank 1 * __fasthttp__, lang: Go, TFB round 16th rank 2 -* __ulib-plaintext_fit__, lang: C++, TFB round 16th rank 3 +* __libreactor__, lang: C, TFB round 16th rank 5 Participants from TFB round 16th are impressive, how asyncdispatch2 compared to them? This is the goals of this benchmark. The rank of TFB round 16th is based on plaintext category, running on a physical server. @@ -38,7 +38,7 @@ Each test is executed as follows: * Run a 5-second __primer__ at 8 client-concurrency to verify that the server is in fact running. These results are not captured. * Run a 15-second __warmup__ at 256 client-concurrency to allow lazy-initialization to execute and just-in-time compilation to run. These results are not captured. * Run a 15-second __captured test__ for each of the concurrency levels (or iteration counts) exercised by the test type. - The high-concurrency plaintext test type is tested at 128, 256, and 512 client-side concurrency. + The high-concurrency plaintext test type is tested at 128, 256, 512, and 1024 client-side concurrency. * Stop the platform and framework. ## How to replicate this test locally? @@ -53,5 +53,16 @@ Each test is executed as follows: * inside that directory prepare a `plaintext.dockerfile` and all necessary source code. * add a entry in `bot.nim` participants constant list with the directory name. -## How to switch to multi thread mode? -You can find a commented line in the source code of each framework to enable/disable multithread +## Summary +* __mofuw__, mofuw use asyncdispatch, expected performance should not more than asynchdispatch itself. +* __asyncdispatch__, although it is slower than asyncdispatch2, it can handle high concurrency quite well. +* __asyncdispatch2__, at high concurrency it has tendency become slower significantly, + but surpringsingly it is the only framework in this test that can handle non pipeline request faster than other + frameworks although using almost identical code with asyncdispatch when handle request/response. +* __actix-raw__, very fast when multi threaded, not so when single threaded. +* __fasthttp__, very fast when multi threaded, not so when single threaded. +* __libreactor__, still very fast although in single thread mode. + +## Conclusion +* asyncdispatch2 could be a good candidate to replace asycndispatch +* it still has room for improvement especially when handle high count connections. diff --git a/tests/benchmark/ulib/plaintext.dockerfile b/tests/benchmark/ulib/plaintext.dockerfile deleted file mode 100644 index d90ecb815..000000000 --- a/tests/benchmark/ulib/plaintext.dockerfile +++ /dev/null @@ -1,74 +0,0 @@ -FROM ubuntu:18.04 - -COPY ./ ./ - -RUN apt update -yqq && \ - apt install -yqq software-properties-common build-essential locales wget \ - libreadline6-dev \ - libgdbm-dev ncurses-dev \ - libffi-dev libtool bison libevent-dev \ - liborc-0.4-0 \ - libmcrypt-dev libicu-dev \ - libcap2-bin && \ - add-apt-repository ppa:ubuntu-toolchain-r/test -y && \ - apt update -yqq && \ - apt install -yqq gcc-8 g++-8 - -RUN locale-gen en_US.UTF-8 - -ENV LANG en_US.UTF-8 -ENV LANGUAGE en_US:en -ENV LC_ALL en_US.UTF-8 -ENV DEBIAN_FRONTEND noninteractive - -ENV CC=gcc-8 -ENV CXX=g++-8 -ENV AR=gcc-ar-8 -ENV RANLIB=gcc-ranlib-8 -ENV IROOT=/install -ENV ULIB_ROOT=$IROOT/ULib -ENV ULIB_VERSION=1.4.2 -ENV ULIB_DOCUMENT_ROOT=$ULIB_ROOT/ULIB_DOCUMENT_ROOT - -WORKDIR $IROOT - -RUN mkdir -p $ULIB_DOCUMENT_ROOT -RUN wget -q -O ULib-${ULIB_VERSION}.tar.gz https://github.com/stefanocasazza/ULib/archive/v${ULIB_VERSION}.tar.gz -RUN tar xf ULib-${ULIB_VERSION}.tar.gz - -WORKDIR $IROOT/ULib-$ULIB_VERSION - -# AVOID "configure: error: newly created file is older than distributed files! Check your system clock" -#RUN cp /src/* src/ulib/net/server/plugin/usp -RUN find . -exec touch {} \; - -RUN echo "userver {" >> $ULIB_ROOT/benchmark.cfg -RUN echo "PORT 8080" >> $ULIB_ROOT/benchmark.cfg -RUN echo "PREFORK_CHILD $(nproc)" >> $ULIB_ROOT/benchmark.cfg -RUN echo "TCP_LINGER_SET 0" >> $ULIB_ROOT/benchmark.cfg -RUN echo "LISTEN_BACKLOG 16384" >> $ULIB_ROOT/benchmark.cfg -RUN echo "DOCUMENT_ROOT $ULIB_DOCUMENT_ROOT " >> $ULIB_ROOT/benchmark.cfg -RUN echo "}" >> $ULIB_ROOT/benchmark.cfg - -RUN USP_FLAGS="-DAS_cpoll_cppsp_DO" \ - ./configure --prefix=$ULIB_ROOT \ - --disable-static --disable-examples \ - --without-ssl --without-pcre --without-expat \ - --without-libz --without-libuuid --without-magic --without-libares \ - --enable-static-server-plugin=http - -RUN make install && \ - cd examples/userver && make install && \ - cd ../../src/ulib/net/server/plugin/usp && \ - make plaintext.la && \ - cp .libs/plaintext.so $ULIB_DOCUMENT_ROOT - -ENV PATH=${ULIB_ROOT}/bin:${PATH} - -ADD ./ /ulib -WORKDIR /ulib - -ENV UMEMPOOL="58,0,0,41,16401,-14,-15,11,25" - -CMD setcap cap_sys_nice,cap_sys_resource,cap_net_bind_service,cap_net_raw+eip $IROOT/ULib/bin/userver_tcp && \ - $IROOT/ULib/bin/userver_tcp -c $IROOT/ULib/benchmark.cfg diff --git a/tests/benchmark/ulib/src/plaintext.usp b/tests/benchmark/ulib/src/plaintext.usp deleted file mode 100644 index ba06d3f32..000000000 --- a/tests/benchmark/ulib/src/plaintext.usp +++ /dev/null @@ -1,18 +0,0 @@ - - From b457c22e9d6e6ac1c4bc7733ae1cfb6729e55f8a Mon Sep 17 00:00:00 2001 From: jangko Date: Tue, 11 Sep 2018 20:46:17 +0700 Subject: [PATCH 10/12] fix bot.nim --- tests/benchmark/benchmark_result.txt | 72 ++++++++++++++-------------- tests/benchmark/bot.nim | 1 - 2 files changed, 36 insertions(+), 37 deletions(-) diff --git a/tests/benchmark/benchmark_result.txt b/tests/benchmark/benchmark_result.txt index c16859f3a..d514f82b5 100644 --- a/tests/benchmark/benchmark_result.txt +++ b/tests/benchmark/benchmark_result.txt @@ -1,42 +1,42 @@ fasthttp - concurrency: 128, request/sec: 33192.01, transfer/sec: 4.02MiB - concurrency: 256, request/sec: 31513.33, transfer/sec: 3.817MiB - concurrency: 480, request/sec: 29085.05, transfer/sec: 3.523MiB - concurrency: 128, request/sec: 228440.91, transfer/sec: 27.668MiB, pipeline: 16 - concurrency: 128, request/sec: 223112.01, transfer/sec: 27.023MiB, pipeline: 16 - concurrency: 128, request/sec: 230501.58, transfer/sec: 27.918MiB, pipeline: 16 + concurrency: 128, request/sec: 35642.59, transfer/sec: 4.317MiB + concurrency: 256, request/sec: 31259.81, transfer/sec: 3.786MiB + concurrency: 480, request/sec: 30540.95, transfer/sec: 3.699MiB + concurrency: 128, request/sec: 233601.35, transfer/sec: 28.293MiB, pipeline: 16 + concurrency: 256, request/sec: 245405.55, transfer/sec: 29.723MiB, pipeline: 16 + concurrency: 480, request/sec: 241039.37, transfer/sec: 29.194MiB, pipeline: 16 mofuw - concurrency: 128, request/sec: 36500.58, transfer/sec: 4.525MiB - concurrency: 256, request/sec: 27654.28, transfer/sec: 3.429MiB - concurrency: 480, request/sec: 25377.77, transfer/sec: 3.146MiB - concurrency: 128, request/sec: 195180.88, transfer/sec: 24.198MiB, pipeline: 16 - concurrency: 128, request/sec: 189706.99, transfer/sec: 23.519MiB, pipeline: 16 - concurrency: 128, request/sec: 194490.21, transfer/sec: 24.112MiB, pipeline: 16 + concurrency: 128, request/sec: 36968.21, transfer/sec: 4.583MiB + concurrency: 256, request/sec: 29659.96, transfer/sec: 3.677MiB + concurrency: 480, request/sec: 26220.49, transfer/sec: 3.251MiB + concurrency: 128, request/sec: 195822.97, transfer/sec: 24.278MiB, pipeline: 16 + concurrency: 256, request/sec: 185300.42, transfer/sec: 22.973MiB, pipeline: 16 + concurrency: 480, request/sec: 184970.62, transfer/sec: 22.932MiB, pipeline: 16 asyncnet - concurrency: 128, request/sec: 40787.28, transfer/sec: 5.173MiB - concurrency: 256, request/sec: 29591.89, transfer/sec: 3.753MiB - concurrency: 480, request/sec: 27282.40, transfer/sec: 3.46MiB - concurrency: 128, request/sec: 209463.78, transfer/sec: 26.568MiB, pipeline: 16 - concurrency: 128, request/sec: 207632.55, transfer/sec: 26.336MiB, pipeline: 16 - concurrency: 128, request/sec: 191504.55, transfer/sec: 24.29MiB, pipeline: 16 + concurrency: 128, request/sec: 40600.51, transfer/sec: 5.15MiB + concurrency: 256, request/sec: 31916.33, transfer/sec: 4.048MiB + concurrency: 480, request/sec: 29051.34, transfer/sec: 3.685MiB + concurrency: 128, request/sec: 205472.21, transfer/sec: 26.062MiB, pipeline: 16 + concurrency: 256, request/sec: 196334.33, transfer/sec: 24.903MiB, pipeline: 16 + concurrency: 480, request/sec: 166179.75, transfer/sec: 21.078MiB, pipeline: 16 asyncdispatch2 - concurrency: 128, request/sec: 333825.72, transfer/sec: 44.252MiB - concurrency: 256, request/sec: 300071.72, transfer/sec: 39.778MiB - concurrency: 480, request/sec: 264046.75, transfer/sec: 35.002MiB - concurrency: 128, request/sec: 414648.06, transfer/sec: 54.966MiB, pipeline: 16 - concurrency: 128, request/sec: 376728.50, transfer/sec: 49.939MiB, pipeline: 16 - concurrency: 128, request/sec: 278118.96, transfer/sec: 36.868MiB, pipeline: 16 + concurrency: 128, request/sec: 340199.45, transfer/sec: 45.097MiB + concurrency: 256, request/sec: 297123.12, transfer/sec: 39.387MiB + concurrency: 480, request/sec: 310110.03, transfer/sec: 41.108MiB + concurrency: 128, request/sec: 429391.11, transfer/sec: 56.92MiB, pipeline: 16 + concurrency: 256, request/sec: 324845.35, transfer/sec: 43.062MiB, pipeline: 16 + concurrency: 480, request/sec: 76255.27, transfer/sec: 10.108MiB, pipeline: 16 libreactor - concurrency: 128, request/sec: 51439.25, transfer/sec: 6.623MiB - concurrency: 256, request/sec: 41798.18, transfer/sec: 5.381MiB - concurrency: 480, request/sec: 35357.21, transfer/sec: 4.552MiB - concurrency: 128, request/sec: 430120.80, transfer/sec: 55.376MiB, pipeline: 16 - concurrency: 128, request/sec: 395858.25, transfer/sec: 50.965MiB, pipeline: 16 - concurrency: 128, request/sec: 426958.23, transfer/sec: 54.969MiB, pipeline: 16 + concurrency: 128, request/sec: 49323.05, transfer/sec: 6.35MiB + concurrency: 256, request/sec: 38011.93, transfer/sec: 4.894MiB + concurrency: 480, request/sec: 34812.91, transfer/sec: 4.482MiB + concurrency: 128, request/sec: 488500.34, transfer/sec: 62.892MiB, pipeline: 16 + concurrency: 256, request/sec: 396812.95, transfer/sec: 51.088MiB, pipeline: 16 + concurrency: 480, request/sec: 391383.08, transfer/sec: 50.389MiB, pipeline: 16 actix-raw - concurrency: 128, request/sec: 47562.36, transfer/sec: 5.897MiB - concurrency: 256, request/sec: 33645.82, transfer/sec: 4.171MiB - concurrency: 480, request/sec: 30841.45, transfer/sec: 3.824MiB - concurrency: 128, request/sec: 262156.16, transfer/sec: 32.502MiB, pipeline: 16 - concurrency: 128, request/sec: 282620.14, transfer/sec: 35.039MiB, pipeline: 16 - concurrency: 128, request/sec: 308789.86, transfer/sec: 38.283MiB, pipeline: 16 + concurrency: 128, request/sec: 47750.42, transfer/sec: 5.92MiB + concurrency: 256, request/sec: 38619.76, transfer/sec: 4.788MiB + concurrency: 480, request/sec: 33416.67, transfer/sec: 4.143MiB + concurrency: 128, request/sec: 297969.98, transfer/sec: 36.942MiB, pipeline: 16 + concurrency: 256, request/sec: 289400.35, transfer/sec: 35.879MiB, pipeline: 16 + concurrency: 480, request/sec: 293231.13, transfer/sec: 36.354MiB, pipeline: 16 diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 6f61e0afb..0fa786457 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -157,7 +157,6 @@ proc runTest(name: string, noDocker: bool): JsonNode = sleep(sleep_duration) for c in levels: - let c = 128 echo " Running Concurrency $1 with pipeline $2" % [$c, $pipeline] let t = max(c, maxThreads) let cmd = "./wrk -H \"Host: $1\" -H \"Accept: $2\" -H \"Connection: keep-alive\" --latency -d $3 -c $4 --timeout 8 -t $5 $6 -s pipeline.lua -- $7" % From 4fa3b6e3c7096cc72c33264c4174abe8334c064b Mon Sep 17 00:00:00 2001 From: Zahary Karadjov Date: Fri, 21 Sep 2018 20:18:05 +0300 Subject: [PATCH 11/12] Command-line options 'nothreads' for the benchmark bot The test programs are supposed to check for the 'USE_THREADS' environment variable to decide wether threads should be used. So far, this has been implemented in the Go and Rust benchmarks. The bot will set this variable by default. Other changes: * The Rust implementation has been updated with the latest code from the TechEmperor benchmark and the incremental Docker build process has been greatly improved --- tests/benchmark/.gitignore | 3 + .../benchmark/actix-raw/plaintext.dockerfile | 16 +++-- tests/benchmark/actix-raw/src/main.rs | 69 +++++-------------- tests/benchmark/asyncdispatch2/server.nim | 2 +- tests/benchmark/bot.nim | 50 +++++++++----- tests/benchmark/fasthttp/plaintext.dockerfile | 3 +- tests/benchmark/fasthttp/src/server/server.go | 8 ++- 7 files changed, 76 insertions(+), 75 deletions(-) create mode 100644 tests/benchmark/.gitignore diff --git a/tests/benchmark/.gitignore b/tests/benchmark/.gitignore new file mode 100644 index 000000000..e298a5e04 --- /dev/null +++ b/tests/benchmark/.gitignore @@ -0,0 +1,3 @@ +bot +wrk +wrksrc/ diff --git a/tests/benchmark/actix-raw/plaintext.dockerfile b/tests/benchmark/actix-raw/plaintext.dockerfile index d7f8fe4ba..a3e0633e1 100644 --- a/tests/benchmark/actix-raw/plaintext.dockerfile +++ b/tests/benchmark/actix-raw/plaintext.dockerfile @@ -1,11 +1,19 @@ FROM rust:1.28-slim -ADD ./ /actix -WORKDIR /actix - RUN apt update -yqq && \ apt install -yqq libpq-dev -RUN cargo clean + +RUN mkdir -p /opt/actix/src +RUN touch /opt/actix/src/lib.rs + +ADD ./Cargo.toml /opt/actix/ +WORKDIR /opt/actix +RUN RUSTFLAGS="-C target-cpu=native" cargo build --release + +RUN rm -rf /opt/actix/src/ +ADD ./src /opt/actix/src + RUN RUSTFLAGS="-C target-cpu=native" cargo build --release CMD ./target/release/actix +# CMD USE_THREADS=1 ./target/release/actix diff --git a/tests/benchmark/actix-raw/src/main.rs b/tests/benchmark/actix-raw/src/main.rs index f8284d015..35b8669af 100644 --- a/tests/benchmark/actix-raw/src/main.rs +++ b/tests/benchmark/actix-raw/src/main.rs @@ -2,65 +2,32 @@ extern crate actix; extern crate actix_web; extern crate futures; -use actix::prelude::*; -use actix_web::server::{ - self, HttpHandler, HttpHandlerTask, HttpServer, Request, Writer, -}; -use actix_web::Error; -use futures::{Async, Poll}; - -const HTTPOK: &[u8] = b"HTTP/1.1 200 OK\r\n"; -const HDR_SERVER: &[u8] = b"Server: Actix\r\n"; -const HDR_CTPLAIN: &[u8] = b"Content-Type: text/plain"; -const BODY: &[u8] = b"Hello, World!"; - -struct App { -} +use std::env; -impl HttpHandler for App { - type Task = Box; - - fn handle(&self, req: Request) -> Result, Request> { - { - let path = req.path(); - match path.len() { - 10 if path == "/plaintext" => return Ok(Box::new(Plaintext)), - _ => (), - } - } - Err(req) - } -} - -struct Plaintext; +use actix::prelude::*; +use actix_web::{http, server, App, HttpRequest, HttpResponse}; -impl HttpHandlerTask for Plaintext { - fn poll_io(&mut self, io: &mut Writer) -> Poll { - { - let mut bytes = io.buffer(); - bytes.reserve(196); - bytes.extend_from_slice(HTTPOK); - bytes.extend_from_slice(HDR_SERVER); - bytes.extend_from_slice(HDR_CTPLAIN); - server::write_content_length(13, &mut bytes); - } - io.set_date(); - io.buffer().extend_from_slice(BODY); - Ok(Async::Ready(true)) - } +fn plaintext(req: &HttpRequest) -> HttpResponse { + HttpResponse::build_from(req) + .header(http::header::SERVER, "Actix") + .header(http::header::CONTENT_TYPE, "text/plain") + .body("Hello, World!") } fn main() { let sys = System::new("techempower"); // start http server - HttpServer::new(move || { - vec![App { }] - }).backlog(8192) - .workers(1) // comment this to enable multithread mode - .bind("0.0.0.0:8080") - .unwrap() - .start(); + let srv = server::new(move || { + App::new() + .resource("/plaintext", |r| r.f(plaintext)) + }).backlog(8192); + + if env::var_os("USE_THREADS").is_none() { + srv.workers(1) + } else { + srv + }.bind("0.0.0.0:8080").unwrap().start(); println!("Started http server: 127.0.0.1:8080"); let _ = sys.run(); diff --git a/tests/benchmark/asyncdispatch2/server.nim b/tests/benchmark/asyncdispatch2/server.nim index 039183341..6576f6a75 100644 --- a/tests/benchmark/asyncdispatch2/server.nim +++ b/tests/benchmark/asyncdispatch2/server.nim @@ -56,9 +56,9 @@ proc resetBuffer(ctx: IncomingCtx) = ctx.bufLen = 0 proc sendMessage(ctx: IncomingCtx, body: string) = + let ol = ctx.respLen while unlikely ctx.respLen + body.len > ctx.resp.len: ctx.resp.setLen(ctx.resp.len + ctx.resp.len) - let ol = ctx.respLen copyMem(addr ctx.resp[ol], unsafeAddr body[0], body.len) ctx.respLen += body.len diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index 0fa786457..c6eff8e5d 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,5 +1,12 @@ import osproc, json, streams, strutils, os +type + Option = enum + optNoDocker + optNoThreads + + Options = set[Option] + const participants = ["fasthttp", "mofuw", "asyncnet", "asyncdispatch2", "libreactor", "actix-raw"] @@ -102,7 +109,7 @@ const pipeline = 16 accept = "text/plain,text/html;q=0.9,application/xhtml+xml;q=0.9,application/xml;q=0.8,*/*;q=0.7" -proc runTest(name: string, noDocker: bool): JsonNode = +proc runTest(name: string, options: Options): JsonNode = let maxThreads = countProcessors() var cmd: string var ret: int @@ -110,10 +117,12 @@ proc runTest(name: string, noDocker: bool): JsonNode = echo "** ", name, " **" - if noDocker: + if optNoDocker in options: server = runServer(name) else: - cmd = "docker run -d -p 8080:8080 bench-$1" % [name] + let useTheadsFlag = if optNoThreads in options: "" + else: "-e USE_THREADS=1" + cmd = "docker run -d -p 8080:8080 $1 bench-$2" % [useTheadsFlag, name] ret = execSilent(cmd) if ret != 0: raise newException(Exception, "cannot run image: " & name & " " & $ret) @@ -175,7 +184,7 @@ proc runTest(name: string, noDocker: bool): JsonNode = json.add("pipeline", newJInt(pipeline)) jar.add json - if noDocker: + if optNoDocker in options: stopServer(server) else: sleep(sleep_duration) @@ -211,8 +220,8 @@ proc renderResult(json: JsonNode, s: Stream) = let concurrency = c["level"].getInt() s.writeLine(" concurrency: $1, failed" % [$concurrency]) -proc runAllTest(noDocker: bool) = - if noDocker: +proc runAllTest(options: Options) = + if optNoDocker in options: buildExes() else: buildImages() @@ -220,7 +229,7 @@ proc runAllTest(noDocker: bool) = var resList = newSeq[JsonNode]() for p in participants: try: - let res = runTest(p, noDocker) + let res = runTest(p, options) resList.add res except Exception as e: echo e.msg @@ -235,13 +244,13 @@ proc runAllTest(noDocker: bool) = res.renderResult(ss) echo ss.data -proc runSingleTest(name: string, noDocker: bool) = - if noDocker: +proc runSingleTest(name: string, options: Options) = + if optNoDocker in options: buildExe(name) else: buildImage(name) - let res = runTest(name, noDocker) + let res = runTest(name, options) var s = newStringStream() res.renderResult(s) echo s.data @@ -287,8 +296,7 @@ proc printHelp() = for c in participants: echo " ", c -proc runCommand(cmd, options: string) = - var noDocker = options == "nodocker" +proc runCommand(cmd: string, options: Options) = case cmd of "clean": killContainers() @@ -298,7 +306,7 @@ proc runCommand(cmd, options: string) = discard installWrk(true) of "all": discard installWrk() - runAllTest(noDocker) + runAllTest(options) of "help": printHelp() else: @@ -306,13 +314,23 @@ proc runCommand(cmd, options: string) = echo cmd & " is not a registered participant" return discard installWrk() - runSingleTest(cmd, noDocker) + runSingleTest(cmd, options) proc main() = if paramCount() > 0: let cmd = paramStr(1) - let options = if paramCount() > 1: - paramStr(2) else: "" + var options: Options = {} + + for i in 2 .. paramCount(): + case paramStr(i) + of "nodocker": + options.incl optNoDocker + of "nothreads": + options.incl optNoThreads + else: + echo "Invalid option: ", paramStr(i) + quit 1 + runCommand(cmd, options) else: printHelp() diff --git a/tests/benchmark/fasthttp/plaintext.dockerfile b/tests/benchmark/fasthttp/plaintext.dockerfile index b87bd2119..b474b10ce 100644 --- a/tests/benchmark/fasthttp/plaintext.dockerfile +++ b/tests/benchmark/fasthttp/plaintext.dockerfile @@ -13,4 +13,5 @@ RUN go get -d -u github.com/valyala/fasthttp/... RUN rm -f ./server RUN go build -gcflags='-l=4' server -CMD GOMAXPROCS=1 ./server +CMD ./server + diff --git a/tests/benchmark/fasthttp/src/server/server.go b/tests/benchmark/fasthttp/src/server/server.go index 94423e00e..b71653bee 100644 --- a/tests/benchmark/fasthttp/src/server/server.go +++ b/tests/benchmark/fasthttp/src/server/server.go @@ -1,7 +1,8 @@ package main import ( - "flag" + "os" + "flag" "log" "github.com/valyala/fasthttp" "runtime" @@ -9,7 +10,10 @@ import ( ) func main() { - runtime.GOMAXPROCS(1) + if os.Getenv("USE_THREADS") == "" { + runtime.GOMAXPROCS(1) + } + flag.Parse() var err error From 4f6cb70bf50483ed4693d4bd1a26d8e717c2563b Mon Sep 17 00:00:00 2001 From: andri lim Date: Mon, 29 Oct 2018 16:31:39 +0700 Subject: [PATCH 12/12] add thread or nothread stuff --- tests/benchmark/asyncdispatch2/build.sh | 7 +++ .../asyncdispatch2/plaintext.dockerfile | 2 +- tests/benchmark/asyncdispatch2/server.nim | 29 +++++++++-- tests/benchmark/asyncnet/build.sh | 6 +++ tests/benchmark/asyncnet/plaintext.dockerfile | 2 +- tests/benchmark/asyncnet/server.nim | 49 +++++++++---------- tests/benchmark/bot.nim | 12 +++-- tests/benchmark/fasthttp/build.sh | 20 ++++++++ tests/benchmark/libreactor/Makefile | 2 +- tests/benchmark/libreactor/build.sh | 40 +++++++++++++++ .../benchmark/libreactor/plaintext.dockerfile | 14 +++--- tests/benchmark/libreactor/plaintext.make | 11 +++++ tests/benchmark/libreactor/src/main.c | 14 ++++-- tests/benchmark/libreactor/src/patch.c | 23 +++++++++ tests/benchmark/mofuw/build.sh | 7 +++ tests/benchmark/mofuw/server.nim | 18 ++++--- 16 files changed, 199 insertions(+), 57 deletions(-) create mode 100644 tests/benchmark/asyncdispatch2/build.sh create mode 100644 tests/benchmark/asyncnet/build.sh create mode 100644 tests/benchmark/fasthttp/build.sh create mode 100644 tests/benchmark/libreactor/build.sh create mode 100644 tests/benchmark/libreactor/plaintext.make create mode 100644 tests/benchmark/libreactor/src/patch.c create mode 100644 tests/benchmark/mofuw/build.sh diff --git a/tests/benchmark/asyncdispatch2/build.sh b/tests/benchmark/asyncdispatch2/build.sh new file mode 100644 index 000000000..aebf58c67 --- /dev/null +++ b/tests/benchmark/asyncdispatch2/build.sh @@ -0,0 +1,7 @@ +#!/bin/bash + +if [ ! -f server ]; then + echo "building asyncdispatch2..." + nimble install -y https://github.com/status-im/nim-asyncdispatch2 > /dev/null + nim c -d:release --verbosity:0 --hints:off --threads:on server.nim +fi diff --git a/tests/benchmark/asyncdispatch2/plaintext.dockerfile b/tests/benchmark/asyncdispatch2/plaintext.dockerfile index 2721b6dad..578c89da2 100644 --- a/tests/benchmark/asyncdispatch2/plaintext.dockerfile +++ b/tests/benchmark/asyncdispatch2/plaintext.dockerfile @@ -2,5 +2,5 @@ FROM statusteam/nim-base RUN nimble install -y https://github.com/status-im/nim-asyncdispatch2 WORKDIR /server COPY server.nim server.nim -RUN nim c -d:release server.nim +RUN nim c -d:release --threads:on server.nim CMD ["./server"] diff --git a/tests/benchmark/asyncdispatch2/server.nim b/tests/benchmark/asyncdispatch2/server.nim index 6576f6a75..7546257c2 100644 --- a/tests/benchmark/asyncdispatch2/server.nim +++ b/tests/benchmark/asyncdispatch2/server.nim @@ -1,5 +1,5 @@ import strutils, posix, deques, times, strformat -import asyncdispatch2 +import asyncdispatch2, threadpool, os, osproc proc getFormattedTime*(): string = result = format(getTime().inZone(utc()), "ddd, dd MMM yyyy hh:mm:ss 'GMT'") @@ -123,11 +123,11 @@ proc updateTime(arg: pointer = nil) {.gcsafe.} = var svr = cast[ServerCtx](arg) svr.updateServerTime() -proc serve(onAddress: string) = +proc incomingConnection(onAddress: string) = let ta = initTAddress(onAddress) ctx = newServerCtx(1024, 1024, 128) - svr = createStreamServer(ta, handleConnection, {ReuseAddr}, backlog = 128, udata = cast[pointer](ctx)) + svr = createStreamServer(ta, handleConnection, {ReuseAddr, ReusePort}, backlog = 128, udata = cast[pointer](ctx)) when not defined(windows): discard addSignal(SIGINT, handleBreak, udata = cast[pointer](svr)) @@ -138,5 +138,24 @@ proc serve(onAddress: string) = echo "Server started at ", ta waitFor svr.join() -when isMainModule: - serve("0.0.0.0:8080") +proc runServer() {.thread.} = + incomingConnection("0.0.0.0:8080") + +proc runSingleThread() = + spawn runServer() + threadpool.sync() + +proc runMultipleThreads() = + for _ in 0 ..< countProcessors(): + spawn runServer() + threadpool.sync() + +proc main() = + if os.getEnv("USE_THREADS") == "1": + echo "use threads" + runMultipleThreads() + else: + echo "no threads" + runSingleThread() + +main() diff --git a/tests/benchmark/asyncnet/build.sh b/tests/benchmark/asyncnet/build.sh new file mode 100644 index 000000000..4f149fb3c --- /dev/null +++ b/tests/benchmark/asyncnet/build.sh @@ -0,0 +1,6 @@ +#!/bin/bash + +if [ ! -f server ]; then + echo "building asyncnet..." + nim c -d:release --verbosity:0 --hints:off --threads:on server.nim +fi diff --git a/tests/benchmark/asyncnet/plaintext.dockerfile b/tests/benchmark/asyncnet/plaintext.dockerfile index b5fbd9776..64271561f 100644 --- a/tests/benchmark/asyncnet/plaintext.dockerfile +++ b/tests/benchmark/asyncnet/plaintext.dockerfile @@ -1,5 +1,5 @@ FROM statusteam/nim-base WORKDIR /server COPY server.nim server.nim -RUN nim c -d:release server.nim +RUN nim c -d:release --threads:on server.nim CMD ["./server"] diff --git a/tests/benchmark/asyncnet/server.nim b/tests/benchmark/asyncnet/server.nim index 02946b3b0..cc32b62c2 100644 --- a/tests/benchmark/asyncnet/server.nim +++ b/tests/benchmark/asyncnet/server.nim @@ -7,7 +7,7 @@ # MIT license (LICENSE-MIT) import asyncnet, asyncdispatch, nativesockets, net, strformat, times -import deques +import deques, threadpool, os, osproc when defined(windows): from winlean import TCP_NODELAY @@ -84,16 +84,12 @@ proc sendMessage(ctx: IncomingCtx, body: string) = const timeOut = 30 * 1000 proc readMessage*(ctx: IncomingCtx, client: AsyncSocket): Future[int] {.async.} = -#proc readMessage*(ctx: IncomingCtx, client: AsyncFD): Future[int] {.async.} = let rcvLimit = block: if unlikely(ctx.buf.len - ctx.bufLen == 0): ctx.buf.setLen(ctx.buf.len + ctx.buf.len) ctx.buf.len - ctx.bufLen - #let fut = client.recvInto(addr ctx.buf[ctx.bufLen], rcvLimit) - #let isSuccess = await withTimeout(fut, timeOut) - #let rcv = if isSuccess: fut.read else: 0 let rcv = await client.recvInto(addr ctx.buf[ctx.bufLen], rcvLimit) ctx.bufLen += rcv return rcv @@ -106,12 +102,10 @@ proc makeResp(serverTime: string): string = "Content-Length: 13\r\L\r\L" & "Hello, World!") -#proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, client: AsyncFD) {.async.} = proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, client: AsyncSocket) {.async.} = while true: let rcv = await ctx.readMessage(client) if rcv == 0: - #closeSocket(client) client.close() srv.freeCtx(ctx) return @@ -128,28 +122,16 @@ proc handleIncoming(srv: ServerCtx, ctx: IncomingCtx, client: AsyncSocket) {.asy let fut = client.send(ctx.resp[0].addr, ctx.respLen) yield fut if fut.failed: - #closeSocket(client) client.close() srv.freeCtx(ctx) return ctx.resetBuffer() -#proc handleConnection(srv: ServerCtx, client: AsyncFD) {.async.} = proc handleConnection(srv: ServerCtx, client: AsyncSocket) {.async.} = var ctx = srv.getIncomingCtx(1024, 1024) ctx.resetBuffer() asyncCheck handleIncoming(srv, ctx, client) -#proc newServerSocket*(port: int): SocketHandle = -# let server = newSocket() -# server.setSockOpt(OptReuseAddr, true) -# server.setSockOpt(OptReusePort, true) -# server.getFd().setSockOptInt(cint(IPPROTO_TCP), TCP_NODELAY, 1) -# server.getFd.setBlocking(false) -# server.bindAddr(Port(port)) -# server.listen() -# return server.getFd() - proc newServerSocket(port: int): AsyncSocket = var server = newAsyncSocket(buffered=false) server.setSockOpt(OptReuseAddr, true) @@ -160,13 +142,11 @@ proc newServerSocket(port: int): AsyncSocket = server.listen() result = server -proc serve() {.async.} = +proc incomingConnection() {.async.} = let - #server = newServerSocket(8080).AsyncFD server = newServerSocket(8080) ctx = newServerCtx(1024, 1024, 128) - #register(server) proc updateTime(fd: AsyncFD): bool = ctx.updateServerTime() @@ -179,13 +159,30 @@ proc serve() {.async.} = cantAccept = true try: - #let data = await acceptAddr(server) - #asyncCheck handleConnection(ctx, data.client) let client = await server.accept() asyncCheck handleConnection(ctx, client) except: cantAccept = true -when isMainModule: +proc runServer() {.thread.} = + waitFor incomingConnection() + +proc runSingleThread() = + spawn runServer() + sync() + +proc runMultipleThreads() = + for _ in 0 ..< countProcessors(): + spawn runServer() + sync() + +proc main() = setControlCHook(handleCtrlC) - waitFor serve() + if os.getEnv("USE_THREADS") == "1": + echo "use threads" + runMultipleThreads() + else: + echo "no threads" + runSingleThread() + +main() diff --git a/tests/benchmark/bot.nim b/tests/benchmark/bot.nim index c6eff8e5d..a98e439ea 100644 --- a/tests/benchmark/bot.nim +++ b/tests/benchmark/bot.nim @@ -1,4 +1,4 @@ -import osproc, json, streams, strutils, os +import osproc, json, streams, strutils, os, strtabs type Option = enum @@ -82,11 +82,15 @@ proc buildExes() = for c in participants: buildExe(c) -proc runServer(id: string): Process = +proc runServer(id: string, opts: Options): Process = const options = {poParentStreams} let parentDir = getAppDir() let name = parentDir & DirSep & id & DirSep & "server" - result = startProcess(name, args=[], env=nil, options=options) + var env: StringTableRef + if optNoThreads notin opts: + env = newStringTable(modeCaseSensitive) + env["USE_THREADS"] = "1" + result = startProcess(name, args=[], env=env, options=options) if not result.running(): raise newException(Exception, "cannot run server: " & id) @@ -118,7 +122,7 @@ proc runTest(name: string, options: Options): JsonNode = echo "** ", name, " **" if optNoDocker in options: - server = runServer(name) + server = runServer(name, options) else: let useTheadsFlag = if optNoThreads in options: "" else: "-e USE_THREADS=1" diff --git a/tests/benchmark/fasthttp/build.sh b/tests/benchmark/fasthttp/build.sh new file mode 100644 index 000000000..08b648eee --- /dev/null +++ b/tests/benchmark/fasthttp/build.sh @@ -0,0 +1,20 @@ +#!/bin/bash +echo $PWD +INSDIR=$PWD/install +GOPATH=$PWD +GOROOT=$INSDIR/go + +if [ ! -f server ]; then + echo "building fasthttp..." + mkdir $INSDIR + if [ ! -f go1.11.linux-amd64.tar.gz ]; then + wget -q https://dl.google.com/go/go1.11.linux-amd64.tar.gz + fi + tar -C $INSDIR -xzf go1.11.linux-amd64.tar.gz + export PATH=$PATH:$GOROOT/bin + export GOPATH=$GOPATH + export GOROOT=$GOROOT + echo $GOROOT + go get -d -u github.com/valyala/fasthttp/... + go build -gcflags='-l=4' server +fi diff --git a/tests/benchmark/libreactor/Makefile b/tests/benchmark/libreactor/Makefile index f1e3f2dd7..97e342bfc 100644 --- a/tests/benchmark/libreactor/Makefile +++ b/tests/benchmark/libreactor/Makefile @@ -1,4 +1,4 @@ -CC = gcc-6 +CC = gcc-7 PROG = server OBJS = src/setup.o src/main.o CFLAGS = -std=gnu11 -Wall -O3 -march=native -mtune=native -flto -fuse-linker-plugin -Isrc diff --git a/tests/benchmark/libreactor/build.sh b/tests/benchmark/libreactor/build.sh new file mode 100644 index 000000000..8bfc145f0 --- /dev/null +++ b/tests/benchmark/libreactor/build.sh @@ -0,0 +1,40 @@ +#!/bin/bash + +if [ ! -f server ]; then + echo "building libreactor..." + + INSDIR=$PWD/install + mkdir $INSDIR + + INCLUDEDIR=$INSDIR/include + LIBDIR=$INSDIR/lib + + wget -q https://github.com/fredrikwidlund/libdynamic/releases/download/v1.1.0/libdynamic-1.1.0.tar.gz + tar xfz libdynamic-1.1.0.tar.gz + cd libdynamic-1.1.0 + ./configure --prefix=$INSDIR + make + make install + cd .. + + wget -q https://github.com/fredrikwidlund/libreactor/releases/download/v1.0.0/libreactor-1.0.0.tar.gz + tar xfz libreactor-1.0.0.tar.gz + cd libreactor-1.0.0 + cat ../src/patch.c >> src/reactor/reactor_core.c + ./configure --prefix=$INSDIR CFLAGS="-I"$INCLUDEDIR LDFLAGS="-L"$LIBDIR + make + make install + cd .. + + wget -q https://github.com/fredrikwidlund/libclo/releases/download/v0.1.0/libclo-0.1.0.tar.gz + tar xfz libclo-0.1.0.tar.gz + cd libclo-0.1.0 + ./configure --prefix=$INSDIR CFLAGS="-I"$INCLUDEDIR LDFLAGS="-L"$LIBDIR + make + make install + cd .. + + make clean + LIBDIR=$LIBDIR INCLUDEDIR=$INCLUDEDIR make -f plaintext.make + +fi diff --git a/tests/benchmark/libreactor/plaintext.dockerfile b/tests/benchmark/libreactor/plaintext.dockerfile index 7332ee459..3f18a56eb 100644 --- a/tests/benchmark/libreactor/plaintext.dockerfile +++ b/tests/benchmark/libreactor/plaintext.dockerfile @@ -5,7 +5,7 @@ RUN apt install -yqq software-properties-common python-software-properties wget RUN add-apt-repository ppa:ubuntu-toolchain-r/test -y && \ apt update -yqq && \ - apt install -yqq gcc-6 g++-6 + apt install -yqq gcc-7 g++-7 ADD ./ /libreactor WORKDIR /libreactor @@ -13,20 +13,20 @@ WORKDIR /libreactor RUN wget -q https://github.com/fredrikwidlund/libdynamic/releases/download/v1.1.0/libdynamic-1.1.0.tar.gz && \ tar xfz libdynamic-1.1.0.tar.gz && \ cd libdynamic-1.1.0 && \ - ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + ./configure CC=gcc-7 AR=gcc-ar-7 NM=gcc-nm-7 RANLIB=gcc-ranlib-7 && \ make && make install - + RUN wget -q https://github.com/fredrikwidlund/libreactor/releases/download/v1.0.0/libreactor-1.0.0.tar.gz && \ tar xfz libreactor-1.0.0.tar.gz && \ - cd libreactor-1.0.0 && \ - sed -i -e 's,REACTOR_POOL_WORKERS_MAX,1,' src/reactor/reactor_pool.c && \ - ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + cd libreactor-1.0.0 && \ + cat ../src/patch.c >> src/reactor/reactor_core.c && \ + ./configure CC=gcc-7 AR=gcc-ar-7 NM=gcc-nm-7 RANLIB=gcc-ranlib-7 && \ make && make install RUN wget -q https://github.com/fredrikwidlund/libclo/releases/download/v0.1.0/libclo-0.1.0.tar.gz && \ tar xfz libclo-0.1.0.tar.gz && \ cd libclo-0.1.0 && \ - ./configure CC=gcc-6 AR=gcc-ar-6 NM=gcc-nm-6 RANLIB=gcc-ranlib-6 && \ + ./configure CC=gcc-7 AR=gcc-ar-7 NM=gcc-nm-7 RANLIB=gcc-ranlib-7 && \ make && make install RUN make clean && make diff --git a/tests/benchmark/libreactor/plaintext.make b/tests/benchmark/libreactor/plaintext.make new file mode 100644 index 000000000..5aeb0ba15 --- /dev/null +++ b/tests/benchmark/libreactor/plaintext.make @@ -0,0 +1,11 @@ +CC = gcc +PROG = server +OBJS = src/setup.o src/main.o +CFLAGS = -std=gnu11 -Wall -O3 -march=native -mtune=native -flto -fuse-linker-plugin -Isrc -I$(INCLUDEDIR) +LDADD = -lreactor -ldynamic -lclo -L$(LIBDIR) + +$(PROG): $(OBJS) + $(CC) -o $@ $^ $(CFLAGS) $(LDADD) + +clean: + rm -f $(PROG) $(OBJS) diff --git a/tests/benchmark/libreactor/src/main.c b/tests/benchmark/libreactor/src/main.c index 05d97bcf0..4daff3981 100644 --- a/tests/benchmark/libreactor/src/main.c +++ b/tests/benchmark/libreactor/src/main.c @@ -61,7 +61,7 @@ static int json(reactor_http *http) { char body[4096], content_length[11]; size_t size; - + (void) clo_encode((clo[]) {clo_object({"message", clo_string("Hello, World!")})}, body, sizeof(body)); size = strlen(body); reactor_util_u32toa(size, content_length); @@ -91,7 +91,7 @@ int http_event(void *state, int type, void *data) return REACTOR_ABORT; } request = data; - + if (reactor_http_header_value(&request->path, "/plaintext")) return plaintext(http); else if (reactor_http_header_value(&request->path, "/json")) @@ -126,13 +126,19 @@ static int tcp_event(void *state, int type, void *data) return REACTOR_OK; } +int reactor_core_construct2(int num_threads); + int main() { reactor_timer timer; reactor_tcp tcp; - setup(1, 0); - (void) reactor_core_construct(); + setup(1, 0); + if(getenv("USE_THREADS") != NULL) { + (void) reactor_core_construct(); + } else { + (void) reactor_core_construct2(1); + } (void) reactor_timer_open(&timer, timer_event, &timer, 1, 1000000000); (void) reactor_tcp_open(&tcp, tcp_event, &tcp, "0.0.0.0", "8080", REACTOR_TCP_FLAG_SERVER); (void) reactor_core_run(); diff --git a/tests/benchmark/libreactor/src/patch.c b/tests/benchmark/libreactor/src/patch.c new file mode 100644 index 000000000..a42459f43 --- /dev/null +++ b/tests/benchmark/libreactor/src/patch.c @@ -0,0 +1,23 @@ + +int reactor_core_construct2(int num_threads) +{ + int e; + + core = (struct reactor_core) {0}; + core.fd = epoll_create1(EPOLL_CLOEXEC); + if (core.fd == -1) + return REACTOR_ERROR; + + e = reactor_pool_construct(&core.pool); + if (e == -1) + { + (void) close(core.fd); + core.fd = -1; + return REACTOR_ERROR; + } + + reactor_pool_limits(&core.pool, 1, num_threads); + return REACTOR_OK; +} + +/* cat ../src/patch.c >> src/reactor/reactor_core.c */ diff --git a/tests/benchmark/mofuw/build.sh b/tests/benchmark/mofuw/build.sh new file mode 100644 index 000000000..b768d0c83 --- /dev/null +++ b/tests/benchmark/mofuw/build.sh @@ -0,0 +1,7 @@ +#!/bin/bash + +if [ ! -f server ]; then + echo "building mofuw..." + nimble install -y mofuw packedjson > /dev/null + nim c -d:release --threads:on -d:bufSize:512 --verbosity:0 --hints:off server.nim +fi diff --git a/tests/benchmark/mofuw/server.nim b/tests/benchmark/mofuw/server.nim index 3a726ca1c..57128d048 100644 --- a/tests/benchmark/mofuw/server.nim +++ b/tests/benchmark/mofuw/server.nim @@ -1,4 +1,4 @@ -import mofuw, packedjson, threadpool +import mofuw, packedjson, threadpool, os proc h(ctx: MofuwCtx) {.async.} = case ctx.getPath @@ -19,11 +19,13 @@ proc serveSingleThread*(ctx: ServeCtx) = when not defined noSync: sync() -newServeCtx( - port = 8080, - handler = h -).serveSingleThread() - -# use this if you want multithread mode -#).serve() +proc main() = + var srv = newServeCtx(port = 8080, handler = h) + if os.getEnv("USE_THREADS") == "1": + echo "use threads" + srv.serve() + else: + echo "no threads" + srv.serveSingleThread() +main()