package core import ( "errors" "fmt" "os" "path/filepath" "strings" "hack-browser-data/core/data" "hack-browser-data/log" ) const ( chromeName = "Chrome" chromeBetaName = "Chrome Beta" chromiumName = "Chromium" edgeName = "Microsoft Edge" firefoxName = "Firefox" speed360Name = "360speed" qqBrowserName = "qq" braveName = "Brave" operaName = "Opera" operaGXName = "OperaGX" vivaldiName = "Vivaldi" ) type Browser interface { // InitSecretKey is init chrome secret key, firefox's key always empty InitSecretKey() error // GetName return browser name GetName() string // GetSecretKey return browser secret key GetSecretKey() []byte // GetAllItems return all items (password|bookmark|cookie|history) GetAllItems() ([]data.Item, error) // GetItem return single one from the password|bookmark|cookie|history GetItem(itemName string) (data.Item, error) } const ( cookie = "cookie" history = "history" bookmark = "bookmark" download = "download" password = "password" creditcard = "creditcard" ) var ( errItemNotSupported = errors.New(`item not supported, default is "all", choose from history|download|password|bookmark|cookie`) errBrowserNotSupported = errors.New("browser not supported") errChromeSecretIsEmpty = errors.New("chrome secret is empty") errDbusSecretIsEmpty = errors.New("dbus secret key is empty") ) var ( chromiumItems = map[string]struct { mainFile string newItem func(mainFile, subFile string) data.Item }{ bookmark: { mainFile: data.ChromeBookmarkFile, newItem: data.NewBookmarks, }, cookie: { mainFile: data.ChromeCookieFile, newItem: data.NewCookies, }, history: { mainFile: data.ChromeHistoryFile, newItem: data.NewHistoryData, }, download: { mainFile: data.ChromeDownloadFile, newItem: data.NewDownloads, }, password: { mainFile: data.ChromePasswordFile, newItem: data.NewCPasswords, }, creditcard: { mainFile: data.ChromeCreditFile, newItem: data.NewCCards, }, } firefoxItems = map[string]struct { mainFile string subFile string newItem func(mainFile, subFile string) data.Item }{ bookmark: { mainFile: data.FirefoxDataFile, newItem: data.NewBookmarks, }, cookie: { mainFile: data.FirefoxCookieFile, newItem: data.NewCookies, }, history: { mainFile: data.FirefoxDataFile, newItem: data.NewHistoryData, }, download: { mainFile: data.FirefoxDataFile, newItem: data.NewDownloads, }, password: { mainFile: data.FirefoxKey4File, subFile: data.FirefoxLoginFile, newItem: data.NewFPasswords, }, } ) type Chromium struct { name string profilePath string keyPath string storage string // storage use for linux and macOS, get secret key secretKey []byte } // NewChromium return Chromium browser interface func NewChromium(profile, key, name, storage string) (Browser, error) { return &Chromium{profilePath: profile, keyPath: key, name: name, storage: storage}, nil } func (c *Chromium) GetName() string { return c.name } func (c *Chromium) GetSecretKey() []byte { return c.secretKey } // GetAllItems return all chromium items from browser // if can't find the item path, log error then continue func (c *Chromium) GetAllItems() ([]data.Item, error) { var items []data.Item for item, choice := range chromiumItems { m, err := getItemPath(c.profilePath, choice.mainFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", c.name, item, err) continue } i := choice.newItem(m, "") log.Debugf("%s find %s File Success", c.name, item) items = append(items, i) } return items, nil } // GetItem return single item func (c *Chromium) GetItem(itemName string) (data.Item, error) { itemName = strings.ToLower(itemName) if item, ok := chromiumItems[itemName]; ok { m, err := getItemPath(c.profilePath, item.mainFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", c.name, item.mainFile, err) } i := item.newItem(m, "") return i, nil } else { return nil, errItemNotSupported } } type Firefox struct { name string profilePath string keyPath string } // NewFirefox return firefox browser interface func NewFirefox(profile, key, name, storage string) (Browser, error) { return &Firefox{profilePath: profile, keyPath: key, name: name}, nil } // GetAllItems return all item with firefox func (f *Firefox) GetAllItems() ([]data.Item, error) { var items []data.Item for item, choice := range firefoxItems { var ( sub, main string err error ) if choice.subFile != "" { sub, err = getItemPath(f.profilePath, choice.subFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", f.name, item, err) continue } } main, err = getItemPath(f.profilePath, choice.mainFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", f.name, item, err) continue } i := choice.newItem(main, sub) log.Debugf("%s find %s file success", f.name, item) items = append(items, i) } return items, nil } func (f *Firefox) GetItem(itemName string) (data.Item, error) { itemName = strings.ToLower(itemName) if item, ok := firefoxItems[itemName]; ok { var ( sub, main string err error ) if item.subFile != "" { sub, err = getItemPath(f.profilePath, item.subFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", f.name, item.subFile, err) } } main, err = getItemPath(f.profilePath, item.mainFile) if err != nil { log.Debugf("%s find %s file failed, ERR:%s", f.name, item.mainFile, err) } i := item.newItem(main, sub) log.Debugf("%s find %s file success", f.name, item.mainFile) return i, nil } else { return nil, errItemNotSupported } } func (f *Firefox) GetName() string { return f.name } // GetSecretKey for firefox is always nil // this method used to implement Browser interface func (f *Firefox) GetSecretKey() []byte { return nil } // InitSecretKey for firefox is always nil // this method used to implement Browser interface func (f *Firefox) InitSecretKey() error { return nil } // PickBrowser return a list of browser interface func PickBrowser(name string) ([]Browser, error) { var browsers []Browser name = strings.ToLower(name) if name == "all" { for _, v := range browserList { b, err := v.New(v.ProfilePath, v.KeyPath, v.Name, v.Storage) if err != nil { log.Error(err) } browsers = append(browsers, b) } return browsers, nil } else if choice, ok := browserList[name]; ok { b, err := choice.New(choice.ProfilePath, choice.KeyPath, choice.Name, choice.Storage) browsers = append(browsers, b) return browsers, err } return nil, errBrowserNotSupported } // PickCustomBrowser pick single browser with custom browser profile path and key file path (windows only). // If custom key file path is empty, but the current browser requires key file (chromium for windows version > 80) // key file path will be automatically found in the profile path's parent directory. func PickCustomBrowser(browserName, cusProfile, cusKey string) ([]Browser, error) { var ( browsers []Browser ) browserName = strings.ToLower(browserName) supportBrowser := strings.Join(ListBrowser(), "|") if browserName == "all" { return nil, fmt.Errorf("can't select all browser, pick one from %s with -b flag\n", supportBrowser) } if choice, ok := browserList[browserName]; ok { // if this browser need key path if choice.KeyPath != "" { var err error // if browser need key path and cusKey is empty, try to get key path with profile dir if cusKey == "" { cusKey, err = getKeyPath(cusProfile) if err != nil { return nil, err } } if err := checkKeyPath(cusKey); err != nil { return nil, err } } b, err := choice.New(cusProfile, cusKey, choice.Name, choice.Storage) browsers = append(browsers, b) return browsers, err } else { return nil, fmt.Errorf("%s not support, pick one from %s with -b flag\n", browserName, supportBrowser) } } func getItemPath(profilePath, file string) (string, error) { p, err := filepath.Glob(filepath.Join(profilePath, file)) if err != nil { return "", err } if len(p) > 0 { return p[0], nil } return "", fmt.Errorf("find %s failed", file) } // getKeyPath try to get key file path with the browser's profile path // default key file path is in the parent directory of the profile dir, and name is [Local State] func getKeyPath(profilePath string) (string, error) { if _, err := os.Stat(filepath.Clean(profilePath)); os.IsNotExist(err) { return "", err } parentDir := getParentDirectory(profilePath) keyPath := filepath.Join(parentDir, "Local State") return keyPath, nil } // check key file path is exist func checkKeyPath(keyPath string) error { if _, err := os.Stat(keyPath); os.IsNotExist(err) { return fmt.Errorf("secret key path not exist, please check %s", keyPath) } return nil } func getParentDirectory(dir string) string { var ( length int ) // filepath.Clean(dir) auto remove dir = strings.ReplaceAll(filepath.Clean(dir), `\`, `/`) length = strings.LastIndex(dir, "/") if length > 0 { if length > len([]rune(dir)) { length = len([]rune(dir)) } return string([]rune(dir)[:length]) } return "" } func ListBrowser() []string { var l []string for k := range browserList { l = append(l, k) } return l } func ListItem() []string { var l []string for k := range chromiumItems { l = append(l, k) } return l }