seaweedfs/README.md

456 lines
20 KiB
Markdown
Raw Normal View History

# SeaweedFS
2014-07-16 09:07:16 +08:00
2015-04-17 04:44:08 +08:00
[![Build Status](https://travis-ci.org/chrislusf/seaweedfs.svg?branch=master)](https://travis-ci.org/chrislusf/seaweedfs)
[![GoDoc](https://godoc.org/github.com/chrislusf/seaweedfs/weed?status.svg)](https://godoc.org/github.com/chrislusf/seaweedfs/weed)
[![Wiki](https://img.shields.io/badge/docs-wiki-blue.svg)](https://github.com/chrislusf/seaweedfs/wiki)
2017-01-16 04:33:38 +08:00
2017-01-16 04:34:32 +08:00
![SeaweedFS Logo](https://raw.githubusercontent.com/chrislusf/seaweedfs/master/note/seaweedfs.png)
2017-09-15 14:25:42 +08:00
<h2 align="center">Supporting SeaweedFS</h2>
SeaweedFS is Apache-licensed open source project, independent project with its ongoing development made
possible entirely thanks to the support by these awesome [backers](https://github.com/chrislusf/seaweedfs/blob/master/backers.md).
2017-09-15 14:49:09 +08:00
If you'd like to grow SeaweedFS even stronger, please consider to
<a href="https://www.patreon.com/seaweedfs">Sponsor SeaweedFS via Patreon</a>.
Platinum ($2500/month), Gold ($500/month): put your company logo on the SeaweedFS github page
Generous Backer($50/month), Backer($10/month): put your name on the SeaweedFS backer page.
Your support will be really appreciated by me and other supporters!
2017-09-15 14:25:42 +08:00
<h3 align="center"><a href="https://www.patreon.com/seaweedfs">Sponsors SeaweedFS via Patreon</a></h3>
<h4 align="center">Platinum</h4>
<p align="center">
<a href="" target="_blank">
Add your name or icon here
</a>
</p>
<h4 align="center">Gold</h4>
<table>
<tbody>
<tr>
<td align="center" valign="middle">
<a href="" target="_blank">
Add your name or icon here
</a>
</td>
</tr>
<tr></tr>
</tbody>
</table>
---
2017-01-16 04:34:32 +08:00
2017-01-16 04:39:39 +08:00
- [Download Binaries for different platforms](https://github.com/chrislusf/seaweedfs/releases/latest)
2018-03-12 16:28:29 +08:00
- [SeaweedFS on Slack](https://join.slack.com/t/seaweedfs/shared_invite/enQtMzI4MTMwMjU2MzA3LTc4MmVlYmFlNjBmZTgzZmJlYmI1MDE1YzkyNWYyZjkwZDFiM2RlMDdjNjVlNjdjYzc4NGFhZGIyYzEyMzJkYTA)
2017-01-16 04:39:39 +08:00
- [SeaweedFS Mailing List](https://groups.google.com/d/forum/seaweedfs)
- [Wiki Documentation](https://github.com/chrislusf/seaweedfs/wiki)
2014-07-21 13:17:36 +08:00
2014-09-26 13:25:51 +08:00
## Introduction
SeaweedFS is a simple and highly scalable distributed file system. There are two objectives:
2014-09-26 13:25:51 +08:00
1. to store billions of files!
2. to serve the files fast!
2018-06-18 04:40:15 +08:00
SeaweedFS started as an Object Store to handle small files efficiently. Instead of managing all file metadata in a central master, the central master only manages file volumes, and it lets these volume servers manage files and their metadata. This relieves concurrency pressure from the central master and spreads file metadata into volume servers, allowing faster file access (just one disk read operation).
2014-09-26 13:25:51 +08:00
There is only a 40 bytes disk storage overhead for each file's metadata. It is so simple with O(1) disk read that you are welcome to challenge the performance with your actual use cases.
2014-09-26 13:25:51 +08:00
2018-06-18 04:40:15 +08:00
SeaweedFS started by implementing [Facebook's Haystack design paper](http://www.usenix.org/event/osdi10/tech/full_papers/Beaver.pdf).
SeaweedFS can work very well with just the object store. [[Filer]] is added later to support directories and POSIX attributes. Filer is a separate linearly-scalable stateless server with customizable metadata stores, e.g., MySql/Postgres/Redis/Cassandra/LevelDB.
2014-09-26 13:25:51 +08:00
## Additional Features
* Can choose no replication or different replication level, rack and data center aware
* Automatic master servers failover - no single point of failure (SPOF)
2014-09-26 13:25:51 +08:00
* Automatic Gzip compression depending on file mime type
* Automatic compaction to reclaimed disk spaces after deletion or update
* Servers in the same cluster can have different disk spaces, file systems, OS etc.
* Adding/Removing servers does **not** cause any data re-balancing
* Optionally fix the orientation for jpeg pictures
2014-09-26 13:25:51 +08:00
* Support Etag, Accept-Range, Last-Modified, etc.
2017-05-27 14:01:40 +08:00
* Support in-memory/leveldb/boltdb/btree mode tuning for memory/performance balance.
2018-10-08 15:15:04 +08:00
## Filer Features
2018-10-18 02:39:40 +08:00
* [filer server][Filer] provide "normal" directories and files via http.
2018-10-08 15:15:04 +08:00
* [mount filer][Mount] to read and write files directly as a local directory via FUSE.
* [Amazon S3 compatible API][AmazonS3API] to access files with S3 tooling.
2018-12-08 16:50:58 +08:00
* [Hadoop Compatible File System][Hadoop] to access files from Hadoop/Spark/Flink/etc jobs.
2018-10-24 14:59:40 +08:00
* [Async Backup To Cloud][BackupToCloud] can enjoy extreme fast local access and backup to Amazon S3, Google Cloud Storage, Azure, BackBlaze.
2014-09-26 13:25:51 +08:00
2018-06-18 07:55:35 +08:00
[Filer]: https://github.com/chrislusf/seaweedfs/wiki/Directories-and-Files
2018-06-17 17:57:30 +08:00
[Mount]: https://github.com/chrislusf/seaweedfs/wiki/Mount
2018-07-24 13:18:26 +08:00
[AmazonS3API]: https://github.com/chrislusf/seaweedfs/wiki/Amazon-S3-API
2018-10-08 15:15:04 +08:00
[BackupToCloud]: https://github.com/chrislusf/seaweedfs/wiki/Backup-to-Cloud
2018-12-08 16:50:58 +08:00
[Hadoop]: https://github.com/chrislusf/seaweedfs/wiki/Hadoop-Compatible-File-System
2014-09-26 13:25:51 +08:00
## Example Usage
2017-09-14 10:48:41 +08:00
By default, the master node runs on port 9333, and the volume nodes run on port 8080.
Here I will start one master node, and two volume nodes on port 8080 and 8081. Ideally, they should be started from different machines. I just use localhost as an example.
2014-09-26 13:25:51 +08:00
SeaweedFS uses HTTP REST operations to write, read, delete. The responses are in JSON or JSONP format.
2014-09-26 13:25:51 +08:00
### Start Master Server
```
> ./weed master
```
### Start Volume Servers ###
```
> weed volume -dir="/tmp/data1" -max=5 -mserver="localhost:9333" -port=8080 &
> weed volume -dir="/tmp/data2" -max=10 -mserver="localhost:9333" -port=8081 &
2016-09-24 00:49:38 +08:00
```
2014-09-26 13:25:51 +08:00
### Write File ###
To upload a file: first, send a HTTP POST, PUT, or GET request to `/dir/assign` to get an `fid` and a volume server url:
2014-09-26 13:25:51 +08:00
```
2018-09-27 16:13:22 +08:00
> curl http://localhost:9333/dir/assign
2014-09-26 13:25:51 +08:00
{"count":1,"fid":"3,01637037d6","url":"127.0.0.1:8080","publicUrl":"localhost:8080"}
```
2018-09-27 16:11:16 +08:00
Second, to store the file content, send a HTTP multi-part POST request to `url + '/' + fid` from the response:
2014-09-26 13:25:51 +08:00
```
2018-09-27 16:11:16 +08:00
> curl -F file=@/home/chris/myphoto.jpg http://127.0.0.1:8080/3,01637037d6
2014-09-26 13:25:51 +08:00
{"size": 43234}
```
2018-09-27 16:11:16 +08:00
To update, send another POST request with updated file content.
For deletion, send an HTTP DELETE request to the same `url + '/' + fid` URL:
2014-09-26 13:25:51 +08:00
```
> curl -X DELETE http://127.0.0.1:8080/3,01637037d6
```
### Save File Id ###
Now you can save the `fid`, 3,01637037d6 in this case, to a database field.
2014-09-26 13:25:51 +08:00
The number 3 at the start represents a volume id. After the comma, it's one file key, 01, and a file cookie, 637037d6.
2014-09-26 13:25:51 +08:00
The volume id is an unsigned 32-bit integer. The file key is an unsigned 64-bit integer. The file cookie is an unsigned 32-bit integer, used to prevent URL guessing.
The file key and file cookie are both coded in hex. You can store the <volume id, file key, file cookie> tuple in your own format, or simply store the `fid` as string.
2014-09-26 13:25:51 +08:00
If stored as a string, in theory, you would need 8+1+16+8=33 bytes. A char(33) would be enough, if not more than enough, since most usage would not need 2^32 volumes.
If space is really a concern, you can store the file id in your own format. You would need one 4-byte integer for volume id, 8-byte long number for file key, 4-byte integer for file cookie. So 16 bytes are more than enough.
2014-09-26 13:25:51 +08:00
### Read File ###
Here is an example of how to render the URL.
2017-09-14 10:48:41 +08:00
First look up the volume server's URLs by the file's volumeId:
2014-09-26 13:25:51 +08:00
```
> curl http://localhost:9333/dir/lookup?volumeId=3
{"locations":[{"publicUrl":"localhost:8080","url":"localhost:8080"}]}
```
2017-09-14 10:48:41 +08:00
Since (usually) there are not too many volume servers, and volumes don't move often, you can cache the results most of the time. Depending on the replication type, one volume can have multiple replica locations. Just randomly pick one location to read.
2014-09-26 13:25:51 +08:00
Now you can take the public url, render the url or directly read from the volume server via url:
2014-09-26 13:25:51 +08:00
```
http://localhost:8080/3,01637037d6.jpg
```
2015-05-02 18:15:25 +08:00
Notice we add a file extension ".jpg" here. It's optional and just one way for the client to specify the file content type.
2014-09-26 13:25:51 +08:00
If you want a nicer URL, you can use one of these alternative URL formats:
2014-09-26 13:25:51 +08:00
```
http://localhost:8080/3/01637037d6/my_preferred_name.jpg
http://localhost:8080/3/01637037d6.jpg
http://localhost:8080/3,01637037d6.jpg
http://localhost:8080/3/01637037d6
http://localhost:8080/3,01637037d6
```
2017-09-14 10:48:41 +08:00
If you want to get a scaled version of an image, you can add some params:
2015-12-25 14:47:33 +08:00
```
http://localhost:8080/3/01637037d6.jpg?height=200&width=200
2017-05-05 17:17:30 +08:00
http://localhost:8080/3/01637037d6.jpg?height=200&width=200&mode=fit
http://localhost:8080/3/01637037d6.jpg?height=200&width=200&mode=fill
2015-12-25 14:47:33 +08:00
```
2014-09-26 13:25:51 +08:00
### Rack-Aware and Data Center-Aware Replication ###
SeaweedFS applies the replication strategy at a volume level. So, when you are getting a file id, you can specify the replication strategy. For example:
2014-09-26 13:25:51 +08:00
```
2018-09-27 16:13:22 +08:00
curl http://localhost:9333/dir/assign?replication=001
2014-09-26 13:25:51 +08:00
```
The replication parameter options are:
2014-09-26 13:25:51 +08:00
```
000: no replication
001: replicate once on the same rack
010: replicate once on a different rack, but same data center
100: replicate once on a different data center
200: replicate twice on two different data center
110: replicate once on a different rack, and once on a different data center
```
More details about replication can be found [on the wiki][Replication].
[Replication]: https://github.com/chrislusf/seaweedfs/wiki/Replication
2014-09-26 13:25:51 +08:00
You can also set the default replication strategy when starting the master server.
### Allocate File Key on specific data center ###
Volume servers can start with a specific data center name:
2014-09-26 13:25:51 +08:00
```
weed volume -dir=/tmp/1 -port=8080 -dataCenter=dc1
weed volume -dir=/tmp/2 -port=8081 -dataCenter=dc2
```
When requesting a file key, an optional "dataCenter" parameter can limit the assigned volume to the specific data center. For example, this specifies that the assigned volume should be limited to 'dc1':
2014-09-26 13:25:51 +08:00
```
http://localhost:9333/dir/assign?dataCenter=dc1
```
### Other Features ###
* [No Single Point of Failure][feat-1]
* [Insert with your own keys][feat-2]
* [Chunking large files][feat-3]
* [Collection as a Simple Name Space][feat-4]
[feat-1]: https://github.com/chrislusf/seaweedfs/wiki/Failover-Master-Server
[feat-2]: https://github.com/chrislusf/seaweedfs/wiki/Optimization#insert-with-your-own-keys
[feat-3]: https://github.com/chrislusf/seaweedfs/wiki/Optimization#upload-large-files
[feat-4]: https://github.com/chrislusf/seaweedfs/wiki/Optimization#collection-as-a-simple-name-space
2014-09-26 13:25:51 +08:00
## Architecture ##
Usually distributed file systems split each file into chunks, a central master keeps a mapping of filenames, chunk indices to chunk handles, and also which chunks each chunk server has.
2014-09-26 13:25:51 +08:00
The main drawback is that the central master can't handle many small files efficiently, and since all read requests need to go through the chunk master, might not scale well for many concurrent users.
2014-09-26 13:25:51 +08:00
Instead of managing chunks, SeaweedFS manages data volumes in the master server. Each data volume is size 32GB, and can hold a lot of files. And each storage node can have many data volumes. So the master node only needs to store the metadata about the volumes, which is fairly small amount of data and is generally stable.
The actual file metadata is stored in each volume on volume servers. Since each volume server only manages metadata of files on its own disk, with only 16 bytes for each file, all file access can read file metadata just from memory and only needs one disk operation to actually read file data.
2014-09-26 13:25:51 +08:00
For comparison, consider that an xfs inode structure in Linux is 536 bytes.
### Master Server and Volume Server ###
2014-09-26 13:25:51 +08:00
The architecture is fairly simple. The actual data is stored in volumes on storage nodes. One volume server can have multiple volumes, and can both support read and write access with basic authentication.
All volumes are managed by a master server. The master server contains volume id to volume server mapping. This is fairly static information, and could be cached easily.
On each write request, the master server also generates a file key, which is a growing 64-bit unsigned integer. Since write requests are not generally as frequent as read requests, one master server should be able to handle the concurrency well.
2014-09-26 13:25:51 +08:00
### Write and Read files ###
When a client sends a write request, the master server returns (volume id, file key, file cookie, volume node url) for the file. The client then contacts the volume node and POSTs the file content.
2014-09-26 13:25:51 +08:00
When a client needs to read a file based on (volume id, file key, file cookie), it can ask the master server by the volume id for the (volume node url, volume node public url), or retrieve this from a cache. Then the client can GET the content, or just render the URL on web pages and let browsers fetch the content.
2014-09-26 13:25:51 +08:00
Please see the example for details on the write-read process.
2014-09-26 13:25:51 +08:00
### Storage Size ###
In the current implementation, each volume can be 8x2^32 bytes (32GiB). This is because of we align content to 8 bytes. We can easily increase this to 64G, or 128G, or more, by changing 2 lines of code, at the cost of some wasted padding space due to alignment.
2014-09-26 13:25:51 +08:00
There can be 2^32 volumes. So total system size is 8 x 2^32 bytes x 2^32 = 8 x 4GiB x 4Gi = 128EiB (2^67 bytes, or 128 exbibytes).
2014-09-26 13:25:51 +08:00
Each individual file size is limited to the volume size.
### Saving memory ###
2014-09-26 13:25:51 +08:00
All file meta information on volume server is readable from memory without disk access. Each file just takes an 16-byte map entry of <64bit key, 32bit offset, 32bit size>. Of course, each map entry has its own the space cost for the map. But usually the disk runs out before the memory does.
## Compared to Other File Systems ##
Most other distributed file systems seem more complicated than necessary.
2014-09-26 13:25:51 +08:00
SeaweedFS is meant to be fast and simple, both during usage and during setup. If you do not understand how it works when you reach here, we failed! Please raise an issue with any questions or update this file with clarifications.
2014-09-26 13:25:51 +08:00
### Compared to HDFS ###
2018-06-18 07:28:25 +08:00
HDFS uses the chunk approach for each file, and is ideal for storing large files.
2014-09-26 13:25:51 +08:00
SeaweedFS is ideal for serving relatively smaller files quickly and concurrently.
2014-09-26 13:25:51 +08:00
SeaweedFS can also store extra large files by splitting them into manageable data chunks, and store the file ids of the data chunks into a meta chunk. This is managed by "weed upload/download" tool, and the weed master or volume servers are agnostic about it.
2014-09-26 13:25:51 +08:00
2018-06-18 07:28:25 +08:00
### Compared to GlusterFS, Ceph ###
2014-09-26 13:25:51 +08:00
2018-06-18 07:34:10 +08:00
The architectures are mostly the same. SeaweedFS aims to store and read files fast, with a simple and flat architecture. The main differences are
* SeaweedFS optimizes for small files, ensuring O(1) disk seek operation, and can also handle large files.
2018-10-18 09:44:49 +08:00
* SeaweedFS statically assigns a volume id for a file. Locating file content becomes just a lookup of the volume id, which can be easily cached.
* SeaweedFS Filer metadata store can be any well-known and proven data stores, e.g., Cassandra, Redis, MySql, Postgres, etc, and is easy to customized.
* SeaweedFS Volume server also communicates directly with clients via HTTP, supporting range queries, direct uploads, etc.
2014-09-26 13:25:51 +08:00
2018-06-18 07:28:25 +08:00
| System | File Meta | File Content Read| POSIX | REST API | Optimized for small files |
| ------------- | ------------------------------- | ---------------- | ------ | -------- | ------------------------- |
| SeaweedFS | lookup volume id, cacheable | O(1) disk seek | | Yes | Yes |
| SeaweedFS Filer| Linearly Scalable, Customizable | O(1) disk seek | FUSE | Yes | Yes |
| GlusterFS | hashing | | FUSE, NFS | | |
| Ceph | hashing + rules | | FUSE | Yes | |
2014-09-26 13:25:51 +08:00
### Compared to GlusterFS ###
2018-06-18 07:28:25 +08:00
GlusterFS stores files, both directories and content, in configurable volumes called "bricks".
2018-06-18 07:28:25 +08:00
GlusterFS hashes the path and filename into ids, and assigned to virtual volumes, and then mapped to "bricks".
### Compared to Ceph ###
2014-09-26 13:25:51 +08:00
2018-10-18 09:44:49 +08:00
Ceph can be setup similar to SeaweedFS as a key->blob store. It is much more complicated, with the need to support layers on top of it. [Here is a more detailed comparison](https://github.com/chrislusf/seaweedfs/issues/120)
SeaweedFS has a centralized master group to look up free volumes, while Ceph uses hashing and metadata servers to locate its objects. Having a centralized master makes it easy to code and manage.
2018-06-18 07:28:25 +08:00
Same as SeaweedFS, Ceph is also based on a object store RADOS. Ceph is rather complicated with mixed reviews.
2014-09-26 13:25:51 +08:00
2018-06-18 07:28:25 +08:00
Ceph uses CRUSH hashing to automatically manage the data placement. SeaweedFS places data by assigned volumes.
2014-09-26 13:25:51 +08:00
2018-10-18 09:44:49 +08:00
SeaweedFS is optimized for small files. Small files are stored as one continuous block of content, with at most 8 unused bytes between files. Small file access is O(1) disk read.
SeaweedFS Filer uses off-the-shelf stores, such as MySql, Postgres, Redis, Cassandra, to manage file directories. There are proven, scalable, and easier to manage.
2018-06-18 07:28:25 +08:00
| SeaweedFS | comparable to Ceph | advantage |
| ------------- | ------------- | ---------------- |
| Master | MDS | simpler |
| Volume | OSD | optimized for small files |
| Filer | Ceph FS | linearly scalable, Customizable, O(1) or O(logN) |
2014-09-26 13:25:51 +08:00
## Dev plan ##
More tools and documentation, on how to maintain and scale the system. For example, how to move volumes, automatically balancing data, how to grow volumes, how to check system status, etc.
2018-10-18 02:39:40 +08:00
Other key features include: Erasure Encoding, JWT security.
2014-09-26 13:25:51 +08:00
This is a super exciting project! And I need helpers and [support](https://www.patreon.com/seaweedfs)!
2014-09-26 13:25:51 +08:00
2014-12-04 13:37:00 +08:00
## Installation guide for users who are not familiar with golang
step 1: install go on your machine and setup the environment by following the instructions from the following link:
https://golang.org/doc/install
make sure you set up your $GOPATH
step 2: also you may need to install Mercurial by following the instructions below
http://mercurial.selenic.com/downloads
step 3: download, compile, and install the project by executing the following command
2014-12-04 13:37:00 +08:00
2018-10-25 13:16:55 +08:00
```bash
2016-06-03 09:15:54 +08:00
go get github.com/chrislusf/seaweedfs/weed
2018-10-25 13:16:55 +08:00
```
once this is done, you should see the executable "weed" under `$GOPATH/bin`
2014-12-04 13:37:00 +08:00
2018-10-25 13:16:55 +08:00
step 4: after you modify your code locally, you could start a local build by calling `go install` under
2014-12-04 13:37:00 +08:00
2018-10-25 13:16:55 +08:00
```
$GOPATH/src/github.com/chrislusf/seaweedfs/weed
```
2014-12-04 13:37:00 +08:00
2014-09-26 13:25:51 +08:00
## Disk Related topics ##
### Hard Drive Performance ###
2017-09-14 10:48:41 +08:00
When testing read performance on SeaweedFS, it basically becomes performance test for your hard drive's random read speed. Hard Drive usually get 100MB/s~200MB/s.
2014-09-26 13:25:51 +08:00
### Solid State Disk
To modify or delete small files, SSD must delete a whole block at a time, and move content in existing blocks to a new block. SSD is fast when brand new, but will get fragmented over time and you have to garbage collect, compacting blocks. SeaweedFS is friendly to SSD since it is append-only. Deletion and compaction are done on volume level in the background, not slowing reading and not causing fragmentation.
2014-09-26 13:25:51 +08:00
## Benchmark
2015-07-08 17:05:11 +08:00
My Own Unscientific Single Machine Results on Mac Book with Solid State Disk, CPU: 1 Intel Core i7 2.6GHz.
Write 1 million 1KB file:
```
2015-07-08 17:05:11 +08:00
Concurrency Level: 16
Time taken for tests: 88.796 seconds
Complete requests: 1048576
Failed requests: 0
2015-07-08 17:05:11 +08:00
Total transferred: 1106764659 bytes
Requests per second: 11808.87 [#/sec]
Transfer rate: 12172.05 [Kbytes/sec]
Connection Times (ms)
min avg max std
2015-07-08 17:05:11 +08:00
Total: 0.2 1.3 44.8 0.9
Percentage of the requests served within a certain time (ms)
2015-07-08 17:05:11 +08:00
50% 1.1 ms
66% 1.3 ms
75% 1.5 ms
80% 1.7 ms
90% 2.1 ms
95% 2.6 ms
98% 3.7 ms
99% 4.6 ms
100% 44.8 ms
```
Randomly read 1 million files:
```
2015-07-08 17:05:11 +08:00
Concurrency Level: 16
Time taken for tests: 34.263 seconds
Complete requests: 1048576
Failed requests: 0
2015-07-08 17:05:11 +08:00
Total transferred: 1106762945 bytes
Requests per second: 30603.34 [#/sec]
Transfer rate: 31544.49 [Kbytes/sec]
Connection Times (ms)
min avg max std
2015-07-08 17:05:11 +08:00
Total: 0.0 0.5 20.7 0.7
Percentage of the requests served within a certain time (ms)
2015-07-08 17:05:11 +08:00
50% 0.4 ms
75% 0.5 ms
95% 0.6 ms
98% 0.8 ms
99% 1.2 ms
100% 20.7 ms
```
2015-04-14 00:52:48 +08:00
## License
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
2018-09-21 13:57:53 +08:00
## Stargazers over time
[![Stargazers over time](https://starcharts.herokuapp.com/chrislusf/seaweedfs.svg)](https://starcharts.herokuapp.com/chrislusf/seaweedfs)