api_test.go 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313
  1. package docker
  2. import (
  3. "bufio"
  4. "bytes"
  5. "encoding/json"
  6. "fmt"
  7. "io"
  8. "io/ioutil"
  9. "net"
  10. "net/http"
  11. "net/http/httptest"
  12. "strings"
  13. "testing"
  14. "time"
  15. "github.com/dotcloud/docker/api"
  16. "github.com/dotcloud/docker/api/server"
  17. "github.com/dotcloud/docker/engine"
  18. "github.com/dotcloud/docker/image"
  19. "github.com/dotcloud/docker/runconfig"
  20. "github.com/dotcloud/docker/runtime"
  21. "github.com/dotcloud/docker/utils"
  22. "github.com/dotcloud/docker/vendor/src/code.google.com/p/go/src/pkg/archive/tar"
  23. )
  24. func TestGetInfo(t *testing.T) {
  25. eng := NewTestEngine(t)
  26. defer mkRuntimeFromEngine(eng, t).Nuke()
  27. job := eng.Job("images")
  28. initialImages, err := job.Stdout.AddListTable()
  29. if err != nil {
  30. t.Fatal(err)
  31. }
  32. if err := job.Run(); err != nil {
  33. t.Fatal(err)
  34. }
  35. req, err := http.NewRequest("GET", "/info", nil)
  36. if err != nil {
  37. t.Fatal(err)
  38. }
  39. r := httptest.NewRecorder()
  40. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  41. t.Fatal(err)
  42. }
  43. assertHttpNotError(r, t)
  44. out := engine.NewOutput()
  45. i, err := out.AddEnv()
  46. if err != nil {
  47. t.Fatal(err)
  48. }
  49. if _, err := io.Copy(out, r.Body); err != nil {
  50. t.Fatal(err)
  51. }
  52. out.Close()
  53. if images := i.GetInt("Images"); images != initialImages.Len() {
  54. t.Errorf("Expected images: %d, %d found", initialImages.Len(), images)
  55. }
  56. expected := "application/json"
  57. if result := r.HeaderMap.Get("Content-Type"); result != expected {
  58. t.Errorf("Expected Content-Type %s, %s found", expected, result)
  59. }
  60. }
  61. func TestGetEvents(t *testing.T) {
  62. eng := NewTestEngine(t)
  63. srv := mkServerFromEngine(eng, t)
  64. // FIXME: we might not need runtime, why not simply nuke
  65. // the engine?
  66. runtime := mkRuntimeFromEngine(eng, t)
  67. defer nuke(runtime)
  68. var events []*utils.JSONMessage
  69. for _, parts := range [][3]string{
  70. {"fakeaction", "fakeid", "fakeimage"},
  71. {"fakeaction2", "fakeid", "fakeimage"},
  72. } {
  73. action, id, from := parts[0], parts[1], parts[2]
  74. ev := srv.LogEvent(action, id, from)
  75. events = append(events, ev)
  76. }
  77. req, err := http.NewRequest("GET", "/events?since=1", nil)
  78. if err != nil {
  79. t.Fatal(err)
  80. }
  81. r := httptest.NewRecorder()
  82. setTimeout(t, "", 500*time.Millisecond, func() {
  83. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  84. t.Fatal(err)
  85. }
  86. assertHttpNotError(r, t)
  87. })
  88. dec := json.NewDecoder(r.Body)
  89. for i := 0; i < 2; i++ {
  90. var jm utils.JSONMessage
  91. if err := dec.Decode(&jm); err == io.EOF {
  92. break
  93. } else if err != nil {
  94. t.Fatal(err)
  95. }
  96. if jm != *events[i] {
  97. t.Fatalf("Event received it different than expected")
  98. }
  99. }
  100. }
  101. func TestGetImagesJSON(t *testing.T) {
  102. eng := NewTestEngine(t)
  103. defer mkRuntimeFromEngine(eng, t).Nuke()
  104. job := eng.Job("images")
  105. initialImages, err := job.Stdout.AddListTable()
  106. if err != nil {
  107. t.Fatal(err)
  108. }
  109. if err := job.Run(); err != nil {
  110. t.Fatal(err)
  111. }
  112. req, err := http.NewRequest("GET", "/images/json?all=0", nil)
  113. if err != nil {
  114. t.Fatal(err)
  115. }
  116. r := httptest.NewRecorder()
  117. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  118. t.Fatal(err)
  119. }
  120. assertHttpNotError(r, t)
  121. images := engine.NewTable("Created", 0)
  122. if _, err := images.ReadListFrom(r.Body.Bytes()); err != nil {
  123. t.Fatal(err)
  124. }
  125. if images.Len() != initialImages.Len() {
  126. t.Errorf("Expected %d image, %d found", initialImages.Len(), images.Len())
  127. }
  128. found := false
  129. for _, img := range images.Data {
  130. if strings.Contains(img.GetList("RepoTags")[0], unitTestImageName) {
  131. found = true
  132. break
  133. }
  134. }
  135. if !found {
  136. t.Errorf("Expected image %s, %+v found", unitTestImageName, images)
  137. }
  138. r2 := httptest.NewRecorder()
  139. // all=1
  140. initialImages = getAllImages(eng, t)
  141. req2, err := http.NewRequest("GET", "/images/json?all=true", nil)
  142. if err != nil {
  143. t.Fatal(err)
  144. }
  145. if err := server.ServeRequest(eng, api.APIVERSION, r2, req2); err != nil {
  146. t.Fatal(err)
  147. }
  148. assertHttpNotError(r2, t)
  149. images2 := engine.NewTable("Id", 0)
  150. if _, err := images2.ReadListFrom(r2.Body.Bytes()); err != nil {
  151. t.Fatal(err)
  152. }
  153. if images2.Len() != initialImages.Len() {
  154. t.Errorf("Expected %d image, %d found", initialImages.Len(), images2.Len())
  155. }
  156. found = false
  157. for _, img := range images2.Data {
  158. if img.Get("Id") == unitTestImageID {
  159. found = true
  160. break
  161. }
  162. }
  163. if !found {
  164. t.Errorf("Retrieved image Id differs, expected %s, received %+v", unitTestImageID, images2)
  165. }
  166. r3 := httptest.NewRecorder()
  167. // filter=a
  168. req3, err := http.NewRequest("GET", "/images/json?filter=aaaaaaaaaa", nil)
  169. if err != nil {
  170. t.Fatal(err)
  171. }
  172. if err := server.ServeRequest(eng, api.APIVERSION, r3, req3); err != nil {
  173. t.Fatal(err)
  174. }
  175. assertHttpNotError(r3, t)
  176. images3 := engine.NewTable("Id", 0)
  177. if _, err := images3.ReadListFrom(r3.Body.Bytes()); err != nil {
  178. t.Fatal(err)
  179. }
  180. if images3.Len() != 0 {
  181. t.Errorf("Expected 0 image, %d found", images3.Len())
  182. }
  183. }
  184. func TestGetImagesHistory(t *testing.T) {
  185. eng := NewTestEngine(t)
  186. defer mkRuntimeFromEngine(eng, t).Nuke()
  187. r := httptest.NewRecorder()
  188. req, err := http.NewRequest("GET", fmt.Sprintf("/images/%s/history", unitTestImageName), nil)
  189. if err != nil {
  190. t.Fatal(err)
  191. }
  192. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  193. t.Fatal(err)
  194. }
  195. assertHttpNotError(r, t)
  196. outs := engine.NewTable("Created", 0)
  197. if _, err := outs.ReadListFrom(r.Body.Bytes()); err != nil {
  198. t.Fatal(err)
  199. }
  200. if len(outs.Data) != 1 {
  201. t.Errorf("Expected 1 line, %d found", len(outs.Data))
  202. }
  203. }
  204. func TestGetImagesByName(t *testing.T) {
  205. eng := NewTestEngine(t)
  206. defer mkRuntimeFromEngine(eng, t).Nuke()
  207. req, err := http.NewRequest("GET", "/images/"+unitTestImageName+"/json", nil)
  208. if err != nil {
  209. t.Fatal(err)
  210. }
  211. r := httptest.NewRecorder()
  212. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  213. t.Fatal(err)
  214. }
  215. assertHttpNotError(r, t)
  216. img := &image.Image{}
  217. if err := json.Unmarshal(r.Body.Bytes(), img); err != nil {
  218. t.Fatal(err)
  219. }
  220. if img.ID != unitTestImageID {
  221. t.Errorf("Error inspecting image")
  222. }
  223. }
  224. func TestGetContainersJSON(t *testing.T) {
  225. eng := NewTestEngine(t)
  226. defer mkRuntimeFromEngine(eng, t).Nuke()
  227. job := eng.Job("containers")
  228. job.SetenvBool("all", true)
  229. outs, err := job.Stdout.AddTable()
  230. if err != nil {
  231. t.Fatal(err)
  232. }
  233. if err := job.Run(); err != nil {
  234. t.Fatal(err)
  235. }
  236. beginLen := len(outs.Data)
  237. containerID := createTestContainer(eng, &runconfig.Config{
  238. Image: unitTestImageID,
  239. Cmd: []string{"echo", "test"},
  240. }, t)
  241. if containerID == "" {
  242. t.Fatalf("Received empty container ID")
  243. }
  244. req, err := http.NewRequest("GET", "/containers/json?all=1", nil)
  245. if err != nil {
  246. t.Fatal(err)
  247. }
  248. r := httptest.NewRecorder()
  249. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  250. t.Fatal(err)
  251. }
  252. assertHttpNotError(r, t)
  253. containers := engine.NewTable("", 0)
  254. if _, err := containers.ReadListFrom(r.Body.Bytes()); err != nil {
  255. t.Fatal(err)
  256. }
  257. if len(containers.Data) != beginLen+1 {
  258. t.Fatalf("Expected %d container, %d found (started with: %d)", beginLen+1, len(containers.Data), beginLen)
  259. }
  260. if id := containers.Data[0].Get("Id"); id != containerID {
  261. t.Fatalf("Container ID mismatch. Expected: %s, received: %s\n", containerID, id)
  262. }
  263. }
  264. func TestGetContainersExport(t *testing.T) {
  265. eng := NewTestEngine(t)
  266. defer mkRuntimeFromEngine(eng, t).Nuke()
  267. // Create a container and remove a file
  268. containerID := createTestContainer(eng,
  269. &runconfig.Config{
  270. Image: unitTestImageID,
  271. Cmd: []string{"touch", "/test"},
  272. },
  273. t,
  274. )
  275. containerRun(eng, containerID, t)
  276. r := httptest.NewRecorder()
  277. req, err := http.NewRequest("GET", "/containers/"+containerID+"/export", nil)
  278. if err != nil {
  279. t.Fatal(err)
  280. }
  281. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  282. t.Fatal(err)
  283. }
  284. assertHttpNotError(r, t)
  285. if r.Code != http.StatusOK {
  286. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  287. }
  288. found := false
  289. for tarReader := tar.NewReader(r.Body); ; {
  290. h, err := tarReader.Next()
  291. if err != nil {
  292. if err == io.EOF {
  293. break
  294. }
  295. t.Fatal(err)
  296. }
  297. if h.Name == "test" {
  298. found = true
  299. break
  300. }
  301. }
  302. if !found {
  303. t.Fatalf("The created test file has not been found in the exported image")
  304. }
  305. }
  306. func TestSaveImageAndThenLoad(t *testing.T) {
  307. eng := NewTestEngine(t)
  308. defer mkRuntimeFromEngine(eng, t).Nuke()
  309. // save image
  310. r := httptest.NewRecorder()
  311. req, err := http.NewRequest("GET", "/images/"+unitTestImageID+"/get", nil)
  312. if err != nil {
  313. t.Fatal(err)
  314. }
  315. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  316. t.Fatal(err)
  317. }
  318. if r.Code != http.StatusOK {
  319. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  320. }
  321. tarball := r.Body
  322. // delete the image
  323. r = httptest.NewRecorder()
  324. req, err = http.NewRequest("DELETE", "/images/"+unitTestImageID, nil)
  325. if err != nil {
  326. t.Fatal(err)
  327. }
  328. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  329. t.Fatal(err)
  330. }
  331. if r.Code != http.StatusOK {
  332. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  333. }
  334. // make sure there is no image
  335. r = httptest.NewRecorder()
  336. req, err = http.NewRequest("GET", "/images/"+unitTestImageID+"/get", nil)
  337. if err != nil {
  338. t.Fatal(err)
  339. }
  340. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  341. t.Fatal(err)
  342. }
  343. if r.Code != http.StatusNotFound {
  344. t.Fatalf("%d NotFound expected, received %d\n", http.StatusNotFound, r.Code)
  345. }
  346. // load the image
  347. r = httptest.NewRecorder()
  348. req, err = http.NewRequest("POST", "/images/load", tarball)
  349. if err != nil {
  350. t.Fatal(err)
  351. }
  352. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  353. t.Fatal(err)
  354. }
  355. if r.Code != http.StatusOK {
  356. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  357. }
  358. // finally make sure the image is there
  359. r = httptest.NewRecorder()
  360. req, err = http.NewRequest("GET", "/images/"+unitTestImageID+"/get", nil)
  361. if err != nil {
  362. t.Fatal(err)
  363. }
  364. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  365. t.Fatal(err)
  366. }
  367. if r.Code != http.StatusOK {
  368. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  369. }
  370. }
  371. func TestGetContainersChanges(t *testing.T) {
  372. eng := NewTestEngine(t)
  373. defer mkRuntimeFromEngine(eng, t).Nuke()
  374. // Create a container and remove a file
  375. containerID := createTestContainer(eng,
  376. &runconfig.Config{
  377. Image: unitTestImageID,
  378. Cmd: []string{"/bin/rm", "/etc/passwd"},
  379. },
  380. t,
  381. )
  382. containerRun(eng, containerID, t)
  383. r := httptest.NewRecorder()
  384. req, err := http.NewRequest("GET", "/containers/"+containerID+"/changes", nil)
  385. if err != nil {
  386. t.Fatal(err)
  387. }
  388. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  389. t.Fatal(err)
  390. }
  391. assertHttpNotError(r, t)
  392. outs := engine.NewTable("", 0)
  393. if _, err := outs.ReadListFrom(r.Body.Bytes()); err != nil {
  394. t.Fatal(err)
  395. }
  396. // Check the changelog
  397. success := false
  398. for _, elem := range outs.Data {
  399. if elem.Get("Path") == "/etc/passwd" && elem.GetInt("Kind") == 2 {
  400. success = true
  401. }
  402. }
  403. if !success {
  404. t.Fatalf("/etc/passwd as been removed but is not present in the diff")
  405. }
  406. }
  407. func TestGetContainersTop(t *testing.T) {
  408. eng := NewTestEngine(t)
  409. defer mkRuntimeFromEngine(eng, t).Nuke()
  410. containerID := createTestContainer(eng,
  411. &runconfig.Config{
  412. Image: unitTestImageID,
  413. Cmd: []string{"/bin/sh", "-c", "cat"},
  414. OpenStdin: true,
  415. },
  416. t,
  417. )
  418. defer func() {
  419. // Make sure the process dies before destroying runtime
  420. containerKill(eng, containerID, t)
  421. containerWait(eng, containerID, t)
  422. }()
  423. startContainer(eng, containerID, t)
  424. setTimeout(t, "Waiting for the container to be started timed out", 10*time.Second, func() {
  425. for {
  426. if containerRunning(eng, containerID, t) {
  427. break
  428. }
  429. time.Sleep(10 * time.Millisecond)
  430. }
  431. })
  432. if !containerRunning(eng, containerID, t) {
  433. t.Fatalf("Container should be running")
  434. }
  435. // Make sure sh spawn up cat
  436. setTimeout(t, "read/write assertion timed out", 2*time.Second, func() {
  437. in, out := containerAttach(eng, containerID, t)
  438. if err := assertPipe("hello\n", "hello", out, in, 150); err != nil {
  439. t.Fatal(err)
  440. }
  441. })
  442. r := httptest.NewRecorder()
  443. req, err := http.NewRequest("GET", "/containers/"+containerID+"/top?ps_args=aux", nil)
  444. if err != nil {
  445. t.Fatal(err)
  446. }
  447. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  448. t.Fatal(err)
  449. }
  450. assertHttpNotError(r, t)
  451. var procs engine.Env
  452. if err := procs.Decode(r.Body); err != nil {
  453. t.Fatal(err)
  454. }
  455. if len(procs.GetList("Titles")) != 11 {
  456. t.Fatalf("Expected 11 titles, found %d.", len(procs.GetList("Titles")))
  457. }
  458. if procs.GetList("Titles")[0] != "USER" || procs.GetList("Titles")[10] != "COMMAND" {
  459. t.Fatalf("Expected Titles[0] to be USER and Titles[10] to be COMMAND, found %s and %s.", procs.GetList("Titles")[0], procs.GetList("Titles")[10])
  460. }
  461. processes := [][]string{}
  462. if err := procs.GetJson("Processes", &processes); err != nil {
  463. t.Fatal(err)
  464. }
  465. if len(processes) != 2 {
  466. t.Fatalf("Expected 2 processes, found %d.", len(processes))
  467. }
  468. if processes[0][10] != "/bin/sh -c cat" {
  469. t.Fatalf("Expected `/bin/sh -c cat`, found %s.", processes[0][10])
  470. }
  471. if processes[1][10] != "/bin/sh -c cat" {
  472. t.Fatalf("Expected `/bin/sh -c cat`, found %s.", processes[1][10])
  473. }
  474. }
  475. func TestGetContainersByName(t *testing.T) {
  476. eng := NewTestEngine(t)
  477. defer mkRuntimeFromEngine(eng, t).Nuke()
  478. // Create a container and remove a file
  479. containerID := createTestContainer(eng,
  480. &runconfig.Config{
  481. Image: unitTestImageID,
  482. Cmd: []string{"echo", "test"},
  483. },
  484. t,
  485. )
  486. r := httptest.NewRecorder()
  487. req, err := http.NewRequest("GET", "/containers/"+containerID+"/json", nil)
  488. if err != nil {
  489. t.Fatal(err)
  490. }
  491. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  492. t.Fatal(err)
  493. }
  494. assertHttpNotError(r, t)
  495. outContainer := &runtime.Container{}
  496. if err := json.Unmarshal(r.Body.Bytes(), outContainer); err != nil {
  497. t.Fatal(err)
  498. }
  499. if outContainer.ID != containerID {
  500. t.Fatalf("Wrong containers retrieved. Expected %s, received %s", containerID, outContainer.ID)
  501. }
  502. }
  503. func TestPostCommit(t *testing.T) {
  504. eng := NewTestEngine(t)
  505. defer mkRuntimeFromEngine(eng, t).Nuke()
  506. srv := mkServerFromEngine(eng, t)
  507. // Create a container and remove a file
  508. containerID := createTestContainer(eng,
  509. &runconfig.Config{
  510. Image: unitTestImageID,
  511. Cmd: []string{"touch", "/test"},
  512. },
  513. t,
  514. )
  515. containerRun(eng, containerID, t)
  516. req, err := http.NewRequest("POST", "/commit?repo=testrepo&testtag=tag&container="+containerID, bytes.NewReader([]byte{}))
  517. if err != nil {
  518. t.Fatal(err)
  519. }
  520. r := httptest.NewRecorder()
  521. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  522. t.Fatal(err)
  523. }
  524. assertHttpNotError(r, t)
  525. if r.Code != http.StatusCreated {
  526. t.Fatalf("%d Created expected, received %d\n", http.StatusCreated, r.Code)
  527. }
  528. var env engine.Env
  529. if err := env.Decode(r.Body); err != nil {
  530. t.Fatal(err)
  531. }
  532. if _, err := srv.ImageInspect(env.Get("Id")); err != nil {
  533. t.Fatalf("The image has not been committed")
  534. }
  535. }
  536. func TestPostContainersCreate(t *testing.T) {
  537. eng := NewTestEngine(t)
  538. defer mkRuntimeFromEngine(eng, t).Nuke()
  539. configJSON, err := json.Marshal(&runconfig.Config{
  540. Image: unitTestImageID,
  541. Memory: 33554432,
  542. Cmd: []string{"touch", "/test"},
  543. })
  544. if err != nil {
  545. t.Fatal(err)
  546. }
  547. req, err := http.NewRequest("POST", "/containers/create", bytes.NewReader(configJSON))
  548. if err != nil {
  549. t.Fatal(err)
  550. }
  551. r := httptest.NewRecorder()
  552. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  553. t.Fatal(err)
  554. }
  555. assertHttpNotError(r, t)
  556. if r.Code != http.StatusCreated {
  557. t.Fatalf("%d Created expected, received %d\n", http.StatusCreated, r.Code)
  558. }
  559. var apiRun engine.Env
  560. if err := apiRun.Decode(r.Body); err != nil {
  561. t.Fatal(err)
  562. }
  563. containerID := apiRun.Get("Id")
  564. containerAssertExists(eng, containerID, t)
  565. containerRun(eng, containerID, t)
  566. if !containerFileExists(eng, containerID, "test", t) {
  567. t.Fatal("Test file was not created")
  568. }
  569. }
  570. func TestPostContainersKill(t *testing.T) {
  571. eng := NewTestEngine(t)
  572. defer mkRuntimeFromEngine(eng, t).Nuke()
  573. containerID := createTestContainer(eng,
  574. &runconfig.Config{
  575. Image: unitTestImageID,
  576. Cmd: []string{"/bin/cat"},
  577. OpenStdin: true,
  578. },
  579. t,
  580. )
  581. startContainer(eng, containerID, t)
  582. // Give some time to the process to start
  583. containerWaitTimeout(eng, containerID, t)
  584. if !containerRunning(eng, containerID, t) {
  585. t.Errorf("Container should be running")
  586. }
  587. r := httptest.NewRecorder()
  588. req, err := http.NewRequest("POST", "/containers/"+containerID+"/kill", bytes.NewReader([]byte{}))
  589. if err != nil {
  590. t.Fatal(err)
  591. }
  592. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  593. t.Fatal(err)
  594. }
  595. assertHttpNotError(r, t)
  596. if r.Code != http.StatusNoContent {
  597. t.Fatalf("%d NO CONTENT expected, received %d\n", http.StatusNoContent, r.Code)
  598. }
  599. if containerRunning(eng, containerID, t) {
  600. t.Fatalf("The container hasn't been killed")
  601. }
  602. }
  603. func TestPostContainersRestart(t *testing.T) {
  604. eng := NewTestEngine(t)
  605. defer mkRuntimeFromEngine(eng, t).Nuke()
  606. containerID := createTestContainer(eng,
  607. &runconfig.Config{
  608. Image: unitTestImageID,
  609. Cmd: []string{"/bin/top"},
  610. OpenStdin: true,
  611. },
  612. t,
  613. )
  614. startContainer(eng, containerID, t)
  615. // Give some time to the process to start
  616. containerWaitTimeout(eng, containerID, t)
  617. if !containerRunning(eng, containerID, t) {
  618. t.Errorf("Container should be running")
  619. }
  620. req, err := http.NewRequest("POST", "/containers/"+containerID+"/restart?t=1", bytes.NewReader([]byte{}))
  621. if err != nil {
  622. t.Fatal(err)
  623. }
  624. r := httptest.NewRecorder()
  625. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  626. t.Fatal(err)
  627. }
  628. assertHttpNotError(r, t)
  629. if r.Code != http.StatusNoContent {
  630. t.Fatalf("%d NO CONTENT expected, received %d\n", http.StatusNoContent, r.Code)
  631. }
  632. // Give some time to the process to restart
  633. containerWaitTimeout(eng, containerID, t)
  634. if !containerRunning(eng, containerID, t) {
  635. t.Fatalf("Container should be running")
  636. }
  637. containerKill(eng, containerID, t)
  638. }
  639. func TestPostContainersStart(t *testing.T) {
  640. eng := NewTestEngine(t)
  641. defer mkRuntimeFromEngine(eng, t).Nuke()
  642. containerID := createTestContainer(
  643. eng,
  644. &runconfig.Config{
  645. Image: unitTestImageID,
  646. Cmd: []string{"/bin/cat"},
  647. OpenStdin: true,
  648. },
  649. t,
  650. )
  651. hostConfigJSON, err := json.Marshal(&runconfig.HostConfig{})
  652. req, err := http.NewRequest("POST", "/containers/"+containerID+"/start", bytes.NewReader(hostConfigJSON))
  653. if err != nil {
  654. t.Fatal(err)
  655. }
  656. req.Header.Set("Content-Type", "application/json")
  657. r := httptest.NewRecorder()
  658. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  659. t.Fatal(err)
  660. }
  661. assertHttpNotError(r, t)
  662. if r.Code != http.StatusNoContent {
  663. t.Fatalf("%d NO CONTENT expected, received %d\n", http.StatusNoContent, r.Code)
  664. }
  665. containerAssertExists(eng, containerID, t)
  666. // Give some time to the process to start
  667. // FIXME: use Wait once it's available as a job
  668. containerWaitTimeout(eng, containerID, t)
  669. if !containerRunning(eng, containerID, t) {
  670. t.Errorf("Container should be running")
  671. }
  672. r = httptest.NewRecorder()
  673. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  674. t.Fatal(err)
  675. }
  676. // Starting an already started container should return an error
  677. // FIXME: verify a precise error code. There is a possible bug here
  678. // which causes this to return 404 even though the container exists.
  679. assertHttpError(r, t)
  680. containerAssertExists(eng, containerID, t)
  681. containerKill(eng, containerID, t)
  682. }
  683. // Expected behaviour: using / as a bind mount source should throw an error
  684. func TestRunErrorBindMountRootSource(t *testing.T) {
  685. eng := NewTestEngine(t)
  686. defer mkRuntimeFromEngine(eng, t).Nuke()
  687. containerID := createTestContainer(
  688. eng,
  689. &runconfig.Config{
  690. Image: unitTestImageID,
  691. Cmd: []string{"/bin/cat"},
  692. OpenStdin: true,
  693. },
  694. t,
  695. )
  696. hostConfigJSON, err := json.Marshal(&runconfig.HostConfig{
  697. Binds: []string{"/:/tmp"},
  698. })
  699. req, err := http.NewRequest("POST", "/containers/"+containerID+"/start", bytes.NewReader(hostConfigJSON))
  700. if err != nil {
  701. t.Fatal(err)
  702. }
  703. req.Header.Set("Content-Type", "application/json")
  704. r := httptest.NewRecorder()
  705. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  706. t.Fatal(err)
  707. }
  708. if r.Code != http.StatusInternalServerError {
  709. containerKill(eng, containerID, t)
  710. t.Fatal("should have failed to run when using / as a source for the bind mount")
  711. }
  712. }
  713. func TestPostContainersStop(t *testing.T) {
  714. eng := NewTestEngine(t)
  715. defer mkRuntimeFromEngine(eng, t).Nuke()
  716. containerID := createTestContainer(eng,
  717. &runconfig.Config{
  718. Image: unitTestImageID,
  719. Cmd: []string{"/bin/top"},
  720. OpenStdin: true,
  721. },
  722. t,
  723. )
  724. startContainer(eng, containerID, t)
  725. // Give some time to the process to start
  726. containerWaitTimeout(eng, containerID, t)
  727. if !containerRunning(eng, containerID, t) {
  728. t.Errorf("Container should be running")
  729. }
  730. // Note: as it is a POST request, it requires a body.
  731. req, err := http.NewRequest("POST", "/containers/"+containerID+"/stop?t=1", bytes.NewReader([]byte{}))
  732. if err != nil {
  733. t.Fatal(err)
  734. }
  735. r := httptest.NewRecorder()
  736. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  737. t.Fatal(err)
  738. }
  739. assertHttpNotError(r, t)
  740. if r.Code != http.StatusNoContent {
  741. t.Fatalf("%d NO CONTENT expected, received %d\n", http.StatusNoContent, r.Code)
  742. }
  743. if containerRunning(eng, containerID, t) {
  744. t.Fatalf("The container hasn't been stopped")
  745. }
  746. }
  747. func TestPostContainersWait(t *testing.T) {
  748. eng := NewTestEngine(t)
  749. defer mkRuntimeFromEngine(eng, t).Nuke()
  750. containerID := createTestContainer(eng,
  751. &runconfig.Config{
  752. Image: unitTestImageID,
  753. Cmd: []string{"/bin/sleep", "1"},
  754. OpenStdin: true,
  755. },
  756. t,
  757. )
  758. startContainer(eng, containerID, t)
  759. setTimeout(t, "Wait timed out", 3*time.Second, func() {
  760. r := httptest.NewRecorder()
  761. req, err := http.NewRequest("POST", "/containers/"+containerID+"/wait", bytes.NewReader([]byte{}))
  762. if err != nil {
  763. t.Fatal(err)
  764. }
  765. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  766. t.Fatal(err)
  767. }
  768. assertHttpNotError(r, t)
  769. var apiWait engine.Env
  770. if err := apiWait.Decode(r.Body); err != nil {
  771. t.Fatal(err)
  772. }
  773. if apiWait.GetInt("StatusCode") != 0 {
  774. t.Fatalf("Non zero exit code for sleep: %d\n", apiWait.GetInt("StatusCode"))
  775. }
  776. })
  777. if containerRunning(eng, containerID, t) {
  778. t.Fatalf("The container should be stopped after wait")
  779. }
  780. }
  781. func TestPostContainersAttach(t *testing.T) {
  782. eng := NewTestEngine(t)
  783. defer mkRuntimeFromEngine(eng, t).Nuke()
  784. containerID := createTestContainer(eng,
  785. &runconfig.Config{
  786. Image: unitTestImageID,
  787. Cmd: []string{"/bin/cat"},
  788. OpenStdin: true,
  789. },
  790. t,
  791. )
  792. // Start the process
  793. startContainer(eng, containerID, t)
  794. stdin, stdinPipe := io.Pipe()
  795. stdout, stdoutPipe := io.Pipe()
  796. // Try to avoid the timeout in destroy. Best effort, don't check error
  797. defer func() {
  798. closeWrap(stdin, stdinPipe, stdout, stdoutPipe)
  799. containerKill(eng, containerID, t)
  800. }()
  801. // Attach to it
  802. c1 := make(chan struct{})
  803. go func() {
  804. defer close(c1)
  805. r := &hijackTester{
  806. ResponseRecorder: httptest.NewRecorder(),
  807. in: stdin,
  808. out: stdoutPipe,
  809. }
  810. req, err := http.NewRequest("POST", "/containers/"+containerID+"/attach?stream=1&stdin=1&stdout=1&stderr=1", bytes.NewReader([]byte{}))
  811. if err != nil {
  812. t.Fatal(err)
  813. }
  814. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  815. t.Fatal(err)
  816. }
  817. assertHttpNotError(r.ResponseRecorder, t)
  818. }()
  819. // Acknowledge hijack
  820. setTimeout(t, "hijack acknowledge timed out", 2*time.Second, func() {
  821. stdout.Read([]byte{})
  822. stdout.Read(make([]byte, 4096))
  823. })
  824. setTimeout(t, "read/write assertion timed out", 2*time.Second, func() {
  825. if err := assertPipe("hello\n", string([]byte{1, 0, 0, 0, 0, 0, 0, 6})+"hello", stdout, stdinPipe, 150); err != nil {
  826. t.Fatal(err)
  827. }
  828. })
  829. // Close pipes (client disconnects)
  830. if err := closeWrap(stdin, stdinPipe, stdout, stdoutPipe); err != nil {
  831. t.Fatal(err)
  832. }
  833. // Wait for attach to finish, the client disconnected, therefore, Attach finished his job
  834. setTimeout(t, "Waiting for CmdAttach timed out", 10*time.Second, func() {
  835. <-c1
  836. })
  837. // We closed stdin, expect /bin/cat to still be running
  838. // Wait a little bit to make sure container.monitor() did his thing
  839. containerWaitTimeout(eng, containerID, t)
  840. // Try to avoid the timeout in destroy. Best effort, don't check error
  841. cStdin, _ := containerAttach(eng, containerID, t)
  842. cStdin.Close()
  843. containerWait(eng, containerID, t)
  844. }
  845. func TestPostContainersAttachStderr(t *testing.T) {
  846. eng := NewTestEngine(t)
  847. defer mkRuntimeFromEngine(eng, t).Nuke()
  848. containerID := createTestContainer(eng,
  849. &runconfig.Config{
  850. Image: unitTestImageID,
  851. Cmd: []string{"/bin/sh", "-c", "/bin/cat >&2"},
  852. OpenStdin: true,
  853. },
  854. t,
  855. )
  856. // Start the process
  857. startContainer(eng, containerID, t)
  858. stdin, stdinPipe := io.Pipe()
  859. stdout, stdoutPipe := io.Pipe()
  860. // Try to avoid the timeout in destroy. Best effort, don't check error
  861. defer func() {
  862. closeWrap(stdin, stdinPipe, stdout, stdoutPipe)
  863. containerKill(eng, containerID, t)
  864. }()
  865. // Attach to it
  866. c1 := make(chan struct{})
  867. go func() {
  868. defer close(c1)
  869. r := &hijackTester{
  870. ResponseRecorder: httptest.NewRecorder(),
  871. in: stdin,
  872. out: stdoutPipe,
  873. }
  874. req, err := http.NewRequest("POST", "/containers/"+containerID+"/attach?stream=1&stdin=1&stdout=1&stderr=1", bytes.NewReader([]byte{}))
  875. if err != nil {
  876. t.Fatal(err)
  877. }
  878. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  879. t.Fatal(err)
  880. }
  881. assertHttpNotError(r.ResponseRecorder, t)
  882. }()
  883. // Acknowledge hijack
  884. setTimeout(t, "hijack acknowledge timed out", 2*time.Second, func() {
  885. stdout.Read([]byte{})
  886. stdout.Read(make([]byte, 4096))
  887. })
  888. setTimeout(t, "read/write assertion timed out", 2*time.Second, func() {
  889. if err := assertPipe("hello\n", string([]byte{2, 0, 0, 0, 0, 0, 0, 6})+"hello", stdout, stdinPipe, 150); err != nil {
  890. t.Fatal(err)
  891. }
  892. })
  893. // Close pipes (client disconnects)
  894. if err := closeWrap(stdin, stdinPipe, stdout, stdoutPipe); err != nil {
  895. t.Fatal(err)
  896. }
  897. // Wait for attach to finish, the client disconnected, therefore, Attach finished his job
  898. setTimeout(t, "Waiting for CmdAttach timed out", 10*time.Second, func() {
  899. <-c1
  900. })
  901. // We closed stdin, expect /bin/cat to still be running
  902. // Wait a little bit to make sure container.monitor() did his thing
  903. containerWaitTimeout(eng, containerID, t)
  904. // Try to avoid the timeout in destroy. Best effort, don't check error
  905. cStdin, _ := containerAttach(eng, containerID, t)
  906. cStdin.Close()
  907. containerWait(eng, containerID, t)
  908. }
  909. // FIXME: Test deleting running container
  910. // FIXME: Test deleting container with volume
  911. // FIXME: Test deleting volume in use by other container
  912. func TestDeleteContainers(t *testing.T) {
  913. eng := NewTestEngine(t)
  914. defer mkRuntimeFromEngine(eng, t).Nuke()
  915. containerID := createTestContainer(eng,
  916. &runconfig.Config{
  917. Image: unitTestImageID,
  918. Cmd: []string{"touch", "/test"},
  919. },
  920. t,
  921. )
  922. req, err := http.NewRequest("DELETE", "/containers/"+containerID, nil)
  923. if err != nil {
  924. t.Fatal(err)
  925. }
  926. r := httptest.NewRecorder()
  927. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  928. t.Fatal(err)
  929. }
  930. assertHttpNotError(r, t)
  931. if r.Code != http.StatusNoContent {
  932. t.Fatalf("%d NO CONTENT expected, received %d\n", http.StatusNoContent, r.Code)
  933. }
  934. containerAssertNotExists(eng, containerID, t)
  935. }
  936. func TestOptionsRoute(t *testing.T) {
  937. eng := NewTestEngine(t)
  938. defer mkRuntimeFromEngine(eng, t).Nuke()
  939. r := httptest.NewRecorder()
  940. req, err := http.NewRequest("OPTIONS", "/", nil)
  941. if err != nil {
  942. t.Fatal(err)
  943. }
  944. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  945. t.Fatal(err)
  946. }
  947. assertHttpNotError(r, t)
  948. if r.Code != http.StatusOK {
  949. t.Errorf("Expected response for OPTIONS request to be \"200\", %v found.", r.Code)
  950. }
  951. }
  952. func TestGetEnabledCors(t *testing.T) {
  953. eng := NewTestEngine(t)
  954. defer mkRuntimeFromEngine(eng, t).Nuke()
  955. r := httptest.NewRecorder()
  956. req, err := http.NewRequest("GET", "/version", nil)
  957. if err != nil {
  958. t.Fatal(err)
  959. }
  960. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  961. t.Fatal(err)
  962. }
  963. assertHttpNotError(r, t)
  964. if r.Code != http.StatusOK {
  965. t.Errorf("Expected response for OPTIONS request to be \"200\", %v found.", r.Code)
  966. }
  967. allowOrigin := r.Header().Get("Access-Control-Allow-Origin")
  968. allowHeaders := r.Header().Get("Access-Control-Allow-Headers")
  969. allowMethods := r.Header().Get("Access-Control-Allow-Methods")
  970. if allowOrigin != "*" {
  971. t.Errorf("Expected header Access-Control-Allow-Origin to be \"*\", %s found.", allowOrigin)
  972. }
  973. if allowHeaders != "Origin, X-Requested-With, Content-Type, Accept" {
  974. t.Errorf("Expected header Access-Control-Allow-Headers to be \"Origin, X-Requested-With, Content-Type, Accept\", %s found.", allowHeaders)
  975. }
  976. if allowMethods != "GET, POST, DELETE, PUT, OPTIONS" {
  977. t.Errorf("Expected hearder Access-Control-Allow-Methods to be \"GET, POST, DELETE, PUT, OPTIONS\", %s found.", allowMethods)
  978. }
  979. }
  980. func TestDeleteImages(t *testing.T) {
  981. eng := NewTestEngine(t)
  982. //we expect errors, so we disable stderr
  983. eng.Stderr = ioutil.Discard
  984. defer mkRuntimeFromEngine(eng, t).Nuke()
  985. initialImages := getImages(eng, t, true, "")
  986. if err := eng.Job("tag", unitTestImageName, "test", "test").Run(); err != nil {
  987. t.Fatal(err)
  988. }
  989. images := getImages(eng, t, true, "")
  990. if len(images.Data[0].GetList("RepoTags")) != len(initialImages.Data[0].GetList("RepoTags"))+1 {
  991. t.Errorf("Expected %d images, %d found", len(initialImages.Data[0].GetList("RepoTags"))+1, len(images.Data[0].GetList("RepoTags")))
  992. }
  993. req, err := http.NewRequest("DELETE", "/images/"+unitTestImageID, nil)
  994. if err != nil {
  995. t.Fatal(err)
  996. }
  997. r := httptest.NewRecorder()
  998. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  999. t.Fatal(err)
  1000. }
  1001. if r.Code != http.StatusConflict {
  1002. t.Fatalf("Expected http status 409-conflict, got %v", r.Code)
  1003. }
  1004. req2, err := http.NewRequest("DELETE", "/images/test:test", nil)
  1005. if err != nil {
  1006. t.Fatal(err)
  1007. }
  1008. r2 := httptest.NewRecorder()
  1009. if err := server.ServeRequest(eng, api.APIVERSION, r2, req2); err != nil {
  1010. t.Fatal(err)
  1011. }
  1012. assertHttpNotError(r2, t)
  1013. if r2.Code != http.StatusOK {
  1014. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  1015. }
  1016. outs := engine.NewTable("Created", 0)
  1017. if _, err := outs.ReadListFrom(r2.Body.Bytes()); err != nil {
  1018. t.Fatal(err)
  1019. }
  1020. if len(outs.Data) != 1 {
  1021. t.Fatalf("Expected %d event (untagged), got %d", 1, len(outs.Data))
  1022. }
  1023. images = getImages(eng, t, false, "")
  1024. if images.Len() != initialImages.Len() {
  1025. t.Errorf("Expected %d image, %d found", initialImages.Len(), images.Len())
  1026. }
  1027. }
  1028. func TestPostContainersCopy(t *testing.T) {
  1029. eng := NewTestEngine(t)
  1030. defer mkRuntimeFromEngine(eng, t).Nuke()
  1031. // Create a container and remove a file
  1032. containerID := createTestContainer(eng,
  1033. &runconfig.Config{
  1034. Image: unitTestImageID,
  1035. Cmd: []string{"touch", "/test.txt"},
  1036. },
  1037. t,
  1038. )
  1039. containerRun(eng, containerID, t)
  1040. r := httptest.NewRecorder()
  1041. var copyData engine.Env
  1042. copyData.Set("Resource", "/test.txt")
  1043. copyData.Set("HostPath", ".")
  1044. jsonData := bytes.NewBuffer(nil)
  1045. if err := copyData.Encode(jsonData); err != nil {
  1046. t.Fatal(err)
  1047. }
  1048. req, err := http.NewRequest("POST", "/containers/"+containerID+"/copy", jsonData)
  1049. if err != nil {
  1050. t.Fatal(err)
  1051. }
  1052. req.Header.Add("Content-Type", "application/json")
  1053. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  1054. t.Fatal(err)
  1055. }
  1056. assertHttpNotError(r, t)
  1057. if r.Code != http.StatusOK {
  1058. t.Fatalf("%d OK expected, received %d\n", http.StatusOK, r.Code)
  1059. }
  1060. found := false
  1061. for tarReader := tar.NewReader(r.Body); ; {
  1062. h, err := tarReader.Next()
  1063. if err != nil {
  1064. if err == io.EOF {
  1065. break
  1066. }
  1067. t.Fatal(err)
  1068. }
  1069. if h.Name == "test.txt" {
  1070. found = true
  1071. break
  1072. }
  1073. }
  1074. if !found {
  1075. t.Fatalf("The created test file has not been found in the copied output")
  1076. }
  1077. }
  1078. func TestPostContainersCopyWhenContainerNotFound(t *testing.T) {
  1079. eng := NewTestEngine(t)
  1080. defer mkRuntimeFromEngine(eng, t).Nuke()
  1081. r := httptest.NewRecorder()
  1082. var copyData engine.Env
  1083. copyData.Set("Resource", "/test.txt")
  1084. copyData.Set("HostPath", ".")
  1085. jsonData := bytes.NewBuffer(nil)
  1086. if err := copyData.Encode(jsonData); err != nil {
  1087. t.Fatal(err)
  1088. }
  1089. req, err := http.NewRequest("POST", "/containers/id_not_found/copy", jsonData)
  1090. if err != nil {
  1091. t.Fatal(err)
  1092. }
  1093. req.Header.Add("Content-Type", "application/json")
  1094. if err := server.ServeRequest(eng, api.APIVERSION, r, req); err != nil {
  1095. t.Fatal(err)
  1096. }
  1097. if r.Code != http.StatusNotFound {
  1098. t.Fatalf("404 expected for id_not_found Container, received %v", r.Code)
  1099. }
  1100. }
  1101. // Mocked types for tests
  1102. type NopConn struct {
  1103. io.ReadCloser
  1104. io.Writer
  1105. }
  1106. func (c *NopConn) LocalAddr() net.Addr { return nil }
  1107. func (c *NopConn) RemoteAddr() net.Addr { return nil }
  1108. func (c *NopConn) SetDeadline(t time.Time) error { return nil }
  1109. func (c *NopConn) SetReadDeadline(t time.Time) error { return nil }
  1110. func (c *NopConn) SetWriteDeadline(t time.Time) error { return nil }
  1111. type hijackTester struct {
  1112. *httptest.ResponseRecorder
  1113. in io.ReadCloser
  1114. out io.Writer
  1115. }
  1116. func (t *hijackTester) Hijack() (net.Conn, *bufio.ReadWriter, error) {
  1117. bufrw := bufio.NewReadWriter(bufio.NewReader(t.in), bufio.NewWriter(t.out))
  1118. conn := &NopConn{
  1119. ReadCloser: t.in,
  1120. Writer: t.out,
  1121. }
  1122. return conn, bufrw, nil
  1123. }