Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: adding self-hosted servers #952

Merged
merged 132 commits into from
Oct 31, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
132 commits
Select commit Hold shift + click to select a range
51ef5d1
Initial server implementation
gmaclennan Oct 1, 2024
76387c9
WIP
gmaclennan Oct 1, 2024
1e568af
fix: use identity keypair for replication
gmaclennan Oct 1, 2024
88fd5b1
server: set device info on initialization
gmaclennan Oct 1, 2024
f997101
chore: move #identityKeypair member higher up
EvanHahn Oct 1, 2024
d272664
chore: remove unused server script (#876)
EvanHahn Oct 1, 2024
72f3190
chore: use `await` instead of `then` in server route (#878)
EvanHahn Oct 1, 2024
e0b4d4a
chore: remove commented-out `PeerSyncController` lines (#877)
EvanHahn Oct 1, 2024
ab49173
chore: remove console.logs from server code (#879)
EvanHahn Oct 1, 2024
e0ec6b8
Add GET /deviceinfo endpoint
EvanHahn Oct 1, 2024
c6629d2
chore: move response body data to a `data` key
EvanHahn Oct 1, 2024
f6f97f5
make createTestServer helper do more
gmaclennan Oct 2, 2024
d6128b0
use @fastify/sensible and more precise errors
gmaclennan Oct 2, 2024
d1dcc85
Test that doesn't work for some reason
gmaclennan Oct 2, 2024
59560b0
fix test
gmaclennan Oct 2, 2024
f018cf7
server integration tests
gmaclennan Oct 2, 2024
77b73be
Change `host` to `baseUrl`, save server URL to members
gmaclennan Oct 2, 2024
55f90d6
WIP: connectServers and disconnectServers()
gmaclennan Oct 2, 2024
a9dad88
Fix debugging event (still broken, but better debugging)
EvanHahn Oct 2, 2024
e3cec3d
Revert sync changes
EvanHahn Oct 3, 2024
7f74297
Sync test passing
EvanHahn Oct 3, 2024
400e305
Remove a useless delay
EvanHahn Oct 3, 2024
54823a8
Some server test cleanup
EvanHahn Oct 3, 2024
efdbee0
More server test cleanup
EvanHahn Oct 3, 2024
e9d5933
Remove a debugging console.log
EvanHahn Oct 3, 2024
6442c37
Clean up server integration tests
EvanHahn Oct 3, 2024
154a9f2
Skeleton GET /observations endpoint
EvanHahn Oct 3, 2024
9d67113
Returning observations without attachments
EvanHahn Oct 3, 2024
04c6e74
Merge branch 'main' into server
EvanHahn Oct 7, 2024
87934f6
Remove workaround for now-fixed bug
EvanHahn Oct 7, 2024
b82a69a
Minor: fix typo in comment
EvanHahn Oct 7, 2024
42999c5
Fix disconnect bug
EvanHahn Oct 7, 2024
2f2eea0
Remove .only from server integration test
EvanHahn Oct 7, 2024
07b6520
Fix test type error
EvanHahn Oct 7, 2024
981cb6e
GET /observations should authenticate
EvanHahn Oct 7, 2024
1f033a8
feat: add server.js script (#892)
gmaclennan Oct 7, 2024
4abd002
Fix type error when starting server
EvanHahn Oct 7, 2024
aef5186
GET /observations should return fetchable attachments
EvanHahn Oct 7, 2024
e72f342
Quiet type errors
EvanHahn Oct 7, 2024
75be09c
feat: use req.hostname instead of manually configured server host (#893)
gmaclennan Oct 8, 2024
c5c4a35
Add Dockerfile and fly.io deployment instructions and config (#894)
gmaclennan Oct 8, 2024
5d0a0aa
Remove old environment variable
EvanHahn Oct 8, 2024
4a275d8
Start sync when opening sync websocket
EvanHahn Oct 8, 2024
350eb8a
GET /projects to list projects
EvanHahn Oct 8, 2024
e47b743
chore: add CI workflow for server e2e tests (#895)
gmaclennan Oct 8, 2024
d6dbf2b
fix: avoid uncaught websocket errors (#897)
gmaclennan Oct 9, 2024
4fd5039
GET /projects should return project name
EvanHahn Oct 9, 2024
619894b
Remove an already-addressed TODO
EvanHahn Oct 9, 2024
df60da6
Proxy all headers through with attachments endpoint
EvanHahn Oct 9, 2024
d4a3999
Merge branch 'main' into server
EvanHahn Oct 9, 2024
22e1975
Merge branch 'main' into server
EvanHahn Oct 9, 2024
ac18cef
Add `deleted` field to GET /observations
EvanHahn Oct 9, 2024
be6dfe7
Add response schema for GET /observations
EvanHahn Oct 9, 2024
7f4f129
GET /observations should include tags
EvanHahn Oct 9, 2024
6d0a947
Remove/move some TODOs
EvanHahn Oct 9, 2024
7cc3a1f
Clean up project public key in MapeoProject
EvanHahn Oct 9, 2024
bb74456
Split addServerPeer into smaller methods
EvanHahn Oct 9, 2024
7449eb5
Wait for initial sync with server, not all peers
EvanHahn Oct 9, 2024
9aaeb37
waitForSyncWithServer shouldn't just wait
EvanHahn Oct 9, 2024
9135128
Improve validation of server base URLs
EvanHahn Oct 9, 2024
257b991
Move some members higher up in SyncApi
EvanHahn Oct 9, 2024
4d34392
Fix broken test
EvanHahn Oct 9, 2024
bee1ca4
Merge branch 'main' into server
EvanHahn Oct 16, 2024
1642109
test: server integration tests should use dynamic port
EvanHahn Oct 16, 2024
3b0a8e0
Move server integration tests into a separate server test folder
EvanHahn Oct 17, 2024
ad681f4
test: move GET /info test to separate file
EvanHahn Oct 17, 2024
10fb67b
test: move allowedhosts tests to separate file
EvanHahn Oct 17, 2024
7b443a8
test: move GET /projects tests to separate file
EvanHahn Oct 17, 2024
1b6e2ea
test: move POST /projects test to separate file
EvanHahn Oct 17, 2024
d2624f3
test: Move GET /observations to its own test file
EvanHahn Oct 17, 2024
d21ea62
test: rename remaining server integration tests
EvanHahn Oct 17, 2024
1920bce
Cleanups to sync endpoint tests
EvanHahn Oct 17, 2024
32e403c
WIP: additional tests for POST /proejcts
EvanHahn Oct 17, 2024
74fd08a
Clean up some add project endpoint tests
EvanHahn Oct 17, 2024
f6b65a3
fix: suspend rather than stop fly machines
gmaclennan Oct 21, 2024
bf8cec0
Merge branch 'main' into server
EvanHahn Oct 21, 2024
9aa74f3
Require server base URL to be relative to root path
EvanHahn Oct 21, 2024
44612cc
Test if server returns a non-200
EvanHahn Oct 21, 2024
4c41246
Remove a TODO
EvanHahn Oct 21, 2024
bb1d236
Also allow server to return 201
EvanHahn Oct 21, 2024
5e8df03
Test server returning bad data
EvanHahn Oct 21, 2024
297983d
Fix server test concurrency
EvanHahn Oct 21, 2024
f106f07
Fail if we can't connect to the server
EvanHahn Oct 21, 2024
d5380b6
Use a port guaranteed to be open
EvanHahn Oct 21, 2024
654ceb2
Remove a TODO from server test
EvanHahn Oct 21, 2024
33640d1
Handle bad server /sync endpoint
EvanHahn Oct 21, 2024
2661bea
Only include photo attachments in GET /observations endpoint
EvanHahn Oct 21, 2024
719df6b
Move server tests to src/server/test/
EvanHahn Oct 21, 2024
1bb9c06
Merge branch 'main' into server
EvanHahn Oct 21, 2024
c511a60
Basic server root
EvanHahn Oct 21, 2024
ae24889
Return some error codes when adding peers
EvanHahn Oct 21, 2024
fc1ad52
Add some failure codes when adding a server
EvanHahn Oct 21, 2024
1c025a1
test: use helper for finding server peer
EvanHahn Oct 21, 2024
b9dd1c0
Skeleton removal of server peer
EvanHahn Oct 21, 2024
4ce93ef
Merge branch 'main' into server
EvanHahn Oct 22, 2024
749c467
WIP
EvanHahn Oct 17, 2024
503b17e
Use 409, not 403, for duplicates
EvanHahn Oct 22, 2024
9192404
Adding projects multiple times is idempotent
EvanHahn Oct 22, 2024
9625729
Add a comment
EvanHahn Oct 22, 2024
9423285
Add another comment
EvanHahn Oct 22, 2024
cb8ebd3
feat: convert server to fastify plugin (#919)
gmaclennan Oct 23, 2024
a9cfff6
Merge branch 'main' into server
EvanHahn Oct 23, 2024
d1db84c
More work on removing server peers
EvanHahn Oct 23, 2024
9b0d820
Minor: reorder two methods
EvanHahn Oct 23, 2024
cf02e69
Merge branch 'main' into server
EvanHahn Oct 23, 2024
43f07ec
Remove unnecessary server close
EvanHahn Oct 23, 2024
a2ca81e
test: remove unnecessary `await`
EvanHahn Oct 23, 2024
23b7962
Remove unused test file
EvanHahn Oct 23, 2024
883d14b
Remove removeServerPeer
EvanHahn Oct 23, 2024
4827194
Server: PUT /projects, not POST /projects (#938)
EvanHahn Oct 24, 2024
4f47a29
Use string-timing-safe-equal in server code
EvanHahn Oct 24, 2024
47cce87
Install @comapeo/schema 1.2.0
EvanHahn Oct 24, 2024
f47d9f0
Use string-timing-safe-equal in member API code
EvanHahn Oct 24, 2024
02aedad
Merge branch 'main' into server
EvanHahn Oct 24, 2024
4482f59
Fix CI: we now use the real schema package
EvanHahn Oct 24, 2024
4f834c3
Merge branch 'main' into server
EvanHahn Oct 28, 2024
50be333
Remove some `any`s from server code
EvanHahn Oct 28, 2024
1c4f1ed
add reference to issue 25
EvanHahn Oct 28, 2024
50a5271
Send project name to server when adding it
EvanHahn Oct 29, 2024
b95e24c
More tests bad requests to `PUT /projects`
EvanHahn Oct 29, 2024
3f5d607
Handle quick connect/disconnects to server
EvanHahn Oct 29, 2024
966da94
Remove removeServerPeer skeleton test
EvanHahn Oct 29, 2024
0223710
reference a TODO
EvanHahn Oct 29, 2024
97e0902
test: stop using getManagerOptions
EvanHahn Oct 29, 2024
354a0a8
Remove getManagerOptions
EvanHahn Oct 29, 2024
4a82f22
Merge branch 'main' into server
EvanHahn Oct 30, 2024
b3b749b
Merge branch 'main' into server
EvanHahn Oct 30, 2024
3d8bd40
Merge branch 'main' into server
EvanHahn Oct 30, 2024
6d18bf6
src/server should use more realistic imports from core
EvanHahn Oct 30, 2024
92c84c7
Use more realistic imports of server code from core
EvanHahn Oct 30, 2024
5fa6192
Clean up a few small things
EvanHahn Oct 31, 2024
2c89586
Remove server code from project
EvanHahn Oct 31, 2024
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
521 changes: 167 additions & 354 deletions package-lock.json

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,7 @@
"@types/sub-encoder": "^2.1.0",
"@types/throttle-debounce": "^5.0.0",
"@types/varint": "^6.0.1",
"@types/ws": "^8.5.12",
"@types/yauzl-promise": "^4.0.0",
"@types/yazl": "^2.4.5",
"bitfield": "^4.2.0",
Expand Down Expand Up @@ -200,6 +201,7 @@
"type-fest": "^4.5.0",
"undici": "^6.13.0",
"varint": "^6.0.0",
"ws": "^8.18.0",
"yauzl-promise": "^4.0.0"
}
}
10 changes: 10 additions & 0 deletions src/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,19 @@ import {
COORDINATOR_ROLE_ID,
MEMBER_ROLE_ID,
} from './roles.js'
import { kProjectReplicate } from './mapeo-project.js'
export { plugin as CoMapeoMapsFastifyPlugin } from './fastify-plugins/maps.js'
export { FastifyController } from './fastify-controller.js'
export { MapeoManager } from './mapeo-manager.js'
/** @import { MapeoProject } from './mapeo-project.js' */

/**
* @param {MapeoProject} project
* @param {Parameters<MapeoProject.prototype[kProjectReplicate]>} args
* @returns {ReturnType<MapeoProject.prototype[kProjectReplicate]>}
*/
export const replicateProject = (project, ...args) =>
project[kProjectReplicate](...args)

export const roles = /** @type {const} */ ({
CREATOR_ROLE_ID,
Expand Down
47 changes: 47 additions & 0 deletions src/lib/error.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
/**
* Create an `Error` with a `code` property.
*
* @example
* const err = new ErrorWithCode('INVALID_DATA', 'data was invalid')
* err.message
* // => 'data was invalid'
* err.code
* // => 'INVALID_DATA'
*/
export class ErrorWithCode extends Error {
/**
* @param {string} code
* @param {string} message
* @param {object} [options]
* @param {unknown} [options.cause]
*/
constructor(code, message, options) {
super(message, options)
/** @readonly */ this.code = code
}
}

/**
* Get the error message from an object if possible.
* Otherwise, stringify the argument.
*
* @param {unknown} maybeError
* @returns {string}
* @example
* try {
* // do something
* } catch (err) {
* console.error(getErrorMessage(err))
* }
*/
export function getErrorMessage(maybeError) {
if (maybeError && typeof maybeError === 'object' && 'message' in maybeError) {
try {
const { message } = maybeError
if (typeof message === 'string') return message
} catch (_err) {
// Ignored
}
}
return 'unknown error'
}
10 changes: 10 additions & 0 deletions src/lib/get-own.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
/**
* @template {object} T
* @template {keyof T} K
* @param {T} obj
* @param {K} key
* @returns {undefined | T[K]}
*/
export function getOwn(obj, key) {
return Object.hasOwn(obj, key) ? obj[key] : undefined
}
26 changes: 26 additions & 0 deletions src/lib/is-hostname-ip-address.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import { isIPv4, isIPv6 } from 'node:net'

/**
* Is this hostname an IP address?
*
* @param {string} hostname
* @returns {boolean}
* @example
* isHostnameIpAddress('100.64.0.42')
* // => false
*
* isHostnameIpAddress('[2001:0db8:85a3:0000:0000:8a2e:0370:7334]')
* // => true
*
* isHostnameIpAddress('example.com')
* // => false
*/
export function isHostnameIpAddress(hostname) {
if (isIPv4(hostname)) return true

if (hostname.startsWith('[') && hostname.endsWith(']')) {
return isIPv6(hostname.slice(1, -1))
}

return false
}
47 changes: 47 additions & 0 deletions src/lib/ws-core-replicator.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
import { pipeline } from 'node:stream/promises'
import { Transform } from 'node:stream'
import { createWebSocketStream } from 'ws'
/** @import { WebSocket } from 'ws' */
/** @import { ReplicationStream } from '../types.js' */

/**
* @param {WebSocket} ws
* @param {ReplicationStream} replicationStream
* @returns {Promise<void>}
*/
export function wsCoreReplicator(ws, replicationStream) {
// This is purely to satisfy typescript at its worst. `pipeline` expects a
// NodeJS ReadWriteStream, but our replicationStream is a streamx Duplex
// stream. The difference is that streamx does not implement the
// `setEncoding`, `unpipe`, `wrap` or `isPaused` methods. The `pipeline`
// function does not depend on any of these methods (I have read through the
// NodeJS source code at cebf21d (v22.9.0) to confirm this), so we can safely
// cast the stream to a NodeJS ReadWriteStream.
const _replicationStream = /** @type {NodeJS.ReadWriteStream} */ (
/** @type {unknown} */ (replicationStream)
)
return pipeline(
_replicationStream,
wsSafetyTransform(ws),
createWebSocketStream(ws),
_replicationStream
)
}

/**
* Avoid writing data to a closing or closed websocket, which would result in an
* error. Instead we drop the data and wait for the stream close/end events to
* propagate and close the streams cleanly.
*
* @param {WebSocket} ws
*/
function wsSafetyTransform(ws) {
return new Transform({
transform(chunk, encoding, callback) {
if (ws.readyState === ws.CLOSED || ws.readyState === ws.CLOSING) {
return callback()
}
callback(null, chunk)
},
})
}
2 changes: 1 addition & 1 deletion src/mapeo-manager.js
Original file line number Diff line number Diff line change
Expand Up @@ -556,7 +556,7 @@ export class MapeoManager extends TypedEmitter {
* downloaded their proof of project membership and the project config.
*
* @param {Pick<import('./generated/rpc.js').ProjectJoinDetails, 'projectKey' | 'encryptionKeys'> & { projectName: string }} projectJoinDetails
* @param {{ waitForSync?: boolean }} [opts] For internal use in tests, set opts.waitForSync = false to not wait for sync during addProject()
* @param {{ waitForSync?: boolean }} [opts] Set opts.waitForSync = false to not wait for sync during addProject()
* @returns {Promise<string>}
*/
addProject = async (
Expand Down
63 changes: 53 additions & 10 deletions src/mapeo-project.js
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,11 @@ import {
import { migrate } from './lib/drizzle-helpers.js'
import { omit } from './lib/omit.js'
import { MemberApi } from './member-api.js'
import { SyncApi, kHandleDiscoveryKey } from './sync/sync-api.js'
import {
SyncApi,
kHandleDiscoveryKey,
kWaitForInitialSyncWithPeer,
} from './sync/sync-api.js'
import { Logger } from './logger.js'
import { IconApi } from './icon-api.js'
import { readConfig } from './config-import.js'
Expand Down Expand Up @@ -77,8 +81,9 @@ const EMPTY_PROJECT_SETTINGS = Object.freeze({})
* @extends {TypedEmitter<{ close: () => void }>}
*/
export class MapeoProject extends TypedEmitter {
#projectId
#projectKey
#deviceId
#identityKeypair
#coreManager
#indexWriter
#dataStores
Expand Down Expand Up @@ -135,10 +140,12 @@ export class MapeoProject extends TypedEmitter {

this.#l = Logger.create('project', logger)
this.#deviceId = getDeviceId(keyManager)
this.#projectId = projectKeyToId(projectKey)
this.#projectKey = projectKey
this.#loadingConfig = false
this.#isArchiveDevice = isArchiveDevice

const getReplicationStream = this[kProjectReplicate].bind(this, true)

///////// 1. Setup database

this.#sqlite = new Database(dbPath)
Expand Down Expand Up @@ -317,7 +324,7 @@ export class MapeoProject extends TypedEmitter {
},
}),
}
const identityKeypair = keyManager.getIdentityKeypair()
this.#identityKeypair = keyManager.getIdentityKeypair()
const coreKeypairs = getCoreKeypairs({
projectKey,
projectSecretKey,
Expand All @@ -326,31 +333,33 @@ export class MapeoProject extends TypedEmitter {
this.#coreOwnership = new CoreOwnership({
dataType: this.#dataTypes.coreOwnership,
coreKeypairs,
identityKeypair,
identityKeypair: this.#identityKeypair,
})
this.#roles = new Roles({
dataType: this.#dataTypes.role,
coreOwnership: this.#coreOwnership,
coreManager: this.#coreManager,
projectKey: projectKey,
deviceKey: keyManager.getIdentityKeypair().publicKey,
deviceKey: this.#identityKeypair.publicKey,
})

this.#memberApi = new MemberApi({
deviceId: this.#deviceId,
roles: this.#roles,
coreOwnership: this.#coreOwnership,
encryptionKeys,
getProjectName: this.#getProjectName.bind(this),
projectKey,
rpc: localPeers,
getReplicationStream,
waitForInitialSyncWithPeer: (deviceId, abortSignal) =>
this.$sync[kWaitForInitialSyncWithPeer](deviceId, abortSignal),
dataTypes: {
deviceInfo: this.#dataTypes.deviceInfo,
project: this.#dataTypes.projectSettings,
},
})

const projectPublicId = projectKeyToPublicId(projectKey)

this.#blobStore = new BlobStore({
coreManager: this.#coreManager,
})
Expand All @@ -362,7 +371,7 @@ export class MapeoProject extends TypedEmitter {
if (!base.endsWith('/')) {
base += '/'
}
return base + projectPublicId
return base + this.#projectPublicId
},
})

Expand All @@ -374,7 +383,7 @@ export class MapeoProject extends TypedEmitter {
if (!base.endsWith('/')) {
base += '/'
}
return base + projectPublicId
return base + this.#projectPublicId
},
})

Expand All @@ -384,6 +393,24 @@ export class MapeoProject extends TypedEmitter {
roles: this.#roles,
blobDownloadFilter: null,
logger: this.#l,
getServerWebsocketUrls: async () => {
const members = await this.#memberApi.getMany()
/** @type {string[]} */
const serverWebsocketUrls = []
for (const member of members) {
if (
member.deviceType === 'selfHostedServer' &&
member.selfHostedServerDetails
) {
const { baseUrl } = member.selfHostedServerDetails
const wsUrl = new URL(`/sync/${this.#projectPublicId}`, baseUrl)
wsUrl.protocol = wsUrl.protocol === 'http:' ? 'ws:' : 'wss:'
serverWebsocketUrls.push(wsUrl.href)
}
}
return serverWebsocketUrls
},
getReplicationStream,
})

this.#translationApi = new TranslationApi({
Expand Down Expand Up @@ -458,6 +485,14 @@ export class MapeoProject extends TypedEmitter {
return this.#deviceId
}

get #projectId() {
return projectKeyToId(this.#projectKey)
}

get #projectPublicId() {
return projectKeyToPublicId(this.#projectKey)
}

/**
* Resolves when hypercores have all loaded
*
Expand Down Expand Up @@ -603,6 +638,13 @@ export class MapeoProject extends TypedEmitter {
}
}

/**
* @returns {Promise<undefined | string>}
*/
async #getProjectName() {
return (await this.$getProjectSettings()).name
}

async $getOwnRole() {
return this.#roles.getRole(this.#deviceId)
}
Expand Down Expand Up @@ -640,6 +682,7 @@ export class MapeoProject extends TypedEmitter {
* Hypercore types need updating.
* @type {any}
*/ ({
keyPair: this.#identityKeypair,
/** @param {Buffer} discoveryKey */
ondiscoverykey: async (discoveryKey) => {
const protomux =
Expand Down
Loading
Loading