Files
HackBrowserData/browser/chromium/extract_bookmark.go
T
Roger 068b82178f fix: improve extract parsing with proper decoding and error handling (#543)
* fix: implement proper Chromium localStorage LevelDB parsing
* feat: add IsMeta field to StorageEntry and keep META entries
* fix: add error logging for decryption and missing data fields
* fix: address PR review for localStorage parsing
* fix: use naïve instead of café in Latin-1 test to avoid typos false positive
* fix: extension enabled detection and sessionStorage decoding
* fix: session storage origin resolution and extension enabled detection
* fix: address PR review comments for storage parsing
2026-04-04 18:52:54 +08:00

55 lines
1.3 KiB
Go

package chromium
import (
"os"
"sort"
"github.com/tidwall/gjson"
"github.com/moond4rk/hackbrowserdata/types"
"github.com/moond4rk/hackbrowserdata/utils/typeutil"
)
func extractBookmarks(path string) ([]types.BookmarkEntry, error) {
data, err := os.ReadFile(path)
if err != nil {
return nil, err
}
var bookmarks []types.BookmarkEntry
roots := gjson.GetBytes(data, "roots")
roots.ForEach(func(_, value gjson.Result) bool {
walkBookmarks(value, "", &bookmarks)
return true
})
sort.Slice(bookmarks, func(i, j int) bool {
return bookmarks[i].CreatedAt.After(bookmarks[j].CreatedAt)
})
return bookmarks, nil
}
// walkBookmarks recursively traverses the bookmark tree, collecting URL entries.
func walkBookmarks(node gjson.Result, folder string, out *[]types.BookmarkEntry) {
nodeType := node.Get("type").String()
if nodeType == "url" {
*out = append(*out, types.BookmarkEntry{
ID: node.Get("id").Int(),
Name: node.Get("name").String(),
Type: nodeType,
URL: node.Get("url").String(),
Folder: folder,
CreatedAt: typeutil.TimeEpoch(node.Get("date_added").Int()),
})
}
children := node.Get("children")
if !children.Exists() || !children.IsArray() {
return
}
currentFolder := node.Get("name").String()
for _, child := range children.Array() {
walkBookmarks(child, currentFolder, out)
}
}