/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

#include "prio.h"
#include "prprf.h"
#include "prinit.h"
#include "prthread.h"
#include "prinrval.h"

#include "plerror.h"
#include "plgetopt.h"

#include <stdlib.h>

#define BUFFER_SIZE 500

static PRFileDesc *out = NULL, *err = NULL;

static void Help(void)
{
    PR_fprintf(err, "Usage: tail [-n <n>] [-f] [-h] <filename>\n");
    PR_fprintf(err, "\t-t <n>	Dally time in milliseconds\n");
    PR_fprintf(err, "\t-n <n>	Number of bytes before <eof>\n");
    PR_fprintf(err, "\t-f   	Follow the <eof>\n");
    PR_fprintf(err, "\t-h   	This message and nothing else\n");
}  /* Help */

PRIntn main(PRIntn argc, char **argv)
{
    PRIntn rv = 0;
    PLOptStatus os;
    PRStatus status;
    PRFileDesc *file;
    PRFileInfo fileInfo;
    PRIntervalTime dally;
    char buffer[BUFFER_SIZE];
    PRBool follow = PR_FALSE;
    const char *filename = NULL;
    PRUint32 position = 0, seek = 0, time = 0;
    PLOptState *opt = PL_CreateOptState(argc, argv, "hfn:");

    out = PR_GetSpecialFD(PR_StandardOutput);
    err = PR_GetSpecialFD(PR_StandardError);

    while (PL_OPT_EOL != (os = PL_GetNextOpt(opt)))
    {
        if (PL_OPT_BAD == os) {
            continue;
        }
        switch (opt->option)
        {
            case 0:  /* it's the filename */
                filename = opt->value;
                break;
            case 'n':  /* bytes before end of file */
                seek = atoi(opt->value);
                break;
            case 't':  /* dally time */
                time = atoi(opt->value);
                break;
            case 'f':  /* follow the end of file */
                follow = PR_TRUE;
                break;
            case 'h':  /* user wants some guidance */
                Help();  /* so give him an earful */
                return 2;  /* but not a lot else */
                break;
            default:
                break;
        }
    }
    PL_DestroyOptState(opt);

    if (0 == time) {
        time = 1000;
    }
    dally = PR_MillisecondsToInterval(time);

    if (NULL == filename)
    {
        (void)PR_fprintf(out, "Input file not specified\n");
        rv = 1; goto done;
    }
    file = PR_Open(filename, PR_RDONLY, 0);
    if (NULL == file)
    {
        PL_FPrintError(err, "File cannot be opened for reading");
        return 1;
    }

    status = PR_GetOpenFileInfo(file, &fileInfo);
    if (PR_FAILURE == status)
    {
        PL_FPrintError(err, "Cannot acquire status of file");
        rv = 1; goto done;
    }
    if (seek > 0)
    {
        if (seek > fileInfo.size) {
            seek = 0;
        }
        position = PR_Seek(file, (fileInfo.size - seek), PR_SEEK_SET);
        if (-1 == (PRInt32)position) {
            PL_FPrintError(err, "Cannot seek to starting position");
        }
    }

    do
    {
        while (position < fileInfo.size)
        {
            PRInt32 read, bytes = fileInfo.size - position;
            if (bytes > sizeof(buffer)) {
                bytes = sizeof(buffer);
            }
            read = PR_Read(file, buffer, bytes);
            if (read != bytes) {
                PL_FPrintError(err, "Cannot read to eof");
            }
            position += read;
            PR_Write(out, buffer, read);
        }

        if (follow)
        {
            PR_Sleep(dally);
            status = PR_GetOpenFileInfo(file, &fileInfo);
            if (PR_FAILURE == status)
            {
                PL_FPrintError(err, "Cannot acquire status of file");
                rv = 1; goto done;
            }
        }
    } while (follow);

done:
    PR_Close(file);

    return rv;
}  /* main */

/* tail.c */