initial commit

This commit is contained in:
Nils Büchner 2025-02-25 12:47:22 +01:00
commit b6ce27cf82
10 changed files with 1530 additions and 0 deletions

8
.gitignore vendored Normal file
View file

@ -0,0 +1,8 @@
# ignore binaries
/fiche
# ignore default outpit dir
code/
# ignore log files
*.log

21
.travis.yml Normal file
View file

@ -0,0 +1,21 @@
language: c
compiler:
- gcc
- clang
addons:
apt:
packages:
- cppcheck
- clang-3.5
install:
- export PYTHONUSERBASE=~/.local
- easy_install --user scan-build
- easy_install --user typing
script:
- cppcheck --enable=all --error-exitcode=1 --inconclusive main.c fiche.c
- make
- scan-build --status-bugs make -B

20
LICENSE Normal file
View file

@ -0,0 +1,20 @@
The MIT License (MIT)
Copyright (c) 2013 solusipse
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

14
Makefile Normal file
View file

@ -0,0 +1,14 @@
# for debug add -g -O0 to line below
CFLAGS+=-pthread -O2 -Wall -Wextra -Wpedantic -Wstrict-overflow -fno-strict-aliasing -std=gnu11 -g -O0
prefix=/usr/local/bin
all:
${CC} main.c fiche.c $(CFLAGS) -o fiche
install: fiche
install -m 0755 fiche $(prefix)
clean:
rm -f fiche
.PHONY: clean

336
README.md Normal file
View file

@ -0,0 +1,336 @@
fiche [![Build Status](https://travis-ci.org/solusipse/fiche.svg?branch=master)](https://travis-ci.org/solusipse/fiche)
=====
Command line pastebin for sharing terminal output.
# Client-side usage
Self-explanatory live examples (using public server):
```
echo just testing! | nc termbin.com 9999
```
```
cat file.txt | nc termbin.com 9999
```
In case you installed and started fiche on localhost:
```
ls -la | nc localhost 9999
```
You will get an url to your paste as a response, e.g.:
```
http://termbin.com/ydxh
```
You can use our beautification service to get any paste colored and numbered. Just ask for it using `l.termbin.com` subdomain, e.g.:
```
http://l.termbin.com/ydxh
```
-------------------------------------------------------------------------------
## Useful aliases
You can make your life easier by adding a termbin alias to your rc file. We list some of them here:
-------------------------------------------------------------------------------
### Pure-bash alternative to netcat
__Linux/macOS:__
```
alias tb="(exec 3<>/dev/tcp/termbin.com/9999; cat >&3; cat <&3; exec 3<&-)"
```
```
echo less typing now! | tb
```
_See [#42](https://github.com/solusipse/fiche/issues/42), [#43](https://github.com/solusipse/fiche/issues/43) for more info._
-------------------------------------------------------------------------------
### `tb` alias
__Linux (Bash):__
```
echo 'alias tb="nc termbin.com 9999"' >> .bashrc
```
```
echo less typing now! | tb
```
__macOS:__
```
echo 'alias tb="nc termbin.com 9999"' >> .bash_profile
```
```
echo less typing now! | tb
```
-------------------------------------------------------------------------------
### Copy output to clipboard
__Linux (Bash):__
```
echo 'alias tbc="netcat termbin.com 9999 | xclip -selection c"' >> .bashrc
```
```
echo less typing now! | tbc
```
__macOS:__
```
echo 'alias tbc="nc termbin.com 9999 | pbcopy"' >> .bash_profile
```
```
echo less typing now! | tbc
```
__Remember__ to reload the shell with `source ~/.bashrc` or `source ~/.bash_profile` after adding any of provided above!
-------------------------------------------------------------------------------
## Requirements
To use fiche you have to have netcat installed. You probably already have it - try typing `nc` or `netcat` into your terminal!
-------------------------------------------------------------------------------
# Server-side usage
## Installation
1. Clone:
```
git clone https://github.com/solusipse/fiche.git
```
2. Build:
```
make
```
3. Install:
```
sudo make install
```
### Using Ports on FreeBSD
To install the port: `cd /usr/ports/net/fiche/ && make install clean`. To add the package: `pkg install fiche`.
_See [#86](https://github.com/solusipse/fiche/issues/86) for more info._
-------------------------------------------------------------------------------
## Usage
```
usage: fiche [-D6epbsdSolBuw].
[-d domain] [-L listen_addr ] [-p port] [-s slug size]
[-o output directory] [-B buffer size] [-u user name]
[-l log file] [-b banlist] [-w whitelist] [-S]
```
These are command line arguments. You don't have to provide any of them to run the application. Default settings will be used in such case. See section below for more info.
### Settings
-------------------------------------------------------------------------------
#### Output directory `-o`
Relative or absolute path to the directory where you want to store user-posted pastes.
```
fiche -o ./code
```
```
fiche -o /home/www/code/
```
__Default value:__ `./code`
-------------------------------------------------------------------------------
#### Domain `-d`
This will be used as a prefix for an output received by the client.
Value will be prepended with `http`.
```
fiche -d domain.com
```
```
fiche -d subdomain.domain.com
```
```
fiche -d subdomain.domain.com/some_directory
```
__Default value:__ `localhost`
-------------------------------------------------------------------------------
#### Slug size `-s`
This will force slugs to be of required length:
```
fiche -s 6
```
__Output url with default value__: `http://localhost/xxxx`,
where x is a randomized character
__Output url with example value 6__: `http://localhost/xxxxxx`,
where x is a randomized character
__Default value:__ 4
-------------------------------------------------------------------------------
#### HTTPS `-S`
If set, fiche returns url with https prefix instead of http
```
fiche -S
```
__Output url with this parameter__: `https://localhost/xxxx`,
where x is a randomized character
-------------------------------------------------------------------------------
#### User name `-u`
Fiche will try to switch to the requested user on startup if any is provided.
```
fiche -u _fiche
```
__Default value:__ not set
__WARNING:__ This requires that fiche is started as a root.
-------------------------------------------------------------------------------
#### Buffer size `-B`
This parameter defines size of the buffer used for getting data from the user.
Maximum size (in bytes) of all input files is defined by this value.
```
fiche -B 2048
```
__Default value:__ 32768
-------------------------------------------------------------------------------
#### Log file `-l`
```
fiche -l /home/www/fiche-log.txt
```
__Default value:__ not set
__WARNING:__ this file has to be user-writable
-------------------------------------------------------------------------------
#### Ban list `-b`
Relative or absolute path to a file containing IP addresses of banned users.
```
fiche -b fiche-bans.txt
```
__Format of the file:__ this file should contain only addresses, one per line.
__Default value:__ not set
__WARNING:__ not implemented yet
-------------------------------------------------------------------------------
#### White list `-w`
If whitelist mode is enabled, only addresses from the list will be able
to upload files.
```
fiche -w fiche-whitelist.txt
```
__Format of the file:__ this file should contain only addresses, one per line.
__Default value:__ not set
__WARNING:__ not implemented yet
-------------------------------------------------------------------------------
### Running as a service
There's a simple systemd example:
```
[Unit]
Description=FICHE-SERVER
[Service]
ExecStart=/usr/local/bin/fiche -d yourdomain.com -o /path/to/output -l /path/to/log -u youruser
[Install]
WantedBy=multi-user.target
```
__WARNING:__ In service mode you have to set output directory with `-o` parameter.
-------------------------------------------------------------------------------
### Example nginx config
Fiche has no http server built-in, thus you need to setup one if you want to make files available through http.
There's a sample configuration for nginx:
```
server {
listen 80;
server_name mysite.com www.mysite.com;
charset utf-8;
location / {
root /home/www/code/;
index index.txt index.html;
}
}
```
## License
Fiche is MIT licensed.

0
extras/lines/__init__.py Normal file
View file

51
extras/lines/lines.py Normal file
View file

@ -0,0 +1,51 @@
from flask import Flask, abort, redirect
app = Flask(__name__)
import argparse, os, pygments
from pygments import highlight
from pygments.lexers import guess_lexer
from pygments.formatters import HtmlFormatter
parser = argparse.ArgumentParser()
parser.add_argument("root_dir", help="Path to directory with pastes")
args = parser.parse_args()
@app.route('/')
def main():
return redirect("http://termbin.com", code=302)
@app.route('/<slug>')
def beautify(slug):
# Return 404 in case of urls longer than 64 chars
if len(slug) > 64:
abort(404)
# Create path for the target dir
target_dir = os.path.join(args.root_dir, slug)
# Block directory traversal attempts
if not target_dir.startswith(args.root_dir):
abort(404)
# Check if directory with requested slug exists
if os.path.isdir(target_dir):
target_file = os.path.join(target_dir, "index.txt")
# File index.txt found inside that dir
with open(target_file) as f:
code = f.read()
# Identify language
lexer = guess_lexer(code)
# Create formatter with line numbers
formatter = HtmlFormatter(linenos=True, full=True)
# Return parsed code
return highlight(code, lexer, formatter)
# Not found
abort(404)
if __name__ == '__main__':
app.run()

814
fiche.c Normal file
View file

@ -0,0 +1,814 @@
/*
Fiche - Command line pastebin for sharing terminal output.
-------------------------------------------------------------------------------
License: MIT (http://www.opensource.org/licenses/mit-license.php)
Repository: https://github.com/solusipse/fiche/
Live example: http://termbin.com
-------------------------------------------------------------------------------
usage: fiche [-DepbsdolBuw].
[-D] [-e] [-d domain] [-p port] [-s slug size]
[-o output directory] [-B buffer size] [-u user name]
[-l log file] [-b banlist] [-w whitelist]
-D option is for daemonizing fiche
-e option is for using an extended character set for the URL
Compile with Makefile or manually with -O2 and -pthread flags.
To install use `make install` command.
Use netcat to push text - example:
$ cat fiche.c | nc localhost 9999
-------------------------------------------------------------------------------
*/
#include "fiche.h"
#include <stdio.h>
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
#include <pwd.h>
#include <time.h>
#include <unistd.h>
#include <pthread.h>
#include <fcntl.h>
#include <netdb.h>
#include <sys/time.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/in.h>
/******************************************************************************
* Various declarations
*/
const char *Fiche_Symbols = "abcdefghijklmnopqrstuvwxyz0123456789";
/******************************************************************************
* Inner structs
*/
struct fiche_connection {
int socket;
struct sockaddr_in6 address;
Fiche_Settings *settings;
};
/******************************************************************************
* Static function declarations
*/
// Settings-related
/**
* @brief Sets domain name
* @warning settings.domain has to be freed after using this function!
*/
static int set_domain_name(Fiche_Settings *settings);
/**
* @brief Changes user running this program to requested one
* @warning Application has to be run as root to use this function
*/
static int perform_user_change(const Fiche_Settings *settings);
// Server-related
/**
* @brief Starts server with settings provided in Fiche_Settings struct
*/
static int start_server(Fiche_Settings *settings);
/**
* @brief Dispatches incoming connections by spawning threads
*/
static void dispatch_connection(int socket, Fiche_Settings *settings);
/**
* @brief Handles connections
* @remarks Is being run by dispatch_connection in separate threads
* @arg args Struct fiche_connection containing connection details
*/
static void *handle_connection(void *args);
char* replace_substr(const char *str, const char *old, const char *new);
// Server-related utils
/**
* @brief Generates a slug that will be used for paste creation
* @warning output has to be freed after using!
*
* @arg output pointer to output string containing full path to directory
* @arg length default or user-requested length of a slug
* @arg extra_length additional length that was added to speed-up the
* generation process
*
* This function is used in connection with create_directory function
* It generates strings that are used to create a directory for
* user-provided data. If directory already exists, we ask this function
* to generate another slug with increased size.
*/
static void generate_slug(char **output, uint8_t length, uint8_t extra_length);
/**
* @brief Creates a directory at requested path using requested slug
* @returns 0 if succeded, 1 if failed or dir already existed
*
* @arg output_dir root directory for all pastes
* @arg slug directory name for a particular paste
*/
static int create_directory(char *output_dir, char *slug);
/**
* @brief Saves data to file at requested path
*
* @arg data Buffer with data received from the user
* @arg path Path at which file containing data from the buffer will be created
*/
static int save_to_file(const Fiche_Settings *s, uint8_t *data, char *slug);
// Logging-related
/**
* @brief Displays error messages
*/
static void print_error(const char *format, ...);
/**
* @brief Displays status messages
*/
static void print_status(const char *format, ...);
/**
* @brief Displays horizontal line
*/
static void print_separator();
/**
* @brief Saves connection entry to the logfile
*/
static void log_entry(const Fiche_Settings *s, const char *ip,
const char *hostname, const char *slug);
/**
* @brief Returns string containing current date
* @warning Output has to be freed!
*/
static void get_date(char *buf);
/**
* @brief Time seed
*/
unsigned int seed;
/******************************************************************************
* Public fiche functions
*/
void fiche_init(Fiche_Settings *settings) {
// Initialize everything to default values
// or to NULL in case of pointers
struct Fiche_Settings def = {
// domain
"example.com",
// output dir
"code",
// listen_addr
"::",
// port
9999,
// slug length
4,
// https
false,
// buffer length
32768,
// user name
NULL,
// path to log file
NULL,
// path to banlist
NULL,
// path to whitelist
NULL
};
// Copy default settings to provided instance
*settings = def;
}
int fiche_run(Fiche_Settings settings) {
seed = time(NULL);
// Display welcome message
{
char date[64];
get_date(date);
print_status("Starting fiche on %s...", date);
}
// Try to set requested user
if ( perform_user_change(&settings) != 0) {
print_error("Was not able to change the user!");
return -1;
}
// Check if output directory is writable
// - First we try to create it
{
mkdir(
settings.output_dir_path,
S_IRWXU | S_IRGRP | S_IROTH | S_IXOTH | S_IXGRP
);
// - Then we check if we can write there
if ( access(settings.output_dir_path, W_OK) != 0 ) {
print_error("Output directory not writable!");
return -1;
}
}
// Check if log file is writable (if set)
if ( settings.log_file_path ) {
// Create log file if it doesn't exist
FILE *f = fopen(settings.log_file_path, "a+");
fclose(f);
// Then check if it's accessible
if ( access(settings.log_file_path, W_OK) != 0 ) {
print_error("Log file not writable!");
return -1;
}
}
// Try to set domain name
if ( set_domain_name(&settings) != 0 ) {
print_error("Was not able to set domain name!");
return -1;
}
// Main loop in this method
start_server(&settings);
// Perform final cleanup
// This is allways allocated on the heap
free(settings.domain);
return 0;
}
/******************************************************************************
* Static functions below
*/
static void print_error(const char *format, ...) {
va_list args;
va_start(args, format);
printf("[Fiche][ERROR] ");
vprintf(format, args);
printf("\n");
va_end(args);
}
static void print_status(const char *format, ...) {
va_list args;
va_start(args, format);
printf("[Fiche][STATUS] ");
vprintf(format, args);
printf("\n");
va_end(args);
}
static void print_separator() {
printf("============================================================\n");
}
static void log_entry(const Fiche_Settings *s, const char *ip,
const char *hostname, const char *slug)
{
// Logging to file not enabled, finish here
if (!s->log_file_path) {
return;
}
FILE *f = fopen(s->log_file_path, "a");
if (!f) {
print_status("Was not able to save entry to the log!");
return;
}
char date[64];
get_date(date);
// Write entry to file
fprintf(f, "%s -- %s -- %s (%s)\n", slug, date, ip, hostname);
fclose(f);
}
static void get_date(char *buf) {
struct tm curtime;
time_t ltime;
ltime=time(&ltime);
localtime_r(&ltime, &curtime);
// Save data to provided buffer
if (asctime_r(&curtime, buf) == 0) {
// Couldn't get date, setting first byte of the
// buffer to zero so it won't be displayed
buf[0] = 0;
return;
}
// Remove newline char
buf[strlen(buf)-1] = 0;
}
static int set_domain_name(Fiche_Settings *settings) {
char *prefix = "";
if (settings->https) {
prefix = "https://";
} else {
prefix = "http://";
}
const int len = strlen(settings->domain) + strlen(prefix) + 1;
char *b = malloc(len);
if (!b) {
return -1;
}
strcpy(b, prefix);
strcat(b, settings->domain);
settings->domain = b;
print_status("Domain set to: %s.", settings->domain);
return 0;
}
static int perform_user_change(const Fiche_Settings *settings) {
// User change wasn't requested, finish here
if (settings->user_name == NULL) {
return 0;
}
// Check if root, if not - finish here
if (getuid() != 0) {
print_error("Run as root if you want to change the user!");
return -1;
}
// Get user details
const struct passwd *userdata = getpwnam(settings->user_name);
const int uid = userdata->pw_uid;
const int gid = userdata->pw_gid;
if (uid == -1 || gid == -1) {
print_error("Could find requested user: %s!", settings->user_name);
return -1;
}
if (setgid(gid) != 0) {
print_error("Couldn't switch to requested user: %s!", settings->user_name);
}
if (setuid(uid) != 0) {
print_error("Couldn't switch to requested user: %s!", settings->user_name);
}
print_status("User changed to: %s.", settings->user_name);
return 0;
}
static int start_server(Fiche_Settings *settings) {
// Perform socket creation
int s = socket(AF_INET6, SOCK_STREAM, 0);
if (s < 0) {
print_error("Couldn't create a socket!");
return -1;
}
// Set socket settings
if ( setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &(int){ 1 } , sizeof(int)) != 0 ) {
print_error("Couldn't prepare the socket!");
return -1;
}
// Prepare address and port handler for IPv6
struct sockaddr_in6 address;
address.sin6_family = AF_INET6;
address.sin6_addr = in6addr_any; // Bind to any address, for a specific address use inet_pton
address.sin6_port = htons(settings->port);
// Convert IPv4 address to IPv6 if needed
if (settings->listen_addr) {
if (inet_pton(AF_INET6, settings->listen_addr, &address.sin6_addr) != 1) {
print_error("Invalid IPv6 address!");
return -1;
}
}
// Bind to port
if (bind(s, (struct sockaddr *) &address, sizeof(address)) != 0) {
print_error("Couldn't bind to the port: %d!", settings->port);
return -1;
}
// Start listening
if (listen(s, 128) != 0) {
print_error("Couldn't start listening on the socket!");
return -1;
}
print_status("Server started listening on: %s:%d.",
settings->listen_addr, settings->port);
print_separator();
// Run dispatching loop
while (1) {
dispatch_connection(s, settings);
}
// Give some time for all threads to finish
// NOTE: this code is reached only in testing environment
// There is currently no way to kill the main thread from any thread
// Something like this can be done for testing purpouses:
// int i = 0;
// while (i < 3) {
// dispatch_connection(s, settings);
// i++;
// }
sleep(5);
return 0;
}
static void dispatch_connection(int socket, Fiche_Settings *settings) {
// Create address structs for this socket
struct sockaddr_in6 address;
socklen_t addlen = sizeof(address);
// Accept a connection and get a new socket id
const int s = accept(socket, (struct sockaddr *) &address, &addlen);
if (s < 0 ) {
print_error("Error on accepting connection!");
return;
}
// Set timeout for accepted socket
const struct timeval timeout = { 5, 0 };
if ( setsockopt(s, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout)) != 0 ) {
print_error("Couldn't set a timeout!");
}
if ( setsockopt(s, SOL_SOCKET, SO_SNDTIMEO, &timeout, sizeof(timeout)) != 0 ) {
print_error("Couldn't set a timeout!");
}
// Create an argument for the thread function
struct fiche_connection *c = malloc(sizeof(*c));
if (!c) {
print_error("Couldn't allocate memory!");
return;
}
c->socket = s;
c->address = address;
c->settings = settings;
// Spawn a new thread to handle this connection
pthread_t id;
if ( pthread_create(&id, NULL, &handle_connection, c) != 0 ) {
print_error("Couldn't spawn a thread!");
return;
}
// Detach thread if created succesfully
// TODO: consider using pthread_tryjoin_np
pthread_detach(id);
}
char* replace_substr(const char *str, const char *old, const char *new) {
char *result;
int i, count = 0;
int newlen = strlen(new);
int oldlen = strlen(old);
// Counting the number of times the old substring occurs in the string
for (i = 0; str[i] != '\0'; i++) {
if (strstr(&str[i], old) == &str[i]) {
count++;
i += oldlen - 1;
}
}
// Allocating memory for the new string
result = (char *)malloc(i + count * (newlen - oldlen) + 1);
i = 0;
while (*str) {
// Compare the substring with the result
if (strstr(str, old) == str) {
strcpy(&result[i], new);
i += newlen;
str += oldlen;
} else {
result[i++] = *str++;
}
}
result[i] = '\0';
return result;
}
static void *handle_connection(void *args) {
// Cast args to it's previous type
struct fiche_connection *c = (struct fiche_connection *) args;
//struct sockaddr_in6 *addr = (struct sockaddr_in6 *)&c->address;
char ipstr[INET6_ADDRSTRLEN];
char *ip;
struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)&c->address;
inet_ntop(AF_INET6, &addr6->sin6_addr, ipstr, sizeof(ipstr));
ip = replace_substr(ipstr, "::ffff:", "");
// Get client's hostname
char hostname[1024];
if (getnameinfo((struct sockaddr *)&c->address, sizeof(c->address),
hostname, sizeof(hostname), NULL, 0, 0) != 0 ) {
// Couldn't resolve a hostname
strcpy(hostname, "n/a");
}
// Print status on this connection
{
char date[64];
get_date(date);
print_status("%s", date);
print_status("Incoming connection from: %s (%s).", ip, hostname);
}
// Create a buffer
uint8_t buffer[c->settings->buffer_len];
memset(buffer, 0, c->settings->buffer_len);
const int r = recv(c->socket, buffer, sizeof(buffer), MSG_WAITALL);
if (r <= 0) {
print_error("No data received from the client!");
print_separator();
// Close the socket
close(c->socket);
// Cleanup
free(c);
free(ip);
pthread_exit(NULL);
return 0;
}
char *slug;
uint8_t extra = 0;
do {
// Generate slugs until it's possible to create a directory
// with generated slug on disk
generate_slug(&slug, c->settings->slug_len, extra);
// Something went wrong in slug generation, break here
if (!slug) {
break;
}
// Increment counter for additional letters needed
++extra;
// If i was incremented more than 128 times, something
// for sure went wrong. We are closing connection and
// killing this thread in such case
if (extra > 128) {
print_error("Couldn't generate a valid slug!");
print_separator();
// Cleanup
close(c->socket);
free(c);
free(slug);
pthread_exit(NULL);
return NULL;
}
}
while(create_directory(c->settings->output_dir_path, slug) != 0);
// Slug generation failed, we have to finish here
if (!slug) {
print_error("Couldn't generate a slug!");
print_separator();
close(c->socket);
// Cleanup
free(c);
pthread_exit(NULL);
return NULL;
}
// Save to file failed, we have to finish here
if ( save_to_file(c->settings, buffer, slug) != 0 ) {
print_error("Couldn't save a file!");
print_separator();
close(c->socket);
// Cleanup
free(c);
free(slug);
pthread_exit(NULL);
return NULL;
}
// Write a response to the user
{
// Create an url (additional byte for slash and one for new line)
const size_t len = strlen(c->settings->domain) + strlen(slug) + 3;
char url[len];
snprintf(url, len, "%s%s%s%s", c->settings->domain, "/", slug, "\n");
// Send the response
write(c->socket, url, len);
}
print_status("Received %d bytes, saved to: %s.", r, slug);
print_separator();
// Log connection
// TODO: log unsuccessful and rejected connections
log_entry(c->settings, ip, hostname, slug);
// Close the connection
close(c->socket);
// Perform cleanup of values used in this thread
free(slug);
free(c);
pthread_exit(NULL);
return NULL;
}
static void generate_slug(char **output, uint8_t length, uint8_t extra_length) {
// Realloc buffer for slug when we want it to be bigger
// This happens in case when directory with this name already
// exists. To save time, we don't generate new slugs until
// we spot an available one. We add another letter instead.
if (extra_length > 0) {
free(*output);
}
// Create a buffer for slug with extra_length if any
*output = calloc(length + 1 + extra_length, sizeof(char));
if (*output == NULL) {
return;
}
// Take n-th symbol from symbol table and use it for slug generation
for (int i = 0; i < length + extra_length; i++) {
int n = rand_r(&seed) % strlen(Fiche_Symbols);
*(output[0] + sizeof(char) * i) = Fiche_Symbols[n];
}
}
static int create_directory(char *output_dir, char *slug) {
if (!slug) {
return -1;
}
// Additional byte is for the slash
size_t len = strlen(output_dir) + strlen(slug) + 2;
// Generate a path
char *path = malloc(len);
if (!path) {
return -1;
}
snprintf(path, len, "%s%s%s", output_dir, "/", slug);
// Create output directory, just in case
mkdir(output_dir, S_IRWXU | S_IRGRP | S_IROTH | S_IXOTH | S_IXGRP);
// Create slug directory
const int r = mkdir(
path,
S_IRWXU | S_IRGRP | S_IROTH | S_IXOTH | S_IXGRP
);
free(path);
return r;
}
static int save_to_file(const Fiche_Settings *s, uint8_t *data, char *slug) {
char *file_name = "index.txt";
// Additional 2 bytes are for 2 slashes
size_t len = strlen(s->output_dir_path) + strlen(slug) + strlen(file_name) + 3;
// Generate a path
char *path = malloc(len);
if (!path) {
return -1;
}
snprintf(path, len, "%s%s%s%s%s", s->output_dir_path, "/", slug, "/", file_name);
// Attempt file saving
FILE *f = fopen(path, "w");
if (!f) {
free(path);
return -1;
}
// Null-terminate buffer if not null terminated already
data[s->buffer_len - 1] = 0;
if ( fprintf(f, "%s", data) < 0 ) {
fclose(f);
free(path);
return -1;
}
fclose(f);
free(path);
return 0;
}

119
fiche.h Normal file
View file

@ -0,0 +1,119 @@
/*
Fiche - Command line pastebin for sharing terminal output.
-------------------------------------------------------------------------------
License: MIT (http://www.opensource.org/licenses/mit-license.php)
Repository: https://github.com/solusipse/fiche/
Live example: http://termbin.com
-------------------------------------------------------------------------------
usage: fiche [-DepbsdolBuw].
[-D] [-e] [-d domain] [-p port] [-s slug size]
[-o output directory] [-B buffer size] [-u user name]
[-l log file] [-b banlist] [-w whitelist]
Use netcat to push text - example:
$ cat fiche.c | nc localhost 9999
-------------------------------------------------------------------------------
*/
#ifndef FICHE_H
#define FICHE_H
#include <stdint.h>
#include <stdbool.h>
/**
* @brief Used as a container for fiche settings. Create before
* the initialization
*
*/
typedef struct Fiche_Settings {
/**
* @brief Domain used in output links
*/
char *domain;
/**
* @brief Path to directory used for storing uploaded pastes
*/
char *output_dir_path;
/**
* @brief Address on which fiche is waiting for connections
*/
char *listen_addr;
/**
* @brief Port on which fiche is waiting for connections
*/
uint16_t port;
/**
* @brief Length of a paste's name
*/
uint8_t slug_len;
/**
* @brief If set, returns url with https prefix instead of http
*/
bool https;
/**
* @brief Connection buffer length
*
* @remarks Length of this buffer limits max size of uploaded files
*/
uint32_t buffer_len;
/**
* @brief Name of the user that runs fiche process
*/
char *user_name;
/**
* @brief Path to the log file
*/
char *log_file_path;
/**
* @brief Path to the file with banned IPs
*/
char *banlist_path;
/**
* @brief Path to the file with whitelisted IPs
*/
char *whitelist_path;
} Fiche_Settings;
/**
* @brief Initializes Fiche_Settings instance
*/
void fiche_init(Fiche_Settings *settings);
/**
* @brief Runs fiche server
*
* @return 0 if it was able to start, any other value otherwise
*/
int fiche_run(Fiche_Settings settings);
/**
* @brief array of symbols used in slug generation
* @remarks defined in fiche.c
*/
extern const char *Fiche_Symbols;
#endif

147
main.c Normal file
View file

@ -0,0 +1,147 @@
/*
Fiche - Command line pastebin for sharing terminal output.
-------------------------------------------------------------------------------
License: MIT (http://www.opensource.org/licenses/mit-license.php)
Repository: https://github.com/solusipse/fiche/
Live example: http://termbin.com
-------------------------------------------------------------------------------
usage: fiche [-DepbsdolBuw].
[-D] [-e] [-d domain] [-p port] [-s slug size]
[-o output directory] [-B buffer size] [-u user name]
[-l log file] [-b banlist] [-w whitelist]
Use netcat to push text - example:
$ cat fiche.c | nc localhost 9999
-------------------------------------------------------------------------------
*/
#include "fiche.h"
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <getopt.h>
int main(int argc, char **argv) {
// Fiche settings instance
Fiche_Settings fs;
// Initialize settings instance to default values
fiche_init(&fs);
// Note: fiche_run is responsible for checking if these values
// were set correctly
// Note: according to getopt documentation, we don't need to
// copy strings, so we decided to go with pointer approach for these
// Parse input arguments
int c;
while ((c = getopt(argc, argv, "D6eSL:p:b:s:d:o:l:B:u:w:")) != -1) {
switch (c) {
// domain
case 'd':
{
fs.domain = optarg;
}
break;
// port
case 'p':
{
fs.port = atoi(optarg);
}
break;
// listen_addr
case 'L':
{
fs.listen_addr = optarg;
}
break;
// slug size
case 's':
{
fs.slug_len = atoi(optarg);
}
break;
// https
case 'S':
{
fs.https = true;
}
break;
// output directory path
case 'o':
{
fs.output_dir_path = optarg;
}
break;
// buffer size
case 'B':
{
fs.buffer_len = atoi(optarg);
}
break;
// user name
case 'u':
{
fs.user_name = optarg;
}
break;
// log file path
case 'l':
{
fs.log_file_path = optarg;
}
break;
// banlist file path
case 'b':
{
fs.banlist_path = optarg;
}
break;
// whitelist file path
case 'w':
{
fs.whitelist_path = optarg;
}
break;
// Display help in case of any unsupported argument
default:
{
printf("usage: fiche [-dLpsSoBulbw].\n");
printf(" [-d domain] [-L listen_addr] [-p port] [-s slug size]\n");
printf(" [-o output directory] [-B buffer size] [-u user name]\n");
printf(" [-l log file] [-b banlist] [-w whitelist] [-S]\n");
return 0;
}
break;
}
}
fiche_run(fs);
return 0;
}