initial commit
This commit is contained in:
commit
b6ce27cf82
10 changed files with 1530 additions and 0 deletions
8
.gitignore
vendored
Normal file
8
.gitignore
vendored
Normal file
|
@ -0,0 +1,8 @@
|
|||
# ignore binaries
|
||||
/fiche
|
||||
|
||||
# ignore default outpit dir
|
||||
code/
|
||||
|
||||
# ignore log files
|
||||
*.log
|
21
.travis.yml
Normal file
21
.travis.yml
Normal 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
20
LICENSE
Normal 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
14
Makefile
Normal 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
336
README.md
Normal file
|
@ -0,0 +1,336 @@
|
|||
fiche [](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
0
extras/lines/__init__.py
Normal file
51
extras/lines/lines.py
Normal file
51
extras/lines/lines.py
Normal 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
814
fiche.c
Normal 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(<ime);
|
||||
localtime_r(<ime, &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
119
fiche.h
Normal 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
147
main.c
Normal 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;
|
||||
}
|
||||
|
||||
|
Loading…
Add table
Reference in a new issue