Skip to content
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 1 addition & 2 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,7 @@ coverage
# node-waf configuration
.lock-wscript

# Compiled binary addons (http://nodejs.org/api/addons.html)
build/Release
build

# Dependency directory
# https://www.npmjs.org/doc/misc/npm-faq.html#should-i-check-my-node_modules-folder-into-git
Expand Down
107 changes: 107 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,2 +1,109 @@
# js-ipfs-repo

Implementation of the IPFS repo spec (https://github.com/ipfs/specs/tree/master/repo) in JavaScript

## API

### `Repo`

Constructor, accepts a path and options:

```js
var Repo = require('js-ipfs-repo')
var repo = new Repo('/Users/someone/.ipfs', {adaptor: 'fs'})
```

Options:

- `adaptor`: String with the adaptor. Defaults to `fs`

### `#version`

Read/Write the version number of that repository.

```js
repo.version().read(function (err, num) {
console.log(err, num) // => 2
})

repo.version().write(3, function (err) {
console.log(err)
})
```

### `#api`

Read/Write the JSON configuration for that repository.

```js
repo.api().read(function (err, multiaddr) {
console.log(err, multiaddr)
})

repo.api().write('/ip4/127.0.0.1/tcp/5001', function (err) {
console.log(err)
})
```

### `#config`

Read/Write the JSON configuration for that repository.

```js
repo.config().read(function (err, json) {
console.log(err, json)
})

repo.config().write({foo: 'bar'}, function (err) {
console.log(err)
})
```

### `#blocks`

Store data on the block store.

```js
repo.blocks().read('12200007d4e3a319cd8c7c9979280e150fc5dbaae1ce54e790f84ae5fd3c3c1a0475', function (buff, err) {
console.log(err)
})
```

```js
repo.blocks().write(buff, function (buff, err) {
console.log(buff.toString('utf-8'), err)
})
```

### `#repo`

Read/Write the `repo.lock` file.

```js
repo.repo().read(function (err, content) {
console.log(err, content)
})

repo.repo().write('foo', function (err) {
console.log(err)
})
```

## Adaptors
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe keep calling them blob-stores and point to abstract-blob-store?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the issue I'm having is that those adaptors can't just be blob-stores. There is some extra plumbing that needs to be done around. For instance the fs-repo need locking through a repo.lock file which may be implemented completely different on a IndexedDB or localstorage implementation. Also, I may want to list all the blocks in a repository, that's not something doable within a blob-store and will have different implementations depending the adaptor.


By default it will use the `fs-repo` adaptor. Eventually we can write other adaptors
and make those available on configuration.

### `fs-repo`

The default adaptor. Uses the `repo.lock` file to ensure there are no simultaneous reads
nor writes. Uses the `fs-blob-store`.

### `memory-repo`

Ideal for testing purposes. Uses the `abstract-blob-store`.

## Tests

Not there yet! Should ran both in node and in Phantom with compatible
adaptors.
33 changes: 33 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
{
"name": "js-ipfs-repo",
"version": "0.0.1",
"description": "IPFS Repo implementation",
"main": "src/index.js",
"scripts": {
"start": "node src/index.js",
"lint": "./node_modules/.bin/standard"
},
"repository": {
"type": "git",
"url": "https://github.com/diasdavid/js-ipfs-repo.git"
},
"keywords": [
"IPFS"
],
"pre-commit": [
"lint"
],
"homepage": "https://github.com/diasdavid/js-libp2p-record",
"devDependencies": {
"pre-commit": "^1.1.1",
"standard": "^5.1.1"
},
"dependencies": {
"abstract-blob-store": "^3.2.0",
"concat-stream": "^1.5.1",
"fs-blob-store": "^5.2.0",
"level-js": "^2.2.2",
"lockfile": "^1.0.1",
"multihashes": "^0.2.0"
}
}
121 changes: 121 additions & 0 deletions src/adaptors/fs-repo/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
var concat = require('concat-stream')
var fs = require('fs-blob-store')
var path = require('path')
var lockFile = require('lockfile')

function BlobStore (base_path) {
this.store = fs(base_path)
this.LOCK_PATH = path.join(base_path, 'repo.lock')
}

BlobStore.prototype = {

/**
* Read a blob on a given path
* It holds the repo.lock while reading
*
* @param {String} key
* @param {Function} cb
* @return {ReadableStream}
*/
read: function (key, cb) {
var store = this.store
var LOCK_PATH = this.LOCK_PATH
var rs = store.createReadStream(key)

function onFinish (buff) {
lockFile.unlock(LOCK_PATH, function (err) {
if (err) return cb(err)

cb(null, buff.toString('utf8'))
})
}

function onLock (err) {
if (err) return cb(err)

rs.on('error', cb)
rs.pipe(concat(onFinish))
}

lockFile.lock(LOCK_PATH, {}, onLock)

return rs
},

/**
* Read a blob on a given path
* It does not lock
*
* @param {String} key
* @param {Function} cb
* @return {ReadableStream}
*/
readWithoutLock: function (key, cb) {
var rs = this.store.createReadStream(key)

rs.on('error', cb)
rs.pipe(concat(function (buff) {
cb(null, buff.toString('utf8'))
}))

return rs
},

/**
* Write the contents to the blob in the given path
* It holds the repo.lock while reading
*
* @param {String} key
* @param {Function} cb
* @return {WritableStream}
*/
write: function (key, content, cb) {
var store = this.store
var LOCK_PATH = this.LOCK_PATH
var ws = store.createWriteStream(key)

function onFinish (err) {
if (err) return cb(err)

lockFile.unlock(LOCK_PATH, cb)
}

function onLock (err) {
if (err) return cb(err)

ws.on('error', cb)
ws.on('finish', onFinish)

ws.write(content)
ws.end()
}

lockFile.lock(LOCK_PATH, {}, onLock)

return ws
},

/**
* Writes content to a blob on a given path
* It does not lock
*
* @param {String} key
* @param {String} content
* @param {Function} cb
* @return {WritableStream}
*/
writeWithoutLock: function (key, content, cb) {
var ws = this.store.createWriteStream(key)

ws.on('error', cb)
ws.on('finish', cb)

ws.write(content)
ws.end()

return ws
}
}

module.exports = BlobStore
4 changes: 4 additions & 0 deletions src/adaptors/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
module.exports = {
'fs-repo': require('./fs-repo'),
'memory-repo': require('./memory-repo')
}
74 changes: 74 additions & 0 deletions src/adaptors/memory-repo/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
var concat = require('concat-stream')
var ms = require('abstract-blob-store')

function BlobStore () {
this.store = ms()
}

BlobStore.prototype = {

/**
* Read a blob on a given path
*
* @param {String} key
* @param {Function} cb
* @return {ReadableStream}
*/
read: function (key, cb) {
var store = this.store
var rs = store.createReadStream(key)

function onFinish (buff) {
cb(null, buff.toString('utf8'))
}

rs.on('error', cb)
rs.pipe(concat(onFinish))

return rs
},

/**
* Read a blob on a given path
*
* @param {String} key
* @param {Function} cb
* @return {ReadableStream}
*/
readWithoutLock: function (key, cb) {
return this.read(key, cb)
},

/**
* Write the contents to the blob in the given path
*
* @param {String} key
* @param {Function} cb
* @return {WritableStream}
*/
write: function (key, content, cb) {
var store = this.store
var ws = store.createWriteStream(key)

ws.on('error', cb)
ws.on('finish', cb)

ws.write(content)
ws.end()

return ws
},

/**
* Write the contents to the blob in the given path
*
* @param {String} key
* @param {Function} cb
* @return {WritableStream}
*/
writeWithoutLock: function (key, cb) {
return this.write(key, cb)
}
}

module.exports = BlobStore
Loading