123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100 |
- /*
- Copyright 2017 The Kubernetes Authors.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package tail
- import (
- "bytes"
- "io"
- "io/ioutil"
- "os"
- )
- const (
- // blockSize is the block size used in tail.
- blockSize = 1024
- )
- var (
- // eol is the end-of-line sign in the log.
- eol = []byte{'\n'}
- )
- // ReadAtMost reads at most max bytes from the end of the file identified by path or
- // returns an error. It returns true if the file was longer than max. It will
- // allocate up to max bytes.
- func ReadAtMost(path string, max int64) ([]byte, bool, error) {
- f, err := os.Open(path)
- if err != nil {
- return nil, false, err
- }
- defer f.Close()
- fi, err := f.Stat()
- if err != nil {
- return nil, false, err
- }
- size := fi.Size()
- if size == 0 {
- return nil, false, nil
- }
- if size < max {
- max = size
- }
- offset, err := f.Seek(-max, io.SeekEnd)
- if err != nil {
- return nil, false, err
- }
- data, err := ioutil.ReadAll(f)
- return data, offset > 0, err
- }
- // FindTailLineStartIndex returns the start of last nth line.
- // * If n < 0, return the beginning of the file.
- // * If n >= 0, return the beginning of last nth line.
- // Notice that if the last line is incomplete (no end-of-line), it will not be counted
- // as one line.
- func FindTailLineStartIndex(f io.ReadSeeker, n int64) (int64, error) {
- if n < 0 {
- return 0, nil
- }
- size, err := f.Seek(0, io.SeekEnd)
- if err != nil {
- return 0, err
- }
- var left, cnt int64
- buf := make([]byte, blockSize)
- for right := size; right > 0 && cnt <= n; right -= blockSize {
- left = right - blockSize
- if left < 0 {
- left = 0
- buf = make([]byte, right)
- }
- if _, err := f.Seek(left, io.SeekStart); err != nil {
- return 0, err
- }
- if _, err := f.Read(buf); err != nil {
- return 0, err
- }
- cnt += int64(bytes.Count(buf, eol))
- }
- for ; cnt > n; cnt-- {
- idx := bytes.Index(buf, eol) + 1
- buf = buf[idx:]
- left += int64(idx)
- }
- return left, nil
- }
|