Skip to content

Latest commit

 

History

History
65 lines (45 loc) · 4.3 KB

README.md

File metadata and controls

65 lines (45 loc) · 4.3 KB

Multi-source Downloader

The multi-source downloader is a Golang application designed to download a single file in parallel chunks from a public web server. The purpose is to emulate the functionality of protocols like BitTorrent where a file is downloaded in portions from multiple sources. However, this application downloads different parts of the same file from a single source. It is capable of handling large files (60MB or more) and doesn't need to know the file size beforehand.

The application also has the added functionality of verifying the downloaded file, requiring the server to return an etag in a MD5 hash format. This ensures the integrity of the downloaded file.

Directory Structure

.
├── cmd
│   └── main.go
├── docker-compose.yml
├── Dockerfile
├── go.mod
├── go.sum
├── LICENSE
├── pkg
│   ├── downloader
│   │   ├── downloader.go
│   │   └── downloader_test.go
│   └── logger
│       └── logger.go
│       └── logger_test.go
├── README.md
└── .env.example

How to Run

Before running the application, create a .env file at the root of the project and populate it with the necessary values. Refer to .env.example for the required keys.

You can easily run the downloader using Golang or Docker:

Golang

If you do not wish to use Docker or if you have issues with Docker, you can run the application directly on your system using Go:

  1. Navigate to the root directory of the project.
  2. Run the command: go run cmd/main.go

Docker

This application can be run using Docker. Before using Docker, please ensure you have Docker and Docker Compose installed on your system.

  1. Build the Docker image with: docker-compose build
  2. Start the service with: docker-compose up

Note: The Docker configuration creates an output directory on the host machine for the downloaded file. This directory is mounted as a Docker volume and is, by default, owned by the root user because the Docker process runs as root. This means that root permissions are required to access or manage the files in the output directory. If you encounter permission issues while trying to access the downloaded file, you might need to change the file permissions with a command like sudo chown -R $(id -u):$(id -g) ./output or use root permissions to access the files.

Running the Tests

To run the test suite, navigate to the root directory of the project and execute go test ./....

Optimization History & Future Enhancement

The application has undergone several optimization phases and future enhancements are planned:

  1. Previous hacky version: The initial implementation downloaded the file in 4 chunks directly into memory. It was relatively slow with the average download time clocking around 12 minutes per file, depending on the connection speed, server response time, and the host machine's performance.
  2. Previous semi-optimized version: The next iteration downloaded the file in 3 chunks into temporary files. This version improved the speed, reducing the average download time to approximately 5 minutes per file. However, the actual times may still vary based on the connection, server, and host machine.
  3. Current more-optimized version: The current version reuses the HTTP client across chunks, enabling TCP connections to be reused. This has significantly boosted the download speed, bringing it on par with the average speed of a typical browser's download capabilities, at around 2 minutes per file. This time, too, can fluctuate depending on the connection speed, server response time, and the host machine's performance.
  4. Future further optimized version: The next planned iteration will utilize multiple connections, smart chunking, various types of caching, pipelining, and other techniques to approach the maximum download speed of a browser. The aim is to further optimize the download speed, targeting around 1 minute per file. However, it is essential to note that these times will continue to depend on factors such as connection speed, server response times, and host machine performance.

License

This project is licensed under the terms of the Apache License Version 2.0. For more information, see the LICENSE file.