* feat(tools): add seed/solution restore script * chore(curriculum): remove empty sections' markers * chore(curriculum): add seed + solution to Chinese * chore: remove old formatter * fix: update getChallenges parse translated challenges separately, without reference to the source * chore(curriculum): add dashedName to English * chore(curriculum): add dashedName to Chinese * refactor: remove unused challenge property 'name' * fix: relax dashedName requirement * fix: stray tag Remove stray `pre` tag from challenge file. Signed-off-by: nhcarrigan <nhcarrigan@gmail.com> Co-authored-by: nhcarrigan <nhcarrigan@gmail.com>
		
			
				
	
	
		
			80 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			80 lines
		
	
	
		
			2.5 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| ---
 | ||
| id: 587d7fb1367417b2b2512bf4
 | ||
| title: Chain Middleware to Create a Time Server
 | ||
| challengeType: 2
 | ||
| forumTopicId: 301510
 | ||
| dashedName: chain-middleware-to-create-a-time-server
 | ||
| ---
 | ||
| 
 | ||
| # --description--
 | ||
| 
 | ||
| Middleware can be mounted at a specific route using `app.METHOD(path, middlewareFunction)`. Middleware can also be chained inside route definition.
 | ||
| 
 | ||
| Look at the following example:
 | ||
| 
 | ||
| ```js
 | ||
| app.get('/user', function(req, res, next) {
 | ||
|   req.user = getTheUserSync();  // Hypothetical synchronous operation
 | ||
|   next();
 | ||
| }, function(req, res) {
 | ||
|   res.send(req.user);
 | ||
| });
 | ||
| ```
 | ||
| 
 | ||
| This approach is useful to split the server operations into smaller units. That leads to a better app structure, and the possibility to reuse code in different places. This approach can also be used to perform some validation on the data. At each point of the middleware stack you can block the execution of the current chain and pass control to functions specifically designed to handle errors. Or you can pass control to the next matching route, to handle special cases. We will see how in the advanced Express section.
 | ||
| 
 | ||
| # --instructions--
 | ||
| 
 | ||
| In the route `app.get('/now', ...)` chain a middleware function and the final handler. In the middleware function you should add the current time to the request object in the `req.time` key. You can use `new Date().toString()`. In the handler, respond with a JSON object, taking the structure `{time: req.time}`.
 | ||
| 
 | ||
| **Note:** The test will not pass if you don’t chain the middleware. If you mount the function somewhere else, the test will fail, even if the output result is correct.
 | ||
| 
 | ||
| # --hints--
 | ||
| 
 | ||
| The /now endpoint should have mounted middleware
 | ||
| 
 | ||
| ```js
 | ||
| (getUserInput) =>
 | ||
|   $.get(getUserInput('url') + '/_api/chain-middleware-time').then(
 | ||
|     (data) => {
 | ||
|       assert.equal(
 | ||
|         data.stackLength,
 | ||
|         2,
 | ||
|         '"/now" route has no mounted middleware'
 | ||
|       );
 | ||
|     },
 | ||
|     (xhr) => {
 | ||
|       throw new Error(xhr.responseText);
 | ||
|     }
 | ||
|   );
 | ||
| ```
 | ||
| 
 | ||
| The /now endpoint should return a time that is +/- 20 secs from now
 | ||
| 
 | ||
| ```js
 | ||
| (getUserInput) =>
 | ||
|   $.get(getUserInput('url') + '/_api/chain-middleware-time').then(
 | ||
|     (data) => {
 | ||
|       var now = new Date();
 | ||
|       assert.isAtMost(
 | ||
|         Math.abs(new Date(data.time) - now),
 | ||
|         20000,
 | ||
|         'the returned time is not between +- 20 secs from now'
 | ||
|       );
 | ||
|     },
 | ||
|     (xhr) => {
 | ||
|       throw new Error(xhr.responseText);
 | ||
|     }
 | ||
|   );
 | ||
| ```
 | ||
| 
 | ||
| # --solutions--
 | ||
| 
 | ||
| ```js
 | ||
| /**
 | ||
|   Backend challenges don't need solutions, 
 | ||
|   because they would need to be tested against a full working project. 
 | ||
|   Please check our contributing guidelines to learn more.
 | ||
| */
 | ||
| ```
 |