Pigeon-Ruby/lib/pigeon/database.rb

166 lines
4.5 KiB
Ruby
Raw Normal View History

module Pigeon
class Database
attr_reader :who_am_i
2020-04-18 15:30:35 +00:00
def initialize(path: PIGEON_DB_PATH)
@store = Pigeon::Storage.new(path: path)
2020-04-18 15:51:08 +00:00
init_ident
end
2020-04-19 15:26:21 +00:00
# === PEERS
2020-04-18 15:30:35 +00:00
def add_peer(p); store.add_peer(p); end
def block_peer(p); store.block_peer(p); end
2020-04-19 15:26:21 +00:00
def remove_peer(p); store.remove_peer(p); end
2020-04-20 02:32:37 +00:00
def peer_blocked?(p); store.peer_blocked?(p); end
def all_blocks(); store.all_blocks(); end
def all_peers(); store.all_peers(); end
2020-04-19 15:26:21 +00:00
# === MESSAGES
def all_messages(mhash = nil); store.all_messages(mhash); end
2020-04-20 02:32:37 +00:00
def message_saved?(multihash); store.message_saved?(multihash); end
2020-04-18 14:59:56 +00:00
def _save_message(msg_obj)
2020-04-18 14:59:56 +00:00
store.insert_message(Helpers.verify_message(self, msg_obj))
end
2020-04-17 13:55:18 +00:00
def read_message(multihash); store.read_message(multihash); end
def get_message_count_for(multihash)
store.get_message_count_for(multihash)
end
def get_message_by_depth(multihash, depth)
store.get_message_by_depth(multihash, depth)
end
def add_message(kind, params)
2020-04-19 21:27:09 +00:00
publish_draft(new_draft(kind: kind, body: params))
end
2020-04-19 15:26:21 +00:00
# Store a message that someone (not the LocalIdentity)
# has authored.
def _ingest_message(author:,
body:,
depth:,
kind:,
lipmaa:,
prev:,
signature:)
2020-04-19 15:26:21 +00:00
msg = Message.new(author: RemoteIdentity.new(author),
kind: kind,
body: body,
prev: prev,
lipmaa: lipmaa,
signature: signature,
depth: depth)
_save_message(msg)
end
2020-04-19 15:26:21 +00:00
# === DRAFTS
def reset_draft; add_config(CURRENT_DRAFT, nil); end
2020-04-19 21:27:09 +00:00
def new_draft(kind:, body: {})
old = get_config(CURRENT_DRAFT)
if old
raise "PUBLISH OR RESET CURRENT DRAFT (#{old.kind}) FIRST"
end
save_draft(Draft.new(kind: kind, body: body))
end
def save_draft(draft)
add_config(CURRENT_DRAFT, draft)
draft
end
def get_draft
2020-04-19 21:27:09 +00:00
draft = store.get_config(CURRENT_DRAFT)
if draft
return draft
else
raise "THERE IS NO DRAFT. CREATE ONE FIRST."
end
end
2020-04-19 21:27:09 +00:00
def update_draft(k, v); Helpers.update_draft(self, k, v); end
def reset_draft
add_config(CURRENT_DRAFT, nil)
2020-04-18 14:13:53 +00:00
end
# Author a new message.
def publish_draft(draft = self.get_draft)
2020-04-18 14:13:53 +00:00
Helpers.publish_draft(self, draft)
end
2020-04-19 15:26:21 +00:00
# === BUNDLES
2020-04-24 11:20:02 +00:00
def export_bundle(file_path = DEFAULT_BUNDLE_PATH)
Helpers.mkdir_p(file_path)
# Fetch messages for all peers
peers = all_peers + [who_am_i.multihash]
messages = peers.map do |peer|
all_messages(peer)
.map { |multihash| read_message(multihash) }
.sort_by(&:depth)
end.flatten
# Attach blobs for all messages in bundle.
2020-04-22 13:21:55 +00:00
messages
.map(&:collect_blobs)
.flatten
.uniq
.map do |mhash|
blob_path = File.join(file_path, Helpers.hash2file_path(mhash))
Helpers.write_to_disk(blob_path, mhash, get_blob(mhash))
end
# Render messages for all peers.
content = messages
.map { |message| message.render }
.join(BUNDLE_MESSAGE_SEPARATOR)
File.write(File.join(file_path, "gossip.pgn"), content + CR)
2020-04-19 15:26:21 +00:00
end
2020-04-24 11:20:02 +00:00
def import_bundle(file_path = DEFAULT_BUNDLE_PATH)
bundle = File.read(File.join(file_path, "gossip.pgn"))
2020-04-19 15:26:21 +00:00
tokens = Pigeon::Lexer.tokenize(bundle)
blobs = tokens.reduce(Set.new) do |set, (a, b, c)|
[b, c].map do |d|
set.add(d) if Helpers.blob_multihash?(d)
end
set
end.map do |multihash|
if !store.have_blob?(multihash)
binding.pry
end
end
2020-04-19 15:26:21 +00:00
Pigeon::Parser.parse(self, tokens)
2020-04-18 14:37:43 +00:00
end
2020-04-19 15:26:21 +00:00
# === BLOBS
def get_blob(b); store.get_blob(b); end
def add_blob(b); store.add_blob(b); end
2020-04-19 15:26:21 +00:00
# === DB Management
def get_config(k); store.get_config(k); end
def add_config(k, v); store.add_config(k, v); end
2020-04-19 15:26:21 +00:00
def reset_database; store.reset; init_ident; end
private
attr_reader :store
2020-04-18 15:30:35 +00:00
def init_ident
secret = get_config(SEED_CONFIG_KEY)
if secret
@who_am_i = LocalIdentity.new(secret)
else
2020-04-18 15:30:35 +00:00
new_seed = SecureRandom.random_bytes(Ed25519::KEY_SIZE)
add_config(SEED_CONFIG_KEY, new_seed)
@who_am_i = LocalIdentity.new(new_seed)
end
end
end
end