2022-02-21 17:33:58 +08:00
|
|
|
// Copyright 2022 Liuxiangchao iwind.liu@gmail.com. All rights reserved.
|
|
|
|
|
|
|
|
|
|
package caches
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
2022-03-03 19:36:28 +08:00
|
|
|
"errors"
|
2022-08-04 11:34:06 +08:00
|
|
|
"os"
|
2022-02-21 17:33:58 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// PartialRanges 内容分区范围定义
|
|
|
|
|
type PartialRanges struct {
|
2022-11-19 21:20:53 +08:00
|
|
|
ExpiresAt int64 `json:"expiresAt"` // 过期时间
|
|
|
|
|
Ranges [][2]int64 `json:"ranges"`
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewPartialRanges 获取新对象
|
2022-11-19 21:20:53 +08:00
|
|
|
func NewPartialRanges(expiresAt int64) *PartialRanges {
|
|
|
|
|
return &PartialRanges{
|
|
|
|
|
Ranges: [][2]int64{},
|
|
|
|
|
ExpiresAt: expiresAt,
|
|
|
|
|
}
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewPartialRangesFromJSON 从JSON中解析范围
|
|
|
|
|
func NewPartialRangesFromJSON(data []byte) (*PartialRanges, error) {
|
2022-11-19 21:20:53 +08:00
|
|
|
var rs = NewPartialRanges(0)
|
2022-02-21 17:33:58 +08:00
|
|
|
err := json.Unmarshal(data, &rs)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2022-03-03 19:36:28 +08:00
|
|
|
|
|
|
|
|
return rs, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func NewPartialRangesFromFile(path string) (*PartialRanges, error) {
|
2022-08-04 11:34:06 +08:00
|
|
|
data, err := os.ReadFile(path)
|
2022-03-03 19:36:28 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return NewPartialRangesFromJSON(data)
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add 添加新范围
|
|
|
|
|
func (this *PartialRanges) Add(begin int64, end int64) {
|
|
|
|
|
if begin > end {
|
|
|
|
|
begin, end = end, begin
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var nr = [2]int64{begin, end}
|
|
|
|
|
|
2022-03-03 19:36:28 +08:00
|
|
|
var count = len(this.Ranges)
|
2022-02-21 17:33:58 +08:00
|
|
|
if count == 0 {
|
2022-03-03 19:36:28 +08:00
|
|
|
this.Ranges = [][2]int64{nr}
|
2022-02-21 17:33:58 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// insert
|
|
|
|
|
// TODO 将来使用二分法改进
|
|
|
|
|
var index = -1
|
2022-03-03 19:36:28 +08:00
|
|
|
for i, r := range this.Ranges {
|
2022-02-21 17:33:58 +08:00
|
|
|
if r[0] > begin || (r[0] == begin && r[1] >= end) {
|
|
|
|
|
index = i
|
2022-03-03 19:36:28 +08:00
|
|
|
this.Ranges = append(this.Ranges, [2]int64{})
|
|
|
|
|
copy(this.Ranges[index+1:], this.Ranges[index:])
|
|
|
|
|
this.Ranges[index] = nr
|
2022-02-21 17:33:58 +08:00
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if index == -1 {
|
|
|
|
|
index = count
|
2022-03-03 19:36:28 +08:00
|
|
|
this.Ranges = append(this.Ranges, nr)
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.merge(index)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Contains 检查是否包含某个范围
|
|
|
|
|
func (this *PartialRanges) Contains(begin int64, end int64) bool {
|
2022-03-03 19:36:28 +08:00
|
|
|
if len(this.Ranges) == 0 {
|
|
|
|
|
return false
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO 使用二分法查找改进性能
|
2022-03-03 19:36:28 +08:00
|
|
|
for _, r2 := range this.Ranges {
|
2022-02-21 17:33:58 +08:00
|
|
|
if r2[0] <= begin && r2[1] >= end {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-04 22:42:03 +08:00
|
|
|
// Nearest 查找最近的某个范围
|
|
|
|
|
func (this *PartialRanges) Nearest(begin int64, end int64) (r [2]int64, ok bool) {
|
|
|
|
|
if len(this.Ranges) == 0 {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO 使用二分法查找改进性能
|
|
|
|
|
for _, r2 := range this.Ranges {
|
|
|
|
|
if r2[0] <= begin && r2[1] > begin {
|
|
|
|
|
r = [2]int64{begin, this.min(end, r2[1])}
|
|
|
|
|
ok = true
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-21 17:33:58 +08:00
|
|
|
// AsJSON 转换为JSON
|
|
|
|
|
func (this *PartialRanges) AsJSON() ([]byte, error) {
|
2022-03-03 19:36:28 +08:00
|
|
|
return json.Marshal(this)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// WriteToFile 写入到文件中
|
|
|
|
|
func (this *PartialRanges) WriteToFile(path string) error {
|
|
|
|
|
data, err := this.AsJSON()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return errors.New("convert to json failed: " + err.Error())
|
|
|
|
|
}
|
2022-08-04 11:34:06 +08:00
|
|
|
return os.WriteFile(path, data, 0666)
|
2022-03-03 19:36:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ReadFromFile 从文件中读取
|
|
|
|
|
func (this *PartialRanges) ReadFromFile(path string) (*PartialRanges, error) {
|
2022-08-04 11:34:06 +08:00
|
|
|
data, err := os.ReadFile(path)
|
2022-03-03 19:36:28 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return NewPartialRangesFromJSON(data)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (this *PartialRanges) Max() int64 {
|
|
|
|
|
if len(this.Ranges) > 0 {
|
|
|
|
|
return this.Ranges[len(this.Ranges)-1][1]
|
|
|
|
|
}
|
|
|
|
|
return 0
|
2022-02-21 17:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
2022-11-19 21:20:53 +08:00
|
|
|
func (this *PartialRanges) Reset() {
|
|
|
|
|
this.Ranges = [][2]int64{}
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-21 17:33:58 +08:00
|
|
|
func (this *PartialRanges) merge(index int) {
|
|
|
|
|
// forward
|
|
|
|
|
var lastIndex = index
|
|
|
|
|
for i := index; i >= 1; i-- {
|
2022-03-03 19:36:28 +08:00
|
|
|
var curr = this.Ranges[i]
|
|
|
|
|
var prev = this.Ranges[i-1]
|
2022-02-21 17:33:58 +08:00
|
|
|
var w1 = this.w(curr)
|
|
|
|
|
var w2 = this.w(prev)
|
|
|
|
|
if w1+w2 >= this.max(curr[1], prev[1])-this.min(curr[0], prev[0])-1 {
|
|
|
|
|
prev = [2]int64{this.min(curr[0], prev[0]), this.max(curr[1], prev[1])}
|
2022-03-03 19:36:28 +08:00
|
|
|
this.Ranges[i-1] = prev
|
|
|
|
|
this.Ranges = append(this.Ranges[:i], this.Ranges[i+1:]...)
|
2022-02-21 17:33:58 +08:00
|
|
|
lastIndex = i - 1
|
|
|
|
|
} else {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// backward
|
|
|
|
|
index = lastIndex
|
2022-03-03 19:36:28 +08:00
|
|
|
for index < len(this.Ranges)-1 {
|
|
|
|
|
var curr = this.Ranges[index]
|
|
|
|
|
var next = this.Ranges[index+1]
|
2022-02-21 17:33:58 +08:00
|
|
|
var w1 = this.w(curr)
|
|
|
|
|
var w2 = this.w(next)
|
|
|
|
|
if w1+w2 >= this.max(curr[1], next[1])-this.min(curr[0], next[0])-1 {
|
|
|
|
|
curr = [2]int64{this.min(curr[0], next[0]), this.max(curr[1], next[1])}
|
2022-03-03 19:36:28 +08:00
|
|
|
this.Ranges = append(this.Ranges[:index], this.Ranges[index+1:]...)
|
|
|
|
|
this.Ranges[index] = curr
|
2022-02-21 17:33:58 +08:00
|
|
|
} else {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (this *PartialRanges) w(r [2]int64) int64 {
|
|
|
|
|
return r[1] - r[0]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (this *PartialRanges) min(n1 int64, n2 int64) int64 {
|
|
|
|
|
if n1 <= n2 {
|
|
|
|
|
return n1
|
|
|
|
|
}
|
|
|
|
|
return n2
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (this *PartialRanges) max(n1 int64, n2 int64) int64 {
|
|
|
|
|
if n1 >= n2 {
|
|
|
|
|
return n1
|
|
|
|
|
}
|
|
|
|
|
return n2
|
|
|
|
|
}
|