A quick note on package managers

Quick introduction
A package manager is a software tool that automates the process of installing and managing software dependencies. Common examples include:
- npm for Node.js
- pip for Python
- apt for Debian-based Linux distributions
While different package managers differ in terms of syntax and configurations, they pretty much follow the same workflow:
- Define dependencies: Developers define the dependencies required for their project in a configuration file.
- Install dependencies: The package manager automatically downloads and installs the required dependencies, including any additional packages needed to satisfy dependencies.
- Manage dependencies: Developers can use the package manager to update, remove, or add new dependencies to their project.
- Dependency resolution: The package manager ensures that all dependencies are compatible with each other and are installed in a specific order to avoid any conflicts.
Choosing the appropriate package manager for your project is crucial as it plays an essential role in modern software development, which can significantly impact productivity and efficiency.
Comparison of npm and pip
We’ll talk about npm for Node.js and pip for Python here and limit our discussion to these two. Through comparison, we’ll understand some of the subtle things we may have overlooked in the past:
Dependency Configuration
- Node.js uses a package.json file to define dependencies, while Python uses a requirements.txt file.
- The configuration data format could be anything specific to the ecosystem, for example, XML or YAML.
Dependency Resolution
- In Node.js, packages are versioned using Semantic Versioning or SemVer for short.
- SemVer defines a set of rules for how version numbers should be assigned and incremented, based on a three-part numbering system: major.minor.patch.
- To specify dependencies in a Node.js project, you can use the package.json file, which lists all the dependencies of the project along with their version numbers.
- Before updating, it’s a good practice to “freeze” the versions of all dependencies, which locks the versions to the specific ones listed in package-lock.json.
- In Python, packages can be versioned using either exact version numbers or version ranges.
- To specify dependencies in a Python project, you can use the requirements.txt file, which lists all the project’s dependencies along with their version numbers or version ranges.
- Before updating, it’s a good practice to “freeze” the versions of all dependencies by running
pip freeze > requirements.txt
.
Dependency Isolation
- The goal is to manage dependencies in a software project in a way that ensures they are separate and isolated from each other, as well as from the system’s dependencies.
- This helps to prevent conflicts between dependencies, and ensures that the project can be easily reproduced on different systems.
- In Node.js,
package-lock.json
is used to lock down exact package versions and prevent updates to dependencies without explicit approval. - In Python, virtual environments can be used to create sandboxed environments for each project, with its own set of dependencies.
To summarize, in order to update packages in both Node.js and Python projects, it is crucial to “freeze” the versions of all dependencies by generating a lock file (i.e., package-lock.json for Node.js or requirements.txt for Python) that lists the exact versions of all dependencies. These lock files should be committed to version control along with the corresponding package.json or setup.py file, respectively, to ensure that everyone working on the project uses the same set of dependencies.
That is it for now, thanks for giving it a read.