Compare commits
10 Commits
ec7cfb4448
...
v0.1.0
| Author | SHA1 | Date | |
|---|---|---|---|
| d921a2b144 | |||
| 31d7c63114 | |||
| e575f1b3d3 | |||
| e55136e433 | |||
| 9b31e4081d | |||
| 017983d672 | |||
| da0032b531 | |||
| 8021dfce8a | |||
| e15a63ddb0 | |||
| dddb8648f4 |
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
/bulb
|
||||||
30
Makefile
Normal file
30
Makefile
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
.POSIX:
|
||||||
|
.SUFFIXES:
|
||||||
|
HARE=hare
|
||||||
|
HAREFLAGS=
|
||||||
|
|
||||||
|
DESTDIR=
|
||||||
|
PREFIX=/usr/local
|
||||||
|
BINDIR=$(PREFIX)/bin
|
||||||
|
MANDIR=$(PREFIX)/share/man
|
||||||
|
|
||||||
|
all: bulb
|
||||||
|
|
||||||
|
bulb:
|
||||||
|
$(HARE) build $(HAREFLAGS) -o $@ cmd/$@/
|
||||||
|
|
||||||
|
check:
|
||||||
|
$(HARE) test $(HAREFLAGS)
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -f bulb
|
||||||
|
|
||||||
|
install:
|
||||||
|
mkdir -p $(DESTDIR)$(BINDIR) $(DESTDIR)$(MANDIR)/man1
|
||||||
|
install -Dm755 bulb $(DESTDIR)$(BINDIR)/bulb
|
||||||
|
install -Dm755 doc/*.1 $(DESTDIR)$(MANDIR)/man1
|
||||||
|
|
||||||
|
uninstall:
|
||||||
|
rm -f $(DESTDIR)$(BINDIR)/bulb
|
||||||
|
|
||||||
|
.PHONY: all check clean install uninstall
|
||||||
27
README.md
27
README.md
@@ -22,26 +22,15 @@ re-writing chat history.
|
|||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
To list available boards:
|
|
||||||
|
|
||||||
```
|
|
||||||
bulb
|
|
||||||
```
|
```
|
||||||
|
bulb: bulletin board
|
||||||
|
|
||||||
To post a message:
|
Usage: bulb [-hlpu] [-b <board>] [-n <number>]
|
||||||
|
|
||||||
|
-h: print this help text
|
||||||
|
-b <board>: specify a board other than general
|
||||||
|
-l: list available boards and exit
|
||||||
|
-n <number>: display N most recent messages
|
||||||
|
-p: post a message (from stdin)
|
||||||
|
-u: operate undercover (anonymously)
|
||||||
```
|
```
|
||||||
bulb post
|
|
||||||
bulb post -b <board-name>
|
|
||||||
```
|
|
||||||
|
|
||||||
To read recent messages:
|
|
||||||
|
|
||||||
```
|
|
||||||
bulb read
|
|
||||||
bulb read -b <board-name>
|
|
||||||
bulb read -n <message-count>
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that when not specified, the board name defaults to `general`. For this
|
|
||||||
reason, it is recommended to ensure that a board with this name exists.
|
|
||||||
|
|||||||
105
cmd/bulb/main.ha
105
cmd/bulb/main.ha
@@ -1,8 +1,8 @@
|
|||||||
use bulb;
|
|
||||||
use errors;
|
use errors;
|
||||||
use fmt;
|
use fmt;
|
||||||
use fs;
|
use fs;
|
||||||
use getopt;
|
use getopt;
|
||||||
|
use internal::bulb;
|
||||||
use io;
|
use io;
|
||||||
use os;
|
use os;
|
||||||
use path;
|
use path;
|
||||||
@@ -19,82 +19,67 @@ export fn main() void = {
|
|||||||
const cmd = getopt::parse(
|
const cmd = getopt::parse(
|
||||||
os::args,
|
os::args,
|
||||||
"bulletin board",
|
"bulletin board",
|
||||||
("post", [
|
('b', "board", "specify a board other than general"),
|
||||||
('a', "post anonymously"),
|
('l', "list available boards and exit"),
|
||||||
('b', "board", "post on a board other than general"),
|
('n', "number", "display N most recent messages"),
|
||||||
]): getopt::subcmd_help,
|
('p', "post a message (from stdin)"),
|
||||||
("read", [
|
('u', "operate undercover (anonymously)"));
|
||||||
('b', "board", "read a board other than general"),
|
|
||||||
('n', "number", "display N most recent messages"),
|
|
||||||
]): getopt::subcmd_help);
|
|
||||||
defer getopt::finish(&cmd);
|
defer getopt::finish(&cmd);
|
||||||
|
|
||||||
match (cmd.subcmd) {
|
|
||||||
case let subcmd: (str, *getopt::command) =>
|
|
||||||
if (subcmd.0 == "post") {
|
|
||||||
subcmd_post(subcmd.1);
|
|
||||||
} else if (subcmd.0 == "read") {
|
|
||||||
match (subcmd_read(subcmd.1)) {
|
|
||||||
case errors::invalid =>
|
|
||||||
getopt::printhelp(os::stdout, name, cmd.help)!;
|
|
||||||
os::exit(os::status::FAILURE);
|
|
||||||
case void => void;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
case void =>
|
|
||||||
subcmd_ls(&cmd);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
fn subcmd_post(cmd: *getopt::command) void = {
|
|
||||||
let anonymous = false;
|
|
||||||
let board = default_board;
|
let board = default_board;
|
||||||
for (let opt .. cmd.opts) {
|
let list = false;
|
||||||
switch (opt.0) {
|
let number = 8;
|
||||||
case 'a' => anonymous = true;
|
let post = false;
|
||||||
case 'b' => board = opt.1;
|
let anonymous = false;
|
||||||
case => abort();
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
let user_name = if (anonymous) void else get_user_name();
|
|
||||||
let message = strings::join(" ", cmd.args...);
|
|
||||||
defer free(message);
|
|
||||||
|
|
||||||
match (bulb::post(os::stdin, board, user_name)) {
|
|
||||||
case let err: bulb::error =>
|
|
||||||
fmt::errorf("{}: Could not read {}: {}\n", name, board, bulb::strerror(err))!;
|
|
||||||
case void => void;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
fn subcmd_read(cmd: *getopt::command) (void | errors::invalid) = {
|
|
||||||
let board = default_board;
|
|
||||||
let number = 8;
|
|
||||||
for (let opt .. cmd.opts) {
|
for (let opt .. cmd.opts) {
|
||||||
switch (opt.0) {
|
switch (opt.0) {
|
||||||
case 'b' =>
|
case 'b' =>
|
||||||
board = opt.1;
|
board = opt.1;
|
||||||
|
case 'l' =>
|
||||||
|
list = true;
|
||||||
case 'n' =>
|
case 'n' =>
|
||||||
match (strconv::stoi(opt.1)) {
|
match (strconv::stoi(opt.1)) {
|
||||||
case let num: int => number = num;
|
case let num: int =>
|
||||||
case => return errors::invalid;
|
number = num;
|
||||||
|
case =>
|
||||||
|
fmt::errorf("{}: option -{} requires a number\n", name, opt.0)!;
|
||||||
|
getopt::printusage(os::stdout, name, cmd.help)!;
|
||||||
|
os::exit(os::status::FAILURE);
|
||||||
};
|
};
|
||||||
|
case 'p' =>
|
||||||
|
post = true;
|
||||||
|
case 'u' =>
|
||||||
|
anonymous = true;
|
||||||
case => abort();
|
case => abort();
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (list) {
|
||||||
|
// list boards and exit
|
||||||
|
match (bulb::list(os::stdout)) {
|
||||||
|
case let err: bulb::error =>
|
||||||
|
fmt::errorf("{}: could not list boards: {}\n", name, bulb::strerror(err))!;
|
||||||
|
os::exit(os::status::FAILURE);
|
||||||
|
case void =>
|
||||||
|
os::exit(os::status::SUCCESS);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
if (post) {
|
||||||
|
// post a message (from stdin)
|
||||||
|
let user_name = if (anonymous) void else get_user_name();
|
||||||
|
match (bulb::post(os::stdin, board, user_name)) {
|
||||||
|
case let err: bulb::error =>
|
||||||
|
fmt::errorf("{}: could not post to {}: {}\n", name, board, bulb::strerror(err))!;
|
||||||
|
case void => void;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// read latest posts on the board
|
||||||
match (bulb::read(os::stdout, board, number)) {
|
match (bulb::read(os::stdout, board, number)) {
|
||||||
case let err: bulb::error =>
|
case let err: bulb::error =>
|
||||||
fmt::errorf("{}: Could not read {}: {}\n", name, board, bulb::strerror(err))!;
|
fmt::errorf("{}: could not read {}: {}\n", name, board, bulb::strerror(err))!;
|
||||||
case void => void;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
fn subcmd_ls(cmd: *getopt::command) void = {
|
|
||||||
match (bulb::list(os::stdout)) {
|
|
||||||
case let err: bulb::error =>
|
|
||||||
fmt::errorf("{}: Could not list boards: {}\n", name, bulb::strerror(err))!;
|
|
||||||
case void => void;
|
case void => void;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
102
doc/bulb.1
Normal file
102
doc/bulb.1
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
.\" Copyright (c) 2024 Sasha Koshka <sashakoshka@tebibyte.media>
|
||||||
|
.\"
|
||||||
|
.\" This work is licensed under CC BY-SA 4.0. To see a copy of this license,
|
||||||
|
.\" visit <http://creativecommons.org/licenses/by-sa/4.0/>.
|
||||||
|
.\"
|
||||||
|
.TH BULB 1 2024-10-11
|
||||||
|
.SH NAME
|
||||||
|
bulb \(en bulletin board
|
||||||
|
.\"
|
||||||
|
.SH SYNOPSIS
|
||||||
|
|
||||||
|
bulb
|
||||||
|
.RB [ -hlpu ]
|
||||||
|
.RB [ -b\ board ]
|
||||||
|
.RB [ -n\ number ]
|
||||||
|
.\"
|
||||||
|
.SH DESCRIPTION
|
||||||
|
|
||||||
|
Read from and post to boards (text files) on the system. Posts are plain text,
|
||||||
|
multiline, and each one is tagged with the user name of its author.
|
||||||
|
.\"
|
||||||
|
.SH OPTIONS
|
||||||
|
|
||||||
|
.IP \fB-h\fP
|
||||||
|
Prints the help text and exits.
|
||||||
|
.IP \fB-b\fP\ \fIboard\fP
|
||||||
|
Specifies a board other than \(lqgeneral\(rq.
|
||||||
|
.IP \fB-l\fP
|
||||||
|
Lists available boards and exits.
|
||||||
|
.IP \fB-n\fP\ \fInumber\fP
|
||||||
|
Takes a numeric argument as the amount of recent posts to print out, the default
|
||||||
|
being 8.
|
||||||
|
.IP \fB-p\fP
|
||||||
|
Posts a message read from the standard input. See the STANDARD INPUT section.
|
||||||
|
.IP \fB-u\fP
|
||||||
|
Operates undercover. All posts made will be under the name \(lqanonymous\(rq.
|
||||||
|
.\"
|
||||||
|
.SH ENVIRONMENT
|
||||||
|
|
||||||
|
The BULBPATH environment variable specifies a colon-separated list of
|
||||||
|
directories (much like PATH) in which to search for boards. Directories at the
|
||||||
|
start take precedence over directories at the end. It has a value of
|
||||||
|
.I /var/bulb
|
||||||
|
by default.
|
||||||
|
.SH FILES
|
||||||
|
.I /var/bulb/*
|
||||||
|
.\"
|
||||||
|
.SH STANDARD INPUT
|
||||||
|
|
||||||
|
When the
|
||||||
|
.B -p
|
||||||
|
option is supplied, the standard input will be read until EOF. The resulting
|
||||||
|
text will be posted under the name of the current user in whichever board is
|
||||||
|
selected.
|
||||||
|
.\"
|
||||||
|
.SH STANDARD OUTPUT
|
||||||
|
|
||||||
|
If the
|
||||||
|
.B -l
|
||||||
|
option is supplied, a list of available boards will be written to the standard
|
||||||
|
output. If not, the last
|
||||||
|
.B -n
|
||||||
|
messages in the selected board will be written.
|
||||||
|
.\"
|
||||||
|
.SH DIAGNOSTICS
|
||||||
|
|
||||||
|
In the event of an I/O error, a debug message will be printed and the program
|
||||||
|
will exit with an error code. In the event of a bad invocation, a debug message
|
||||||
|
will be printed alongside usage text and the program will exit with an error
|
||||||
|
code.
|
||||||
|
.\"
|
||||||
|
.SH RATIONALE
|
||||||
|
|
||||||
|
The commands
|
||||||
|
.BR talk (1p),
|
||||||
|
and
|
||||||
|
.BR write (1p),
|
||||||
|
and
|
||||||
|
.BR wall (1p)
|
||||||
|
can be used effectively for textual intra-system communication, but they require
|
||||||
|
that the receiving user(s) be both logged in and monitoring their terminal for
|
||||||
|
any messages to ever get through. The
|
||||||
|
.BR bulb (1)
|
||||||
|
command was written to support asynchronous, persistent conversations between
|
||||||
|
users of the same system.
|
||||||
|
.\"
|
||||||
|
.SH AUTHOR
|
||||||
|
|
||||||
|
Written by Sasha Koshka
|
||||||
|
.MT sashakoshka@tebibyte.media
|
||||||
|
.ME .
|
||||||
|
.\"
|
||||||
|
.SH COPYRIGHT
|
||||||
|
|
||||||
|
Copyright \(co 2024 Sasha Koshka (pseudonymous). License GPLv3+: GNU GPL version
|
||||||
|
3 or later <https://gnu.org/licenses/gpl.html>
|
||||||
|
\."
|
||||||
|
.SH SEE ALSO
|
||||||
|
|
||||||
|
.BR talk (1p),
|
||||||
|
.BR write (1p),
|
||||||
|
.BR wall (1p)
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
use ascii;
|
||||||
use bufio;
|
use bufio;
|
||||||
use fs;
|
use fs;
|
||||||
use fmt;
|
use fmt;
|
||||||
@@ -6,6 +7,17 @@ use os;
|
|||||||
use path;
|
use path;
|
||||||
use strings;
|
use strings;
|
||||||
|
|
||||||
|
// TODO
|
||||||
|
// two issues:
|
||||||
|
// A: posts are stored exactly as they are formatted
|
||||||
|
// B: posts need to be sanitized from control chars going in and going out
|
||||||
|
// C: specifying a number of posts to read only works with lines because it
|
||||||
|
// cant separate out the individual posts
|
||||||
|
//
|
||||||
|
// all of these can be fixed by escaping line breaks as the post is written to
|
||||||
|
// the board, and then rendering them once they get to the reading stage.
|
||||||
|
// sanitize the text at both stages.
|
||||||
|
|
||||||
let bulb_path: []str = [];
|
let bulb_path: []str = [];
|
||||||
@init fn bulb_path() void = bulb_path = strings::split(os::tryenv("BULBPATH", "/var/bulb"), ":");
|
@init fn bulb_path() void = bulb_path = strings::split(os::tryenv("BULBPATH", "/var/bulb"), ":");
|
||||||
@fini fn bulb_path() void = free(bulb_path);
|
@fini fn bulb_path() void = free(bulb_path);
|
||||||
@@ -29,7 +41,37 @@ export fn read(output: io::handle, board: str, number: int) (void | error) = {
|
|||||||
// we add 1 to account for the blank line that will always be at the
|
// we add 1 to account for the blank line that will always be at the
|
||||||
// bottom
|
// bottom
|
||||||
seek_to_nth_last_line(file, number + 1)?;
|
seek_to_nth_last_line(file, number + 1)?;
|
||||||
io::copy(output, file)?;
|
|
||||||
|
static let output_wbuf: [os::BUFSZ]u8 = [0...];
|
||||||
|
let output = bufio::init(output, [], output_wbuf);
|
||||||
|
let file = bufio::newscanner(file);
|
||||||
|
defer bufio::finish(&file);
|
||||||
|
|
||||||
|
let saw_escape = false;
|
||||||
|
for (true) {
|
||||||
|
let byte = match (bufio::scan_byte(&file)?) {
|
||||||
|
case let byte: u8 => yield byte;
|
||||||
|
case io::EOF => break;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (byte == '\x1b') {
|
||||||
|
saw_escape = true;
|
||||||
|
} else {
|
||||||
|
if (byte == '\n') {
|
||||||
|
if (saw_escape) {
|
||||||
|
fmt::fprint(&output, "\n ")?;
|
||||||
|
} else {
|
||||||
|
fmt::fprint(&output, "\n")?;
|
||||||
|
};
|
||||||
|
} else if (!ascii::iscntrl(byte: rune)) {
|
||||||
|
let fake_buffer: [1]u8 = [byte];
|
||||||
|
io::write(&output, fake_buffer)?;
|
||||||
|
};
|
||||||
|
saw_escape = false;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
bufio::flush(&output)?;
|
||||||
};
|
};
|
||||||
|
|
||||||
export fn post(input: io::handle, board: str, user_name: (str | void)) (void | error) = {
|
export fn post(input: io::handle, board: str, user_name: (str | void)) (void | error) = {
|
||||||
@@ -44,27 +86,26 @@ export fn post(input: io::handle, board: str, user_name: (str | void)) (void | e
|
|||||||
};
|
};
|
||||||
|
|
||||||
static let file_wbuf: [os::BUFSZ]u8 = [0...];
|
static let file_wbuf: [os::BUFSZ]u8 = [0...];
|
||||||
let file = bufio::init(file, [], file_wbuf);
|
let file = bufio::init(file, [], file_wbuf);
|
||||||
let scanner = bufio::newscanner(input);
|
let input = bufio::newscanner(input);
|
||||||
defer bufio::finish(&scanner);
|
defer bufio::finish(&input);
|
||||||
|
|
||||||
let saw_break = false;
|
let saw_break = false;
|
||||||
fmt::fprintf(&file, "{}: ", user)?;
|
fmt::fprintf(&file, "{}: ", user)?;
|
||||||
for (true) {
|
for (true) {
|
||||||
let byte = match (bufio::scan_byte(&scanner)?) {
|
let byte = match (bufio::scan_byte(&input)?) {
|
||||||
case let byte: u8 => yield byte;
|
case let byte: u8 => yield byte;
|
||||||
case io::EOF => break;
|
case io::EOF => break;
|
||||||
};
|
};
|
||||||
|
|
||||||
if (saw_break) {
|
if (saw_break) {
|
||||||
fmt::fprint(&file, "\n ")?;
|
fmt::fprint(&file, "\x1b\n")?;
|
||||||
saw_break = false;
|
saw_break = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
switch (byte) {
|
if (byte == '\n') {
|
||||||
case '\n' =>
|
|
||||||
saw_break = true;
|
saw_break = true;
|
||||||
case =>
|
} else if (!ascii::iscntrl(byte: rune)) {
|
||||||
let fake_buffer: [1]u8 = [byte];
|
let fake_buffer: [1]u8 = [byte];
|
||||||
io::write(&file, fake_buffer)?;
|
io::write(&file, fake_buffer)?;
|
||||||
};
|
};
|
||||||
Reference in New Issue
Block a user