美文网首页
Python实现linux的tail功能

Python实现linux的tail功能

作者: 53921f46e0b9 | 来源:发表于2016-12-09 16:01 被阅读1158次

基于python3,简单实现tail -n、tail -f功能

原理

  1. 把文件大小分页进行读取,这样读取大日志就无需全部加载到内存中
  2. 文件倒读输出

代码实现

#!/usr/bin/env python
# coding: utf-8
"""
原理:
    https://yq.aliyun.com/articles/60684
功能:
    实现tail -n
    实现tail -f
其它实现:
    http://www.cnblogs.com/bufferfly/p/4878688.html
    https://github.com/shengxinjing/my_blog/issues/11
BUG:
    重定向相同数据到日志文件里,使用>而不是>>的话,输入无法打印出来
"""

import os
import sys
import time

PAGE = 4096

class Tail:
    def __init__(self, filename, callback=sys.stdout.write):
        self.filename = filename
        self.callback = callback

    def reverse(self, n=10):
        """
        实现 tail -n
        """
        with open(self.filename, 'rb') as f:
            f_len = f.seek(0, 2)
            rem = f_len % PAGE
            page_n = f_len // PAGE
            r_len = rem if rem else PAGE
            while True:
                # 如果读取的页大小>=文件大小,直接读取数据输出
                if r_len >= f_len:
                    f.seek(0)
                    lines = f.readlines()[::-1]
                    break

                f.seek(-r_len, 2)
                # print('f_len: {}, rem: {}, page_n: {}, r_len: {}'.format(f_len, rem, page_n, r_len))
                lines = f.readlines()[::-1]
                count = len(lines) -1   # 末行可能不完整,减一行,加大读取量

                if count >= n:  # 如果读取到的行数>=指定行数,则退出循环读取数据
                    break
                else:   # 如果读取行数不够,载入更多的页大小读取数据
                    r_len += PAGE
                    page_n -= 1

        for line in lines[:n]:
            self.callback(line.decode('utf-8'))

    def follow(self):
        """
        实现 tail -f
        """
        with open(self.filename, 'rb') as fd:
            pos = fd.seek(0, 2)  # 打开文件时大小
            try:
                while True:
                    curr_pos = fd.seek(0,2)
                    # print('pos: {}, curr_pos: {}'.format(pos, curr_pos))
                    if pos > curr_pos:  # 表示文件数据减少或清空
                        pos = fd.seek(0, 2)
                        # time.sleep(0.3)
                        continue

                    line = fd.readline()
                    if line:
                        self.callback(line.decode('utf-8'))
                    # time.sleep(0.1)
            except KeyboardInterrupt as e:
                pass

if __name__ == '__main__':
    if len(sys.argv) != 3:
        print('Usage: {} [ -f | -# ] file'.format(sys.argv[0]), file=sys.stderr)
        raise SystemExit(1)

    if not os.path.isfile(sys.argv[2]):
        print('File does not exist.')
        raise SystemExit(1)
    else:
        tail = Tail(sys.argv[2])

    if '-f' == sys.argv[1]:
        tail.reverse()
        tail.follow()
    elif '-' in sys.argv[1]:
        try:
            n = int(sys.argv[1].strip('-'))
            tail.reverse(n)
        except ValueError:
            print('Unknown command {!r}'.format(sys.argv[1]), file=sys.stderr)
            raise SystemExit(1)
    else:
        print('Unknown command {!r}'.format(sys.argv[1]), file=sys.stderr)
        raise SystemExit(1)

测试

  1. 生成测试数据
with open('test.log', 'w+') as f:
    for l in range(1, 21):
        print('This is line {}'.format(l), file=f )
  1. 读取测试
[tail] python test_data.py                                            15:49:04
[tail] python tail.py -5 test.log                                     15:50:09
This is line 20
This is line 19
This is line 18
This is line 17
This is line 16

相关文章

  • Python实现linux的tail功能

    基于python3,简单实现tail -n、tail -f功能 原理 把文件大小分页进行读取,这样读取大日志就无需...

  • everyday:linux tail

    everyday:linux tail [TOC] 功能说明 tail 命令: 将文件从指定点开始写到标准输出。使...

  • python高性能的实现tail -n 功能

    最近有一个项目需要去读取日志文件的后面几行 ,但不想去直接调用linux系统自带的tail。这样太没有挑战性了。 ...

  • Linux 命令20200415

    LINUX 命令 linux - cat 和 tail 区别 cat 命令 和tail命令都是显示数据的命令cat...

  • everyday:linux head

    everyday:linux head [TOC] 功能说明 head 与 tail 就像它的名字一样的浅显易懂,...

  • cat,more,less,head,tail命令用法详解

    cat,more,less,head,tail都是linux中显示文件内容命令。 一、cat命令 主要功能: 1)...

  • linux日志查看

    linux 日志查看 tail、head、 cat、tac、sed、less、echo 1、命令格式: tail ...

  • NGINX 日志分析报警

    利用GO实现类似tail -f 的功能用于日志实时分析 自己实现功能 需要监听文件变化,维护之前处理的上下文调用已...

  • Linux 上的内置链表

    1 简介 在 Linux 的 中定义一系列操作不同链表的宏函数。如: 用来实现 list,tail queue、...

  • linux tail命令的使用方法详解

    摘要:本文介绍Linux下tail命令的使用方法。linux tail命令用途是依照要求将指定的文件的最后部分输出...

网友评论

      本文标题:Python实现linux的tail功能

      本文链接:https://www.haomeiwen.com/subject/mroemttx.html