53 в 1: сборник лучших практик Node.js

Node.js API Express ESLint
53 в 1: сборник лучших практик Node.js

Node.js Best Practices


53 items

Last update: Nov 15, 2017

Updated for Node v.8.9


nodepractices Follow us on Twitter! @nodepractices

Welcome! 3 Things You Ought To Know First:

1. When you read here, you in fact read dozens of the best Node.JS articles - this is a summary and curation of the top-ranked content on Node JS best practices

2. It is the largest compilation, and it is growing every week - currently, more than 50 best practices, style guides, and architectural tips are presented. New issues and PR are created every day to keep this live book updated. We'd love to see you contributing here, whether fixing some code mistake or suggesting brilliant new ideas. See our milestones here

3. Most bullets have additional info - nearby most best practice bullets you'll find :link:Read More link that will present you with code examples, quotes from selected blogs and more info




Table of Contents

  1. Project structure Practices (5)ote
  2. Error Handling Practices (11)
  3. Code Style Practices (12)
  4. Testing And Overall Quality Practices (8)
  5. Going To Production Practices (16)
  6. Security Practices (coming soon)
  7. Performance Practices (coming soon)




1. Project Structure Practices

✔ 1.1 Structure your solution by components

TL;DR: The worst large applications pitfall is maintaining a huge code base with hundreds of dependencies - such a monolith slows down developers as they try to incorporate new features. Instead, partition your code into components, each gets its own folder or a dedicated codebase, and ensure that each unit is kept small and simple. Visit 'Read More' below to see examples of correct project structure

Otherwise: When developers who code new features struggle to realize the impact of their change and fear to break other dependant components - deployments become slower and more risky. It's also considered harder to scale-out when all the business units are not separated

:link: Read More: structure by components



✔ 1.2 Layer your components, keep Express within its boundaries

TL;DR: Each component should contain 'layers' - a dedicated object for the web, logic and data access code. This not only draws a clean separation of concerns but also significantly eases mocking and testing the system. Though this is a very common pattern, API developers tend to mix layers by passing the web layer objects (Express req, res) to business logic and data layers - this makes your application dependant on and accessible by Express only

Otherwise: App that mixes web objects with other layers can not be accessed by testing code, CRON jobs and other non-Express callers

:link: Read More: layer your app



✔ 1.3 Wrap common utilities as NPM packages

TL;DR: In a large app that constitutes a large code base, cross-cutting-concern utilities like logger, encryption and alike, should be wrapped by your own code and exposed as private NPM packages. This allows sharing them among multiple code bases and projects

Otherwise: You'll have to invent your own deployment and dependency wheel

:link: Read More: Structure by feature



✔ 1.4 Separate Express 'app' and 'server'

TL;DR: Avoid the nasty habit of defining the entire Express app in a single huge file - separate your 'Express' definition to at least two files: the API declaration (app.js) and the networking concerns (WWW). For even better structure, locate your API declaration within components

Otherwise: Your API will be accessible for testing via HTTP calls only (slower and much harder to generate coverage reports). It probably won't be a big pleasure to maintain hundreds of lines of code in a single file

:link: Read More: separate Express 'app' and 'server'



✔ 1.5 Use environment aware, secure and hierarchical config

TL;DR: A perfect and flawless configuration setup should ensure (a) keys can be read from file AND from environment variable (b) secrets are kept outside committed code (c) config is hierarchical for easier findability. There are a few packages that can help tick most of those boxes like rc, nconf and config.

Otherwise: Failing to satisfy any of the config requirements will simply bog down the development or devops team. Probably both

:link: Read More: configuration best practices




⬆ Вернуться к началу

2. Error Handling Practices

✔ 2.1 Use Async-Await or promises for async error handling

TL;DR: Handling async errors in callback style is probably the fastest way to hell (a.k.a the pyramid of doom). The best gift you can give to your code is using a reputable promise library or async-await instead which enables a much more compact and familiar code syntax like try-catch

Otherwise: Node.JS callback style, function(err, response), is a promising way to un-maintainable code due to the mix of error handling with casual code, excessive nesting and awkward coding patterns

:link: Read More: avoiding callbacks



✔ 2.2 Use only the built-in Error object

TL;DR:Многие выдают ошибки в виде строки или в виде некоторого пользовательского типа — это усложняет логику обработки ошибок и взаимодействие между модулями. Объект ошибки повысит единообразие и предотвратит потерю информации

Otherwise:При вызове какого-либо компонента неуверенность в том, какие типы ошибок возвращаются, — это значительно усложняет правильную обработку ошибок, а использование пользовательских типов для описания ошибок может привести к потере критической ошибки. такая информация, как трассировка стека!

:link: Read More: using the built-in error object



✔ 2.3 Distinguish operational vs programmer errors

TL;DR: Operational errors (e.g. API received an invalid input) refer to known cases where the error impact is fully understood and can be handled thoughtfully. On the other hand, programmer error (e.g. trying to read undefined variable) refers to unknown code failures that dictate to gracefully restart the application

Otherwise:Вы всегда можете перезапустить приложение при появлении ошибки, но зачем подводить около 5000 онлайн-пользователей из-за незначительной, прогнозируемой операционной ошибки? Неизвестная проблема (ошибка программиста) может привести к непредвиденному поведению. Их различие позволяет действовать тактично и применять сбалансированный подход, основанный на данном контексте.

:link: Read More: operational vs programmer error



✔ 2.4 Handle errors centrally, not within an Express middleware

TL;DR: Error handling logic such as mail to admin and logging should be encapsulated in a dedicated and centralized object that all endpoints (e.g. Express middleware, cron jobs, unit-testing) call when an error comes in.

Otherwise: Not handling errors within a single place will lead to code duplication and probably to improperly handled errors

:link: Read More: handling errors in a centralized place



✔ 2.5 Document API errors using Swagger

TL;DR: Let your API callers know which errors might come in return so they can handle these thoughtfully without crashing. This is usually done with REST API documentation frameworks like Swagger

Otherwise:Клиент API может решить аварийно завершить работу и перезапуститься только потому, что он получил ошибку, которую не может понять.Примечание: вызывающим API может быть вы (очень типично в среде микросервисов).

:link: Read More: documenting errors in Swagger



✔ 2.6 Shut the process gracefully when a stranger comes to town

TL;DR:Когда возникает неизвестная ошибка (ошибка разработчика, см. передовой опыт № 3) — возникает неуверенность в работоспособности приложения. Общепринятая практика предполагает осторожный перезапуск процесса с помощью инструмента «перезапуск». Как навсегда и PM2

Otherwise: When an unfamiliar exception is caught, some object might be in a faulty state (e.g an event emitter which is used globally and not firing events anymore due to some internal failure) and all future requests might fail or behave crazily

:link: Read More: shutting the process



✔ 2.7 Use a mature logger to increase error visibility

TL;DR: A set of mature logging tools like Winston, Bunyan or Log4J, will speed-up error discovery and understanding. So forget about console.log.

Otherwise: Skimming through console.logs or manually through messy text file without querying tools or a decent log viewer might keep you busy at work until late

:link: Read More: using a mature logger



✔ 2.8 Test error flows using your favorite test framework

TL;DR:Будь то профессиональный автоматизированный контроль качества или обычное ручное тестирование разработчиков — убедитесь, что ваш код не только удовлетворяет положительному сценарию, но также обрабатывает и возвращает правильные ошибки. справиться с этим легко (см. примеры кода во всплывающем окне Gist)

Otherwise:Без тестирования, будь то автоматическое или ручное, вы не можете полагаться на то, что наш код будет возвращать правильные ошибки.Без значимых ошибок — нет обработки ошибок.

:link: Read More: testing error flows



✔ 2.9 Discover errors and downtime using APM products

TL;DR: Monitoring and performance products (a.k.a APM) proactively gauge your codebase or API so they can auto-magically highlight errors, crashes and slow parts that you were missing

Otherwise:Вы можете потратить огромные усилия на измерение производительности и времени простоя API, возможно, вы никогда не узнаете, какие части вашего кода самые медленные в реальных сценариях и как они влияют на UX.

:link: Read More: using APM products



✔ 2.10 Catch unhandled promise rejections

TL;DR:Любое исключение, выброшенное внутри промиса, будет проглочено и отброшено, если только разработчик не забыл явно обработать его.Даже если ваш код подписан на process.uncaughtException! Преодолейте это, зарегистрировавшись на событие process.unhandledRejection

Otherwise: Your errors will get swallowed and leave no trace. Nothing to worry about

:link: Read More: catching unhandled promise rejection



✔ 2.11 Fail fast, validate arguments using a dedicated library

TL;DR:Это должно быть частью ваших передовых практик в Express — Assert API input, чтобы избежать неприятных ошибок, которые гораздо сложнее отследить позже Код проверки обычно утомителен, если только вы не используете очень классные вспомогательные библиотеки, такие как Joi.

Otherwise:Учтите следующее: ваша функция ожидает числовой аргумент «Скидка», который вызывающая сторона забывает передать, позже ваш код проверяет, если Скидка!=0 (сумма разрешенной скидки больше нуля), тогда она разрешает пользователь, чтобы насладиться скидкой OMG, какая неприятная ошибка. Вы видите это?

:link: Read More: failing fast




⬆ Вернуться к началу

3. Code Style Practices

✔ 3.1 Use ESLint

TL;DR: ESLint is the de-facto standard for checking code style, not only to identify nitty-gritty spacing issues but also to detect serious code anti-patterns like developers throwing errors without classification. Using ESLint and following the rest of the code style practices below means following the same styles used by the rest of the community, as well as the same code styles used in the core products themselves.

Otherwise: developers will focus on tedious spacing and line-width concerns



✔ 3.2 Node JS Specific Plugins

TL;DR: On top of ESLint standard rules that cover vanilla JS only, add Node-specific plugins like eslint-plugin-node, eslint-plugin-mocha and eslint-plugin-node-security

Otherwise: Many faulty Node.JS code patterns might escape under the radar. For example, developers might require(variableAsPath) files with a variable given as path which allows attackers to execute any JS script. Node.JS linters can detect such patterns and complain early



✔ 3.3 Start a Codeblock's Curly Braces in the Same Line

TL;DR: The opening curly braces of a code block should be in the same line of the opening statement.

Code Example

  // Do
  function someFunction() {
    // code block
  }

  // Avoid
  function someFunction()
  {
    // code block
  }

Otherwise: Deferring from this best practice might lead to unexpected results, as seen in the Stackoverflow thread below:

:link: Read more: "Why does a results vary based on curly brace placement?" (Stackoverflow)



✔ 3.4 Don't Forget the Semicolon

TL;DR: While not unanimously agreed upon, it is still recommended to put a semicolon at the end of each statement. This will make your code more readable and explicit to other developers who read it.

Otherwise: As seen in the previous section, JavaScript's interpreter automatically adds a semicolon at the end of a statement if there isn't one which might lead to some undesired results.



✔ 3.5 Name Your Functions

TL;DR: Name all functions, including closures and callbacks. Avoid anonymous functions. This is especially useful when profiling a node app. Naming all functions will allow you to easily understand what you're looking at when checking a memory snapshot.

Otherwise: Debugging production issues using a core dump (memory snapshot) might become challenging as you notice significant memory consumption from anonymous functions.



✔ 3.6 Naming conventions for variables, constants, functions and classes

TL;DR: Use lowerCamelCase when naming variables and functions, UpperCamelCase (capital first letter as well) when naming classes and UPPERCASE for constants. This will help you to easily distinguish between plain variables / functions, and classes that require instantiation. Use descriptive names, but try to keep them short.

Otherwise: Javascript is the only language in the world which allows to invoke a constructor ("Class") directly without instantiating it first. Consequently, Classes and function-constructors are differentiated by starting with UpperCamelCase.

Code Example

  // for class name we use UpperCamelCase
  class SomeClassExample { 
    
    // for const name we use UPPERCASE
    const CONFIG = {
      key: 'value'
    };
    
    // for variables and functions names we use lowerCamelCase
    let someVariableExample = 'value';
    function doSomething() {
      
    }

  }



✔ 3.7 Prefer const over let. Ditch the var

TL;DR: Using const means that once a variable is assigned, it cannot be reassigned. Preferring const will help you to not be tempted to use the same variable for different uses, and make your code clearer. If a variable needs to be reassigned, in a for loop for example, use let to declare it. Another important aspect of let is that a variable declared using let is only available in the block scope in which it was defined. var is function scoped, not block scoped, and shouldn't be used in ES6 now that you have const and let at your disposal.

Otherwise: Debugging becomes way more cumbersome when following a variable that frequently changes.

:link: Read more: JavaScript ES6+: var, let, or const?



✔ 3.8 Requires come first, and not inside functions

TL;DR: Require modules at the beginning of each file, before and outside of any functions. This simple best practice will not only help you easily and quickly tell the dependencies of a file right at the top, but also avoids a couple of potential problems.

Otherwise: Requires are run synchronously by NodeJS. If they are called from within a function, it may block other requests from being handled at a more critical time. Also, if a required module or any of its own dependencies throw an error and crash the server, it is best to find out about it as soon as possible, which might not be the case if that module is required from within a function.



✔ 3.9 Do Require on the folders, not directly on the files

TL;DR: When developing a module/library in a folder, place an index.js file that exposes the module's internals so every consumer will pass through it. This serves as an 'interface' to your module and ease future changes without breaking the contract.

Otherwise: Changing to the internal structure of files or the signature may break the interface with clients.

Code example

  // Do
  module.exports.SMSProvider = require('./SMSProvider');
  module.exports.SMSNumberResolver = require('./SMSNumberResolver');

  // Avoid
  module.exports.SMSProvider = require('./SMSProvider/SMSProvider.js');
  module.exports.SMSNumberResolver = require('./SMSNumberResolver/SMSNumberResolver.js');



✔ 3.10 Use the === operator

TL;DR: Prefer the strict equality operator === over the weaker abstract equality operator ==. == will compare two variables after converting them to a common type. There is no type conversion in ===, and both variables must be of the same type to be equal.

Otherwise: Unequal variables might return true when compared with the == operator.

Code example

'' == '0'           // false
0 == ''             // true
0 == '0'            // true

false == 'false'    // false
false == '0'        // true

false == undefined  // false
false == null       // false
null == undefined   // true

' \t\r\n ' == 0     // true

All statements above will return false if used with ===



✔ 3.11 Use Async Await, avoid callbacks

TL;DR: Node 8 LTS now has full support for Async-await. This is a new way of dealing with asynchronous code which supersedes callbacks and promises. Async-await is non-blocking, and it makes asynchronous code look synchronous. The best gift you can give to your code is using async-await which provides a much more compact and familiar code syntax like try-catch.

Otherwise: Handling async errors in callback style is probably the fastest way to hell - this style forces to check errors all over, deal with akward code nesting and make it difficult to reason about the code flow.

:link:Read more: Guide to async await 1.0



✔ 3.12 Use Fat (=>) Arrow Functions

TL;DR: Though it's recommended to use async-await and avoid function parameters, when dealing with older API that accept promises or callbacks - arrow functions make the code structure more compact and keep the lexical context of the root function (i.e. 'this').

Otherwise: Longer code (in ES5 functions) is more prone to bugs and cumbersome to read.

:link: Режим чтения: пришло время использовать функции стрелок




⬆ Вернуться к началу

4. Testing And Overall Quality Practices

✔ 4.1 At the very least, write API (component) testing

TL;DR: Most projects just don't have any automated testing due to short time tables or often the 'testing project' run out of control and being abandoned. For that reason, prioritize and start with API testing which are the easiest to write and provide more coverage than unit testing (you may even craft API tests without code using tools like Postman. Afterwards, should you have more resources and time, continue with advanced test types like unit testing, DB testing, performance testing, etc

Otherwise: You may spend long days on writing unit tests to find out that you got only 20% system coverage



✔ 4.2 Detect code issues with ESLint + specific Node plugin rules

TL;DR: ESLint is the de-facto standard for checking code style, not only to identify nitty-gritty spacing issues but also to detect serious code anti-patterns like developers throwing errors without classification. On top of ESLint standard rules that cover vanilla JS only, add Node-specific plugins like eslint-plugin-node, eslint-plugin-mocha and eslint-plugin-node-security

Otherwise: Many faulty Node.JS code patterns might escape under the radar. For example, developers might require(variableAsPath) files with a variable given as path which allows attackers to execute any JS script. Node.JS linters can detect such patterns and complain early



✔ 4.3 Carefully choose your CI platform (Jenkins vs CircleCI vs Travis vs Rest of the world)

TL;DR: Your continuous integration platform (CICD) will host all the quality tools (e.g test, lint) so it should come with a vibrant ecosystem of plugins. Jenkins used to be the default for many projects as it has the biggest community along with a very powerful platform at the price of complex setup that demands a steep learning curve. Nowadays, it became much easier to setup a CI solution using SaaS tools like CircleCI and others. These tools allow crafting a flexible CI pipeline without the burden of managing the whole infrastructure. Eventually, it's a trade-off between robustness and speed - choose your side carefully.

Otherwise: Choosing some niche vendor might get you blocked once you need some advanced customization. On the other hand, going with Jenkins might burn precious time on infrastructure setup

:link: Read More: Choosing CI platform



✔ 4.4 Constantly inspect for vulnerable dependencies

TL;DR: Even the most reputable dependencies such as Express have known vulnerabilities. This can get easily tamed using community and commercial tools such as :link: nsp that can be invoked from your CI on every build

Otherwise: Keeping your code clean from vulnerabilities without dedicated tools will require to constantly follow online publications about new threats. Quite tedious



✔ 4.5 Tag your tests

TL;DR: Different tests must run on different scenarios: quick smoke, IO-less, tests should run when a developer saves or commits a file, full end-to-end tests usually run when a new pull request is submitted, etc. This can be achieved by tagging tests with keywords like #cold #api #sanity so you can grep with your testing harness and invoke the desired subset. For example, this is how you would invoke only the sanity test group with Mocha: mocha --grep 'sanity'

Otherwise: Running all the tests, including tests that perform dozens of DB queries, any time a developer makes a small change can be extremely slow and keeps developers away from running tests



✔ 4.6 Check your test coverage, it helps to identify wrong test patterns

TL;DR: Code coverage tools like Istanbul/NYC are great for 3 reasons: it comes for free (no effort is required to benefit this reports), it helps to identify a decrease in testing coverage, and last but not least it highlights testing mismatches: by looking at colored code coverage reports you may notice, for example, code areas that are never tested like catch clauses (meaning that tests only invoke the happy paths and not how the app behaves on errors). Set it to fail builds if the coverage falls under a certain threshold

Otherwise: There won't be any automated metric telling you when a large portion of your code is not covered by testing



✔ 4.7 Inspect for outdated packages

TL;DR: Use your preferred tool (e.g. 'npm outdated' or npm-check-updates to detect installed packages which are outdated, inject this check into your CI pipeline and even make a build fail in a severe scenario. For example, a severe scenario might be when an installed package is 5 patch commits behind (e.g. local version is 1.3.1 and repository version is 1.3.8) or it is tagged as deprecated by its author - kill the build and prevent deploying this version

Otherwise: Your production will run packages that have been explicitly tagged by their author as risky



✔ 4.8 Use docker-compose for e2e testing

TL;DR: End to end (e2e) testing which includes live data used to be the weakest link of the CI process as it depends on multiple heavy services like DB. Docker-compose turns this problem into a breeze by crafting production-like environment using a simple text file and easy commands. It allows crafting all the dependent services, DB and isolated network for e2e testing. Last but not least, it can keep a stateless environment that is invoked before each test suite and dies right after

Otherwise: Without docker-compose teams must maintain a testing DB for each testing environment including developers machines, keep all those DBs in sync so test results won't vary across environments




⬆ Вернуться к началу

5. Going To Production Practices

✔ 5.1. Monitoring!

TL;DR:Мониторинг — это игра, в которой нужно выявлять проблемы до того, как это сделают клиенты — очевидно, этому следует придавать беспрецедентное значение. должны следовать (мои предложения внутри), затем просмотрите дополнительные необычные функции и выберите решение, которое отвечает всем требованиям. Нажмите «Суть» ниже, чтобы просмотреть обзор решений.

Otherwise: Failure === disappointed customers. Simple.

:link: Read More: Monitoring!



✔ 5.2. Increase transparency using smart logging

TL;DR: Logs can be a dumb warehouse of debug statements or the enabler of a beautiful dashboard that tells the story of your app. Plan your logging platform from day 1: how logs are collected, stored and analyzed to ensure that the desired information (e.g. error rate, following an entire transaction through services and servers, etc) can really be extracted

Otherwise: You end-up with a blackbox that is hard to reason about, then you start re-writing all logging statements to add additional information

:link: Read More: Increase transparency using smart logging



✔ 5.3. Delegate anything possible (e.g. gzip, SSL) to a reverse proxy

TL;DR:Node ужасно плохо справляется с задачами, интенсивно использующими процессор, такими как gzip, завершение SSL и т. д. Вместо этого используйте «настоящие» промежуточные сервисы, такие как nginx, HAproxy или сервисы облачных поставщиков.

Otherwise: Your poor single thread will keep busy doing networking tasks instead of dealing with your application core and performance will degrade accordingly

:link: Read More: Delegate anything possible (e.g. gzip, SSL) to a reverse proxy



✔ 5.4. Lock dependencies

TL;DR:Ваш код должен быть идентичен во всех средах, но удивительно, что NPM позволяет зависимостям дрейфовать между средами по умолчанию — когда вы устанавливаете пакеты в разных средах, он пытается получить последний патч для пакетов. версии. Преодолейте это с помощью файлов конфигурации NPM, .npmrc, которые предписывают каждой среде сохранять точную (не самую последнюю) версию каждого пакета. В качестве альтернативы, для более точного контроля используйте NPM «shrinkwrap». * Обновление: начиная с NPM5, зависимости заблокированы по умолчанию.Новый менеджер пакетов в городе, Yarn, также обеспечивает нас по умолчанию

Otherwise: QA will thoroughly test the code and approve a version that will behave differently at production. Even worse, different servers at the same production cluster might run different code

:link: Read More: Lock dependencies



✔ 5.5. Guard process uptime using the right tool

TL;DR:Процесс должен продолжаться и перезапускаться при сбоях.Для простого сценария инструментов «перезапуска», таких как PM2, может быть достаточно, но в современном «докеризованном» мире следует также рассмотреть инструменты управления кластером.

Otherwise: Running dozens of instances without clear strategy and too many tools together (cluster management, docker, PM2) might lead to a devops chaos

:link: Read More: Guard process uptime using the right tool



✔ 5.6. Utilize all CPU cores

TL;DR:В своей базовой форме приложение Node работает на одном ядре ЦП, а все остальные остаются бездействующими.Ваша обязанность — реплицировать процесс Node и использовать все ЦП — для приложений малого и среднего размера вы можете использовать кластер узла или PM2. app рассмотрите возможность репликации процесса с использованием некоторого кластера Docker (например, K8S, ECS) или сценариев развертывания, основанных на системе инициализации Linux (например, systemd).

Otherwise: Your app will likely utilize only 25% of its available resources(!) or even less. Note that a typical server has 4 CPU cores or more, naive deployment of Node.JS utilizes only 1 (even using PaaS services like AWS beanstalk!)

:link: Read More: Utilize all CPU cores



✔5.7. Создайте «конечную точку обслуживания»

TL;DR:Предоставляйте набор системной информации, такой как использование памяти, REPL и т. д., в защищенном API.Хотя настоятельно рекомендуется полагаться на стандартные инструменты и инструменты для боевых тестов, некоторая ценная информация и операции проще сделано с помощью кода

Otherwise:Вы обнаружите, что выполняете много «диагностических развертываний» — отправляя код в производство только для извлечения некоторой информации в диагностических целях.

:link: Подробнее: Создайте «конечную точку обслуживания»



✔ 5.8. Discover errors and downtime using APM products

TL;DR: Monitoring and performance products (a.k.a APM) proactively gauge codebase and API so they can auto-magically go beyond traditional monitoring and measure the overall user-experience across services and tiers. For example, some APM products can highlight a transaction that loads too slow on the end-users side while suggesting the root cause

Otherwise:Вы можете потратить огромные усилия на измерение производительности и времени простоя API, возможно, вы никогда не узнаете, какие части вашего кода самые медленные в реальном сценарии и как они влияют на UX.

:link: Read More: Discover errors and downtime using APM products



✔ 5.9. Make your code production-ready

TL;DR:Пишите с учетом конечного результата, планируйте производство с первого дня. Это звучит немного расплывчато, поэтому я собрал несколько советов по разработке, которые тесно связаны с обслуживанием производства (нажмите «Существо» ниже).

Otherwise:Чемпион мира по IT/devops не спасет плохо написанную систему

:link: Read More: Make your code production-ready



✔ 5.10. Measure and guard the memory usage

TL;DR:Node.js имеет противоречивые отношения с памятью: движок v8 имеет мягкие ограничения на использование памяти (1,4 ГБ), и в коде Node есть известные пути утечки памяти, поэтому необходимо следить за памятью процесса Node. приложениях вы можете периодически измерять память с помощью команд оболочки, но в приложениях среднего размера рассмотрите возможность превращения ваших часов памяти в надежную систему мониторинга

Otherwise: Your process memory might leak a hundred megabytes a day like happened in Wallmart

:link: Read More: Measure and guard the memory usage



✔ 5.11. Get your frontend assets out of Node

TL;DR: Serve frontend content using dedicated middleware (nginx, S3, CDN) because Node performance really gets hurt when dealing with many static files due to its single threaded model

Otherwise:Ваш единственный поток Node будет занят потоковой передачей сотен файлов html/images/angular/react вместо того, чтобы выделять все свои ресурсы для задачи, для которой он был создан — обслуживание динамического контента.

:link: Read More: Get your frontend assets out of Node



✔ 5.12. Be stateless, kill your Servers almost every day

TL;DR:Храните данные любого типа (например, сеансы пользователей, кеш, загруженные файлы) во внешних хранилищах данных. Рассмотрите возможность периодического отключения ваших серверов или использования «бессерверной» платформы (например, AWS Lambda), которая явно обеспечивает поведение

Otherwise: Failure at a given server will result in application downtime instead of just killing a faulty machine. Moreover, scaling-out elasticity will get more challenging due to the reliance on a specific server

:link: Read More: Be stateless, kill your Servers almost every day



✔ 5.13. Use tools that automatically detect vulnerabilities

TL;DR: Even the most reputable dependencies such as Express have known vulnerabilities (from time to time) that can put a system at risk. This can get easily tamed using community and commercial tools that constantly check for vulnerabilities and warn (locally or at GitHub), some can even patch them immediately

Otherwise: Otherwise: Keeping your code clean from vulnerabilities without dedicated tools will require to constantly follow online publications about new threats. Quite tedious

:link: Read More: Use tools that automatically detect vulnerabilities



✔5.14. Назначьте TransactionId каждому оператору журнала.

TL;DR: Assign the same identifier, transaction-id: {some value}, to each log entry within a single request. Then when inspecting errors in logs, easily conclude what happened before and after. Unfortunately, this is not easy to achieve in Node due its async nature, see code examples inside

Otherwise:Глядя на журнал производственных ошибок без контекста — то, что происходило раньше, — становится намного сложнее и медленнее рассуждать о проблеме.

:link: Подробнее Назначьте: «TransactionId» для каждого оператора журнала.



✔ 5.15. Set NODE_ENV=production

TL;DR:Установите для переменной среды NODE_ENV значение «производство» или «разработка», чтобы указать, следует ли активировать оптимизацию производства — многие пакеты NPM определяют текущую среду и оптимизируют свой код для производства.

Otherwise: Omitting this simple property might greatly degrade performance. For example, when using Express for server side rendering omitting NODE_ENV makes the slower by a factor of three!

:link: Read More: Set NODE_ENV=production



✔ 5.16. Design automated, atomic and zero-downtime deployments

TL;DR:Исследования показывают, что команды, выполняющие много развертываний, снижают вероятность серьезных производственных проблем Быстрое и автоматизированное развертывание, не требующее рискованных ручных действий и простоя службы, значительно улучшается процесс развертывания. Вероятно, вам следует добиться этого, используя Docker в сочетании с инструментами CI, поскольку они стали отраслевым стандартом для упрощенного развертывания.

Otherwise: Long deployments -> production down time & human-related error -> team unconfident and in making deployment -> less deployments and features




⬆ Вернуться к началу

Security Practices

Our contributors are working on this section. Would you like to join?




Performance Practices

Our contributors are working on this section. Would you like to join?



Contributors

Yoni Goldberg

Independent Node.JS consultant who works with customers at USA, Europe and Israel on building large-scale scalable Node applications. Many of the best practices above were first published on his blog post at www.goldbergyoni.com. Reach Yoni at @goldbergyoni or me@goldbergyoni.com

Ido Richter

:man_technologist: Software engineer, :globe_with_meridians: web developer, :robot: emojis enthusiast.

Refael Ackermann @refack <refack@gmail.com> (he/him)

Node.js Core Collaborator, been noding since 0.4, and have noded in multiple production sites. Founded node4good home of lodash-contrib, formage, and asynctrace. refack on freenode, Twitter, GitHub, GMail, and many other platforms. DMs are open, happy to help.

Bruno Scheufler

:computer: full-stack web developer and Node.js enthusiast.