forked from marijnh/Eloquent-JavaScript
-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy path20_node.txt
1265 lines (1059 loc) · 51.1 KB
/
20_node.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
:chap_num: 20
:prev_link: 19_paint
:next_link: 21_skillsharing
:code_links: ["code/file_server.js", "code/file_server_promises.js"]
= Node.js =
[chapterquote="true"]
[quote, Master Yuan-Ma, The Book of Programming]
____
A student asked ‘The
programmers of old used only simple machines and no programming
languages, yet they made beautiful programs. Why do we use complicated
machines and programming languages?’. Fu-Tzu replied ‘The builders of
old used only sticks and clay, yet they made beautiful huts.’
____
(((command line)))(((Yuan-Ma)))(((Book of Programming)))So far, you
have learned the JavaScript language and used it within a single
environment: the browser. This chapter and the link:21_skillsharing.html#skillsharing[next one] will briefly introduce
you to ((Node.js)), a program that allows you to apply your JavaScript
skills outside of the browser. With it, you can build anything from
simple command-line tools to dynamic HTTP ((server))s.
These chapters aim to teach you the important ideas that Node.js
builds on and to give you enough information to write some useful
programs for it. They do not try to be a complete, or even a thorough,
treatment of Node.
ifdef::interactive_target[]
Whereas you could run the code in previous chapters directly on these
pages, since it was either raw JavaScript or written for the browser,
the code samples in this chapter are written for Node and won't run
in the browser.
endif::interactive_target[]
If you want to follow along and run the code in this chapter, start by
going to http://nodejs.org[_nodejs.org_] and following the
installation instructions for your operating system. Also refer to
that website for further ((documentation)) about Node and its built-in
((module))s.
== Background ==
(((responsiveness)))(((input)))One of the more difficult problems with
writing systems that communicate over the ((network)) is managing input
and ((output))—that is, the reading and writing of data to and from
the network, the ((hard drive)), and other such devices. Moving data
around takes time, and ((scheduling)) it cleverly can make a big
difference in how quickly a system responds to the user or to network
requests.
The traditional way to handle input and output is to have a function, such as
`readFile`, start reading a file and return only when the
file has been fully read. This is called _((synchronous I/O))_ (I/O
stands for input/output).
(((asynchronous I/O)))(((asynchronous programming)))(((callback
function)))Node was initially conceived for the purpose of making
_asynchronous_ I/O easy and convenient. We have seen asynchronous
interfaces before, such as a browser's `XMLHttpRequest` object,
discussed in link:17_http.html#xmlhttprequest[Chapter 17]. An asynchronous
interface allows the script to continue running while it does its
work and calls a callback function when it's done. This is the way
Node does all its I/O.
(((programming language)))(((Node.js)))(((standard)))JavaScript lends
itself well to a system like Node. It is one of the few programming
languages that does not have a built-in way to do I/O. Thus, JavaScript could
be fit onto Node's rather eccentric approach to I/O without ending up
with two inconsistent interfaces. In 2009, when Node was being
designed, people were already doing callback-based I/O in the browser,
so the ((community)) around the language was used to an ((asynchronous
programming)) style.
== Asynchronicity ==
I'll try to illustrate synchronous versus asynchronous
I/O with a small example, where a program needs to fetch two resources
from the Internet and then do some simple processing with the result.
(((synchronous I/O)))In a synchronous environment, the obvious way to
perform this task is to make the requests one after the other. This method has the
drawback that the second request will be started only when the first
has finished. The total time taken will be at least the sum of the two
response times. This is not an effective use of the machine, which
will be mostly idle when it is transmitting and receiving data over
the ((network)).
(((parallelism)))The solution to this problem, in a synchronous
system, is to start additional ((thread))s of control. (Refer to
link:14_event.html#timeline[Chapter 14] for a previous discussion of
threads.) A second thread could start the second request, and then
both threads wait for their results to come back, after which they
resynchronize to combine their results.
(((CPU)))(((blocking)))(((synchronous I/O)))(((asynchronous
I/O)))(((timeline)))(((callback function)))In the following diagram,
the thick lines represent time the program spends running normally,
and the thin lines represent time spent waiting for I/O. In the
synchronous model, the time taken by I/O is _part_ of the timeline for
a given thread of control. In the asynchronous model, starting an I/O
action conceptually causes a _split_ in the timeline. The thread that
initiated the I/O continues running, and the I/O itself is done
alongside it, finally calling a callback function when it is finished.
image::img/control-io.svg[alt="Control flow for synchronous and asynchronous I/O",width="8cm"]
(((control flow)))(((asynchronous programming)))Another way to express
this difference is that waiting for I/O to finish is _implicit_ in the
synchronous model, while it is _explicit_, directly under our
control, in the asynchronous one. But asynchronicity cuts both ways. It
makes expressing programs that do not fit the straight-line
model of control easier, but it also makes expressing
programs that do follow a straight line more awkward.
(((verbosity)))In link:17_http.html#promises[Chapter 17], I already
touched on the fact that all those callbacks add quite a lot of
noise and indirection to a program. Whether this style of
asynchronicity is a good idea in general can be debated. In any case,
it takes some getting used to.
(((asynchronous programming)))(((parallelism)))But for a
JavaScript-based system, I would argue that callback-style
asynchronicity is a sensible choice. One of the strengths of
JavaScript is its simplicity, and trying to add multiple ((thread))s
of control to it would add a lot of complexity. Though callbacks don't tend
to lead to simple _code_, as a _concept_, they're pleasantly
simple yet powerful enough to write high-performance web servers.
== The node command ==
(((node program)))When ((Node.js)) is installed on a system, it
provides a program called `node`, which is used to run JavaScript
files. Say you have a file `hello.js`, containing this code:
[source,javascript]
----
var message = "Hello world";
console.log(message);
----
You can then run `node` from the ((command line)) like this to execute
the program:
----
$ node hello.js
Hello world
----
(((console.log)))The `console.log` method in Node does something
similar to what it does in the browser. It prints out a piece of text.
But in Node, the text will go to the process’ ((standard output))
stream, rather than to a browser's ((JavaScript console)).
(((node program)))(((read-eval-print loop)))If you run `node` without
giving it a file, it provides you with a prompt at which you can type
JavaScript code and immediately see the result.
----
$ node
> 1 + 1
2
> [-1, -2, -3].map(Math.abs)
[1, 2, 3]
> process.exit(0)
$
----
(((process object)))(((global scope)))(((variable,global)))(((exit
method)))(((status code)))The `process` variable, just like the
`console` variable, is available globally in Node. It provides various
ways to inspect and manipulate the current program. The `exit` method
ends the process and can be given an exit status code, which tells
the program that started `node` (in this case, the command-line shell)
whether the program completed successfully (code zero) or encountered
an error (any other code).
(((command line)))(((argv property)))To find the command-line arguments given to
your script, you can read `process.argv`, which is an array of
strings. Note that it also includes the name of the `node` commands
and your script name, so the actual arguments start at index 2. If
`showargv.js` simply contains the statement
`console.log(process.argv)`, you could run it like this:
----
$ node showargv.js one --and two
["node", "/home/marijn/showargv.js", "one", "--and", "two"]
----
(((variable,global)))All the ((standard)) JavaScript global variables,
such as `Array`, `Math`, and `JSON`, are also present in Node's
environment. Browser-related functionality, such as `document` and
`alert`, is absent.
(((global object)))(((global scope)))(((window)))The global scope
object, which is called `window` in the browser, has the more sensible
name `global` in Node.
== Modules ==
(((Node.js)))(((global scope)))(((module loader)))Beyond the few
variables I mentioned, such as `console` and `process`, Node puts
little functionality in the global scope. If you want to access other
built-in functionality, you have to ask the module system for it.
(((library)))(((require function)))The ((CommonJS)) module system,
based on the `require` function, was described in
link:10_modules.html#commonjs[Chapter 10]. This system is built into
Node and is used to load anything from built-in ((module))s to
downloaded libraries to files that are part of your own program.
(((path,file system)))(((relative path)))(((resolution)))When `require` is called, Node has
to resolve the given string to an actual ((file)) to load. Pathnames
that start with `"/"`, `"./"`, or `"../"` are resolved relative to the
current module's path, where `"./"` stands for the current
directory, `"../"` for one directory up, and `"/"` for the root of the
file system. So if you ask for `"./world/world"` from the file
`/home/marijn/elife/run.js`, Node will try to load the file
`/home/marijn/elife/world/world.js`. The `.js` extension may be
omitted.
(((node_modules directory)))When a string that does not look like a
relative or absolute path is given to `require`, it is assumed to
refer to either a built-in ((module)) or a module installed in a
`node_modules` directory. For example, `require("fs")` will give you
Node's built-in file system module, and `require("elife")` will try to
load the library found in `node_modules/elife/`. A common way to
install such libraries is by using ((NPM)), which I will discuss in a
moment.
(((require function)))(((Node.js)))(((garble example)))To illustrate
the use of `require`, let's set up a simple project consisting of two
files. The first one is called `main.js`, which defines a script that
can be called from the ((command line)) to garble a string.
[source,javascript]
----
var garble = require("./garble");
// Index 2 holds the first actual command-line argument
var argument = process.argv[2];
console.log(garble(argument));
----
(((reuse)))The file `garble.js` defines a library for garbling strings,
which can be used both by the command-line tool defined earlier and by
other scripts that need direct access to a garbling function.
[source,javascript]
----
module.exports = function(string) {
return string.split("").map(function(ch) {
return String.fromCharCode(ch.charCodeAt(0) + 5);
}).join("");
};
----
(((exports object)))(((CommonJS)))Remember that replacing
`module.exports`, rather than adding properties to it, allows us to
export a specific value from a module. In this case, we make the
result of requiring our `garble` file the garbling function itself.
(((Unicode)))(((split method)))(((map method)))(((join method)))The
function splits the string it is given into single characters by
splitting on the empty string and then replaces each character with
the character whose code is five points higher. Finally, it joins the
result back into a string.
We can now call our tool like this:
----
$ node main.js JavaScript
Of{fXhwnuy
----
== Installing with NPM ==
(((NPM)))(((Node.js)))(((npm program)))(((library)))NPM, which was
briefly discussed in link:10_modules.html#modules_npm[Chapter 10], is
an online repository of JavaScript ((module))s, many of which are
specifically written for Node. When you install Node on your computer,
you also get a program called `npm`, which provides a convenient
interface to this repository.
(((figlet module)))For example, one module you will find on NPM is
`figlet`, which can convert text into __((ASCII art))__—drawings made
out of text characters. The following transcript shows how to ((install))
and use it:
----
$ npm install figlet
npm GET https://registry.npmjs.org/figlet
npm 200 https://registry.npmjs.org/figlet
npm GET https://registry.npmjs.org/figlet/-/figlet-1.0.9.tgz
npm 200 https://registry.npmjs.org/figlet/-/figlet-1.0.9.tgz
[email protected] node_modules/figlet
$ node
> var figlet = require("figlet");
> figlet.text("Hello world!", function(error, data) {
if (error)
console.error(error);
else
console.log(data);
});
_ _ _ _ _ _ _
| | | | ___| | | ___ __ _____ _ __| | __| | |
| |_| |/ _ \ | |/ _ \ \ \ /\ / / _ \| '__| |/ _` | |
| _ | __/ | | (_) | \ V V / (_) | | | | (_| |_|
|_| |_|\___|_|_|\___/ \_/\_/ \___/|_| |_|\__,_(_)
----
(((require function)))(((node_modules directory)))(((npm
program)))After running `npm install`, ((NPM)) will have created a
directory called `node_modules`. Inside that directory will be a `figlet`
directory, which contains the ((library)). When we run `node` and call
`require("figlet")`, this library is loaded, and we can call its
`text` method to draw some big letters.
(((error handling)))(((return value)))Somewhat unexpectedly perhaps,
instead of simply returning the string that makes up the big letters,
`figlet.text` takes a ((callback function)) that it passes its result
to. It also passes the callback another argument, `error`, which will
hold an error object when something goes wrong or null when
everything is all right.
(((file system)))(((asynchronous programming)))(((Node.js)))This is a
common pattern in Node code. Rendering something with `figlet`
requires the library to read a file that contains the letter shapes.
Reading that file from disk is an asynchronous operation in
Node, so `figlet.text` can't immediately return its
result. Asynchronicity is infectious, in a way—every function that
calls an asynchronous function must itself become asynchronous.
(((dependency)))(((publishing)))(((package.json file)))(((npm
program)))There is much more to ((NPM)) than `npm install`. It reads
`package.json` files, which contain ((JSON))-encoded information about
a program or library, such as which other libraries it depends on.
Doing `npm install` in a directory that contains such a file will
automatically install all dependencies, as well as _their_
dependencies. The `npm` tool is also used to publish libraries to
NPM's online repository of packages so that other people can find,
download, and use them.
(((library)))This book won't delve further into the details of ((NPM))
usage. Refer to http://npmjs.org[_npmjs.org_] for further
documentation and for an easy way to search for libraries.
== The file system module ==
(((directory)))(((fs module)))(((Node.js)))One of the most commonly
used built-in modules that comes with Node is the `"fs"` module, which
stands for _((file system))_. This module provides functions for
working with ((file))s and directories.
(((readFile function)))(((callback function)))For example, there is a
function called `readFile`, which reads a file and then calls a
callback with the file's contents.
[source,javascript]
----
var fs = require("fs");
fs.readFile("file.txt", "utf8", function(error, text) {
if (error)
throw error;
console.log("The file contained:", text);
});
----
(((Buffer type)))The second argument to
`readFile` indicates the _((character encoding))_ used to decode the
file into a string. There are several ways in which ((text)) can be
encoded to ((binary data)), but most modern systems use ((UTF-8)) to
encode text, so unless you have reasons to believe another encoding is
used, passing `"utf8"` when reading a text file is a safe bet. If you
do not pass an encoding, Node will assume you are interested in the
binary data and will give you a `Buffer` object instead of a
string. This is an ((array-like object)) that contains numbers
representing the bytes in the files.
[source,javascript]
----
var fs = require("fs");
fs.readFile("file.txt", function(error, buffer) {
if (error)
throw error;
console.log("The file contained", buffer.length, "bytes.",
"The first byte is:", buffer[0]);
});
----
(((writeFile function)))(((file system)))A similar function,
`writeFile`, is used to write a ((file)) to disk.
[source,javascript]
----
var fs = require("fs");
fs.writeFile("graffiti.txt", "Node was here", function(err) {
if (err)
console.log("Failed to write file:", err);
else
console.log("File written.");
});
----
(((Buffer type)))(((character encoding)))Here, it was not necessary to
specify the encoding since `writeFile` will assume that if it is
given a string to write, rather than a `Buffer` object, it should
write it out as text using its default character encoding, which is
((UTF-8)).
(((fs module)))(((readdir function)))(((stat function)))(((rename
function)))(((unlink function)))The `"fs"` module contains many other
useful functions: `readdir` will return the ((file))s
in a ((directory)) as an array of strings, `stat` will retrieve
information about a file, `rename` will rename a file, `unlink` will
remove one, and so on. See the documentation at
http://nodejs.org[_nodejs.org_] for specifics.
(((synchronous I/O)))(((fs module)))(((readFileSync function)))Many of
the functions in `"fs"` come in both synchronous and asynchronous variants.
For example, there is a synchronous version of
`readFile` called `readFileSync`.
[source,javascript]
----
var fs = require("fs");
console.log(fs.readFileSync("file.txt", "utf8"));
----
(((synchronous
I/O)))(((optimization)))(((performance)))(((blocking)))Synchronous functions require
less ceremony to use and can be useful in simple scripts, where the
extra speed provided by asynchronous I/O is irrelevant. But note that
while such a synchronous operation is being performed, your program
will be stopped entirely. If it should be responding to the user or to
other machines on the network, being stuck on synchronous I/O might
produce annoying delays.
== The HTTP module ==
(((Node.js)))(((http module)))Another central module is called
`"http"`. It provides functionality for running ((HTTP)) ((server))s
and making HTTP ((request))s.
(((listening (TCP))))(((listen method)))(((createServer
function)))This is all it takes to start a simple HTTP server:
[source,javascript]
----
var http = require("http");
var server = http.createServer(function(request, response) {
response.writeHead(200, {"Content-Type": "text/html"});
response.write("<h1>Hello!</h1><p>You asked for <code>" +
request.url + "</code></p>");
response.end();
});
server.listen(8000);
----
(((port)))(((localhost)))If you run this script on your own machine,
you can point your web browser at
http://localhost:8000/hello[_http://localhost:8000/hello_] to make a
request to your server. It will respond with a small HTML page.
(((createServer function)))(((HTTP)))The function passed as an argument
to `createServer` is called every time a client tries to connect to
the server. The `request` and `response` variables are objects
representing the incoming and outgoing data. The first contains
information about the ((request)), such as its `url` property, which
tells us to what URL the request was made.
(((200 (HTTP status code))))(((Content-Type header)))(((writeHead
method)))To send something back, you call methods on the `response`
object. The first, `writeHead`, will write out the ((response))
((header))s (see link:17_http.html#headers[Chapter 17]). You give it
the status code (200 for “OK” in this case) and an object that
contains header values. Here we tell the client that we will be
sending back an HTML document.
(((writable stream)))(((body (HTTP))))(((stream)))(((write
method)))(((end method)))Next, the actual response body (the document
itself) is sent with `response.write`. You are allowed to call this
method multiple times if you want to send the response piece by piece,
possibly streaming data to the client as it becomes available.
Finally, `response.end` signals the end of the response.
(((listen method)))The call to `server.listen` causes the ((server))
to start waiting for connections on ((port)) 8000. This is the reason
you have to connect to _localhost:8000_, rather than just _localhost_
(which would use the default port, 80), to speak to this server.
(((Node.js)))(((kill)))To stop running a Node script like this, which
doesn't finish automatically because it is waiting for further events
(in this case, network connections), press Ctrl-C.
A real web ((server)) usually does more than the one in the previous
example—it looks at the request's
((method)) (the `method` property) to see what action the client is
trying to perform and at the request's ((URL)) to find out which
resource this action is being performed on. You'll see a more advanced
server link:20_node.html#file_server[later in this chapter].
(((http module)))(((request function)))To act as an ((HTTP))
_((client))_, we can use the `request` function in the `"http"`
module.
[source,javascript]
----
var http = require("http");
var request = http.request({
hostname: "eloquentjavascript.net",
path: "/20_node.html",
method: "GET",
headers: {Accept: "text/html"}
}, function(response) {
console.log("Server responded with status code",
response.statusCode);
});
request.end();
----
(((Node.js)))(((callback function)))(((readable stream)))The first
argument to `request` configures the request, telling Node what server
to talk to, what path to request from that server, which method to
use, and so on. The second argument is the function that should be
called when a response comes in. It is given an object that allows us
to inspect the response, for example to find out its status code.
(((GET method)))(((write method)))(((end method)))(((writable
stream)))(((request function)))Just like the `response` object
we saw in the server, the object returned by `request` allows us to
((stream)) data into the ((request)) with the `write` method and
finish the request with the `end` method. The example does not use
`write` because `GET` requests should not contain data
in their request body.
(((HTTPS)))(((https module)))(((request function)))To make requests to secure HTTP (HTTPS)
URLs, Node provides a package called `https`, which contains
its own `request` function, similar to `http.request`.
== Streams ==
(((Node.js)))(((write method)))(((end method)))(((Buffer
type)))(((stream)))(((writable stream)))We have seen two examples of
writable streams in the HTTP examples—namely, the response object that
the server could write to and the request object that was returned
from `http.request`.
(((callback function)))(((asynchronous programming))) Writable streams are a widely used concept in Node
interfaces. All writable streams have a `write` method, which can be
passed a string or a `Buffer` object. Their `end` method closes the
stream and, if given an argument, will also write out a piece of data
before it does so. Both of these
methods can also be given a callback as an additional argument, which
they will call when the writing to or closing of the stream has
finished.
(((createWriteStream function)))(((writeFile function)))It is possible
to create a writable stream that points at a ((file)) with the
`fs.createWriteStream` function. Then you can use the `write` method on
the resulting object to write the file one piece at a time, rather
than in one shot as with `fs.writeFile`.
(((createServer function)))(((request function)))(((event
handling)))(((readable stream)))Readable ((stream))s are a little more
involved. Both the `request` variable that was passed to the HTTP
server's callback function and the `response` variable passed to the
HTTP client are readable streams. (A server reads requests and then
writes responses, whereas a client first writes a request and then
reads a response.) Reading from a stream is done using event handlers,
rather than methods.
(((on method)))(((addEventListener method)))Objects that emit events in
Node have a method called `on` that is similar to the
`addEventListener` method in the browser. You give it an event name
and then a function, and it will register that function to be called
whenever the given event occurs.
(((createReadStream function)))(((data event)))(((end
event)))(((readable stream)))Readable ((stream))s have `"data"` and
`"end"` events. The first is fired every time some data comes in, and
the second is called whenever the stream is at its end. This model is
most suited for “streaming” data, which can be immediately processed,
even when the whole document isn't available yet. A file can be read
as a readable stream by using the `fs.createReadStream` function.
(((upcasing server example)))(((capitalization)))(((toUpperCase
method)))The following code creates a ((server)) that reads request
bodies and streams them back to the client as all-uppercase text:
[source,javascript]
----
var http = require("http");
http.createServer(function(request, response) {
response.writeHead(200, {"Content-Type": "text/plain"});
request.on("data", function(chunk) {
response.write(chunk.toString().toUpperCase());
});
request.on("end", function() {
response.end();
});
}).listen(8000);
----
(((Buffer type)))(((toString method)))The `chunk` variable passed to
the data handler will be a binary `Buffer`, which we can convert to a
string by calling `toString` on it, which will decode it using the
default encoding (UTF-8).
The following piece of code, if run while the uppercasing server is
running, will send a request to that server and write out the response it gets:
// test: no
[source,javascript]
----
var http = require("http");
var request = http.request({
hostname: "localhost",
port: 8000,
method: "POST"
}, function(response) {
response.on("data", function(chunk) {
process.stdout.write(chunk.toString());
});
});
request.end("Hello server");
----
(((stdout property)))(((standard output)))(((writable
stream)))(((console.log)))The example writes to `process.stdout` (the
process’ standard output, as a writable stream) instead of using
`console.log`. We can't use `console.log` because it adds an extra
newline character after each piece of text that it writes, which isn't
appropriate here.
[[file_server]]
== A simple file server ==
(((file server example)))(((Node.js)))Let's combine our newfound
knowledge about ((HTTP)) ((server))s and talking to the ((file
system)) and create a bridge between them: an HTTP server that allows
((remote access)) to a file system. Such a server has many uses. It
allows web applications to store and share data or give a group of
people shared access to a bunch of files.
(((path,URL)))(((GET method)))(((PUT method)))(((DELETE method))) When
we treat ((file))s as HTTP ((resource))s, the HTTP
methods `GET`, `PUT`, and `DELETE` can be used to read, write, and
delete the files, respectively. We will interpret the path in the request
as the path of the file that the request refers to.
(((path,file system)))(((relative path)))We probably don't want to share our whole
file system, so we'll interpret these paths as starting in the
server's working ((directory)), which is the directory in which it was
started. If I ran the server from `/home/marijn/public/` (or
`C:\Users\marijn\public\` on Windows), then a request for `/file.txt`
should refer to `/home/marijn/public/file.txt` (or
`C:\Users\marijn\public\file.txt`).
(((file server example)))(((Node.js)))(((methods object)))We'll build
the program piece by piece, using an object called `methods` to store
the functions that handle the various HTTP methods.
// include_code >code/file_server.js
[source,javascript]
----
var http = require("http"), fs = require("fs");
var methods = Object.create(null);
http.createServer(function(request, response) {
function respond(code, body, type) {
if (!type) type = "text/plain";
response.writeHead(code, {"Content-Type": type});
if (body && body.pipe)
body.pipe(response);
else
response.end(body);
}
if (request.method in methods)
methods[request.method](urlToPath(request.url),
respond, request);
else
respond(405, "Method " + request.method +
" not allowed.");
}).listen(8000);
----
(((405 (HTTP status code))))This starts a server that just returns 405
error responses, which is the code used to indicate that a given
method isn't handled by the server.
(((end method)))(((Content-Type header)))(((response)))(((pipe
method)))(((stream)))The `respond` function is passed to the functions
that handle the various methods and acts as a callback to finish the
request. It takes an HTTP ((status code)), a body, and optionally a
content type as arguments. If the value passed as the body is a ((readable
stream)), it will have a `pipe` method, which is used to forward a
readable stream to a ((writable stream)). If not, it is assumed to be
either `null` (no body) or a string and is passed directly to the
response's `end` method.
(((path,URL)))(((urlToPath function)))(((url module)))(((parsing)))(((escaping,in
URLs)))(((decodeURIComponent function)))To get a path from the
((URL)) in the request, the `urlToPath` function uses Node's built-in
`"url"` module to parse the URL. It takes its pathname, which will be
something like `/file.txt`, decodes that to get rid of the `%20`-style
escape codes, and prefixes a single dot to produce a path relative to
the current directory.
// include_code >code/file_server.js
[source,javascript]
----
function urlToPath(url) {
var path = require("url").parse(url).pathname;
return "." + decodeURIComponent(path);
}
----
(((security)))(((path,file system)))If you are worried about the security of the
`urlToPath` function, you are right. We will return to that in the
exercises.
(((file server example)))(((Node.js)))(((GET method)))We will set up
the `GET` method to return a list of ((file))s when reading a
((directory)) and to return the file's content when reading a regular file.
(((media type)))(((MIME type)))(((Content-Type header)))(((mime
module)))One tricky question is what kind of `Content-Type` header we
should add when returning a file's content. Since these files could be
anything, our server can't simply return the same type for all of
them. But ((NPM)) can help with that. The `mime` package (content type
indicators like `text/plain` are also called _MIME types_) knows the
correct type for a huge number of ((file extension))s.
(((require function)))(((npm program)))If you run the following `npm` command in the directory
where the server script lives, you'll be able to use `require("mime")` to
get access to the library:
----
$ npm install mime
npm http GET https://registry.npmjs.org/mime
npm http 304 https://registry.npmjs.org/mime
[email protected] node_modules/mime
----
(((404 (HTTP status code))))(((stat function)))When a requested file
does not exist, the correct HTTP error code to return is 404. We will
use `fs.stat`, which looks up information on a file, to find out both
whether the ((file)) exists and whether it is a ((directory)).
// include_code >code/file_server.js
[source,javascript]
----
methods.GET = function(path, respond) {
fs.stat(path, function(error, stats) {
if (error && error.code == "ENOENT")
respond(404, "File not found");
else if (error)
respond(500, error.toString());
else if (stats.isDirectory())
fs.readdir(path, function(error, files) {
if (error)
respond(500, error.toString());
else
respond(200, files.join("\n"));
});
else
respond(200, fs.createReadStream(path),
require("mime").lookup(path));
});
};
----
(((createReadStream function)))(((asynchronous programming)))(((error
handling)))(((ENOENT (status code))))(((Error
type)))(((inheritance)))Because it has to touch the disk and thus
might take a while, `fs.stat` is asynchronous. When the file does not
exist, `fs.stat` will pass an error object with a `code` property of
`"ENOENT"` to its callback. It would be nice if Node defined
different subtypes of `Error` for different types of error, but it
doesn't. Instead, it just puts obscure, ((Unix))-inspired codes in
there.
(((500 (HTTP status code))))(((error handling)))(((error response)))We
are going to report any errors we didn't expect with status code 500,
which indicates that the problem exists in the server, as opposed to
codes starting with 4 (such as 404), which refer to bad requests. There
are some situations in which this is not entirely accurate, but for a
small example program like this, it will have to be good enough.
(((Stats type)))(((stat function)))The `stats` object returned by
`fs.stat` tells us a number of things about a ((file)), such as its
size (`size` property) and its ((modification date)) (`mtime`
property). Here we are interested in the question of whether it is a
((directory)) or a regular file, which the `isDirectory` method tells
us.
(((readdir function)))We use `fs.readdir` to read the
list of files in a ((directory)) and, in yet another callback, return
it to the user. For normal files, we create a readable stream with
`fs.createReadStream` and pass it to `respond`, along with the
content type that the `"mime"` module gives us for the file's name.
(((Node.js)))(((file server example)))(((DELETE method)))The code to
handle `DELETE` requests is slightly simpler.
// include_code >code/file_server.js
[source,javascript]
----
methods.DELETE = function(path, respond) {
fs.stat(path, function(error, stats) {
if (error && error.code == "ENOENT")
respond(204);
else if (error)
respond(500, error.toString());
else if (stats.isDirectory())
fs.rmdir(path, respondErrorOrNothing(respond));
else
fs.unlink(path, respondErrorOrNothing(respond));
});
};
----
(((idempotency)))(((error response)))You may be wondering why trying
to delete a nonexistent file returns a 204 status, rather than an
error. When the file that is being deleted is not there, you could say
that the request's objective is already fulfilled. The ((HTTP))
standard encourages people to make requests _idempotent_, which means
that applying them multiple times does not produce a different result.
// include_code >code/file_server.js
[source,javascript]
----
function respondErrorOrNothing(respond) {
return function(error) {
if (error)
respond(500, error.toString());
else
respond(204);
};
}
----
(((204 (HTTP status code))))(((body (HTTP))))When an ((HTTP))
((response)) does not contain any data, the status code 204 (“no
content”) can be used to indicate this. Since we need to provide
callbacks that either report an error or return a 204 response in a
few different situations, I wrote a `respondErrorOrNothing` function
that creates such a callback.
(((file server example)))(((Node.js)))(((PUT method)))This is the
handler for `PUT` requests:
// include_code >code/file_server.js
[source,javascript]
----
methods.PUT = function(path, respond, request) {
var outStream = fs.createWriteStream(path);
outStream.on("error", function(error) {
respond(500, error.toString());
});
outStream.on("finish", function() {
respond(204);
});
request.pipe(outStream);
};
----
(((overwriting)))(((204 (HTTP status code))))(((error
event)))(((finish event)))(((createWriteStream function)))(((pipe
method)))(((stream)))Here, we don't need to check whether the file
exists—if it does, we'll just overwrite it. We again use `pipe` to
move data from a readable stream to a writable one, in this case from
the request to the file. If creating the stream fails, an `"error"`
event is raised for it, which we report in our response. When the data
is transferred successfully, `pipe` will close both streams, which
will cause a `"finish"` event to fire on the writable stream. When
that happens, we can report success to the client with a 204 response.
(((download)))(((file server example)))(((Node.js)))The full script
for the server is available at
http://eloquentjavascript.net/code/file_server.js[_eloquentjavascript.net/code/file_server.js_].
You can download that and run it with Node to start your own file
server. And of course, you can modify and extend it to solve this
chapter's exercises or to experiment.
(((body (HTTP))))(((curl program)))The command-line tool `curl`,
widely available on ((Unix))-like systems, can be used to make ((HTTP))
((request))s. The following session briefly tests our server. Note
that `-X` is used to set the request's ((method)) and `-d` is used to include
a request body.
----
$ curl http://localhost:8000/file.txt
File not found
$ curl -X PUT -d hello http://localhost:8000/file.txt
$ curl http://localhost:8000/file.txt
hello
$ curl -X DELETE http://localhost:8000/file.txt
$ curl http://localhost:8000/file.txt
File not found
----
The first request for `file.txt` fails since the file does not exist
yet. The `PUT` request creates the file, and behold, the next request
successfully retrieves it. After deleting it with a `DELETE` request,
the file is again missing.
== Error handling ==
(((verbosity)))(((error handling)))In the code for the file server,
there are _six_ places where we are explicitly routing exceptions that
we don't know how to handle into ((error response))s. Because
((exception))s aren't automatically propagated to callbacks but
rather passed to them as arguments, they have to be handled explicitly
every time. This completely defeats the advantage of ((exception
handling)), namely, the ability to centralize the handling of
failure conditions.
(((Node.js)))(((stack trace)))(((throw keyword)))(((try
keyword)))(((uncaught exception)))What happens when something actually
_throws_ an exception in this system? Since we are not using any `try`
blocks, the exception will propagate to the top of the call stack. In
Node, that aborts the program and writes information
about the exception (including a stack trace) to the program's
standard error stream.
(((callback function)))This means that our server will ((crash))
whenever a problem is encountered in the server's code itself, as
opposed to asynchronous problems, which will be passed as arguments to
the callbacks. If we wanted to handle all exceptions raised during the
handling of a request, to make sure we send a response, we would
have to add `try/catch` blocks to _every_ callback.
(((exception handling)))This is not workable. Many Node programs are
written to make as little use of exceptions as possible, with the
assumption that if an exception is raised, it is not something the
program can handle, and crashing is the right response.
(((denodeify function)))(((readFile function)))(((promise
module)))(((error handling)))Another approach is to use ((promise))s,
which were introduced in link:17_http.html#promises[Chapter 17]. Those
catch exceptions raised by callback functions and propagate them as
failures. It is possible to load a promise library in Node and use
that to manage your asynchronous control. Few Node libraries
integrate promises, but it is often trivial to wrap them. The
excellent `"promise"` module from ((NPM)) contains a function called
`denodeify`, which takes an asynchronous function like `fs.readFile`
and converts it to a promise-returning function.
[source,javascript]
----
var Promise = require("promise");
var fs = require("fs");
var readFile = Promise.denodeify(fs.readFile);
readFile("file.txt", "utf8").then(function(content) {
console.log("The file contained: " + content);
}, function(error) {
console.log("Failed to read file: " + error);
});
----
(((error handling)))(((exception handling)))(((file server
example)))For comparison, I've written another version of the file
server based on ((promise))s, which you can find at
http://eloquentjavascript.net/code/file_server_promises.js[_eloquentjavascript.net/code/file_server_promises.js_].
It is slightly cleaner because functions can now _return_ their
results, rather than having to call callbacks, and the routing of
exceptions is implicit, rather than explicit.
(((programming style)))I'll list a few lines from the promise-based file
server to illustrate the difference in the style of programming.
(((chaining)))(((fsp object)))The `fsp` object that is used by this
code contains promise-style variants of a number of `fs` functions,
wrapped by `Promise.denodeify`. The object returned from the method handler,
with `code` and `body` properties, will become the final result of the
chain of ((promise))s, and it will be used to determine what kind of
((response)) to send to the client.
// test: no
[source,javascript]
----
methods.GET = function(path) {
return inspectPath(path).then(function(stats) {