mirror of
https://github.com/safing/portmaster
synced 2025-09-02 02:29:12 +00:00
Add support for finding app icons on Linux (MVP)
This commit is contained in:
parent
af712382f8
commit
c999d5559a
11 changed files with 276 additions and 17 deletions
|
@ -10,6 +10,7 @@ import (
|
|||
"github.com/safing/portbase/api"
|
||||
"github.com/safing/portbase/formats/dsd"
|
||||
"github.com/safing/portbase/utils"
|
||||
"github.com/safing/portmaster/profile/icons"
|
||||
)
|
||||
|
||||
func registerAPIEndpoints() error {
|
||||
|
@ -98,7 +99,7 @@ func handleGetProfileIcon(ar *api.Request) (data []byte, err error) {
|
|||
ext := filepath.Ext(name)
|
||||
|
||||
// Get profile icon.
|
||||
data, err = GetProfileIcon(name)
|
||||
data, err = icons.GetProfileIcon(name)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
@ -152,7 +153,7 @@ func handleUpdateProfileIcon(ar *api.Request) (any, error) {
|
|||
}
|
||||
|
||||
// Update profile icon.
|
||||
filename, err := UpdateProfileIcon(ar.InputData, ext)
|
||||
filename, err := icons.UpdateProfileIcon(ar.InputData, ext)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
package profile
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"path"
|
||||
|
@ -146,7 +147,9 @@ func GetLocalProfile(id string, md MatchingData, createProfileCallback func() *P
|
|||
|
||||
// Trigger further metadata fetching from system if profile was created.
|
||||
if created && profile.UsePresentationPath && !special {
|
||||
module.StartWorker("get profile metadata", profile.updateMetadataFromSystem)
|
||||
module.StartWorker("get profile metadata", func(ctx context.Context) error {
|
||||
return profile.updateMetadataFromSystem(ctx, md)
|
||||
})
|
||||
}
|
||||
|
||||
// Prepare profile for first use.
|
||||
|
|
10
profile/icons/find_default.go
Normal file
10
profile/icons/find_default.go
Normal file
|
@ -0,0 +1,10 @@
|
|||
//go:build !linux
|
||||
|
||||
package icons
|
||||
|
||||
import "github.com/safing/portmaster/profile"
|
||||
|
||||
// FindIcon returns nil, nil for unsupported platforms.
|
||||
func FindIcon(binName string, homeDir string) (*profile.Icon, error) {
|
||||
return nil, nil
|
||||
}
|
112
profile/icons/find_linux.go
Normal file
112
profile/icons/find_linux.go
Normal file
|
@ -0,0 +1,112 @@
|
|||
package icons
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// FindIcon finds an icon for the given binary name.
|
||||
// Providing the home directory of the user running the process of that binary can help find an icon.
|
||||
func FindIcon(ctx context.Context, binName string, homeDir string) (*Icon, error) {
|
||||
// Search for icon.
|
||||
iconPath, err := search(binName, homeDir)
|
||||
if iconPath == "" {
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to find icon for %s: %w", binName, err)
|
||||
}
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// Load icon and save it.
|
||||
data, err := os.ReadFile(iconPath)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to read icon %s: %w", iconPath, err)
|
||||
}
|
||||
filename, err := UpdateProfileIcon(data, filepath.Ext(iconPath))
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to import icon %s: %w", iconPath, err)
|
||||
}
|
||||
return &Icon{
|
||||
Type: IconTypeAPI,
|
||||
Value: filename,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func search(binName string, homeDir string) (iconPath string, err error) {
|
||||
binName = strings.ToLower(binName)
|
||||
|
||||
// Search for icon path.
|
||||
for _, iconLoc := range iconLocations {
|
||||
basePath := iconLoc.GetPath(binName, homeDir)
|
||||
if basePath == "" {
|
||||
continue
|
||||
}
|
||||
|
||||
switch iconLoc.Type {
|
||||
case FlatDir:
|
||||
iconPath, err = searchDirectory(basePath, binName)
|
||||
case XDGIcons:
|
||||
iconPath, err = searchXDGIconStructure(basePath, binName)
|
||||
}
|
||||
|
||||
if iconPath != "" {
|
||||
return
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func searchXDGIconStructure(baseDirectory string, binName string) (iconPath string, err error) {
|
||||
for _, xdgIconDir := range xdgIconPaths {
|
||||
directory := filepath.Join(baseDirectory, xdgIconDir)
|
||||
iconPath, err = searchDirectory(directory, binName)
|
||||
if iconPath != "" {
|
||||
return
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func searchDirectory(directory string, binName string) (iconPath string, err error) {
|
||||
entries, err := os.ReadDir(directory)
|
||||
if err != nil {
|
||||
if errors.Is(err, os.ErrNotExist) {
|
||||
return "", nil
|
||||
}
|
||||
return "", fmt.Errorf("failed to read directory %s: %w", directory, err)
|
||||
}
|
||||
fmt.Println(directory)
|
||||
|
||||
var (
|
||||
bestMatch string
|
||||
bestMatchExcessChars int
|
||||
)
|
||||
for _, entry := range entries {
|
||||
// Skip dirs.
|
||||
if entry.IsDir() {
|
||||
continue
|
||||
}
|
||||
|
||||
iconName := strings.ToLower(entry.Name())
|
||||
iconName = strings.TrimSuffix(iconName, filepath.Ext(iconName))
|
||||
switch {
|
||||
case len(iconName) < len(binName):
|
||||
// Continue to next.
|
||||
case iconName == binName:
|
||||
// Exact match, return immediately.
|
||||
return filepath.Join(directory, entry.Name()), nil
|
||||
case strings.HasPrefix(iconName, binName):
|
||||
excessChars := len(iconName) - len(binName)
|
||||
if bestMatch == "" || excessChars < bestMatchExcessChars {
|
||||
bestMatch = entry.Name()
|
||||
bestMatchExcessChars = excessChars
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return bestMatch, nil
|
||||
}
|
32
profile/icons/find_linux_test.go
Normal file
32
profile/icons/find_linux_test.go
Normal file
|
@ -0,0 +1,32 @@
|
|||
package icons
|
||||
|
||||
import (
|
||||
"os"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestFindIcon(t *testing.T) {
|
||||
if testing.Short() {
|
||||
t.Skip("test depends on linux desktop environment")
|
||||
}
|
||||
t.Parallel()
|
||||
|
||||
home := os.Getenv("HOME")
|
||||
testFindIcon(t, "evolution", home)
|
||||
testFindIcon(t, "nextcloud", home)
|
||||
}
|
||||
|
||||
func testFindIcon(t *testing.T, binName string, homeDir string) {
|
||||
t.Helper()
|
||||
|
||||
iconPath, err := search(binName, homeDir)
|
||||
if err != nil {
|
||||
t.Error(err)
|
||||
return
|
||||
}
|
||||
if iconPath == "" {
|
||||
t.Errorf("no icon found for %s", binName)
|
||||
return
|
||||
}
|
||||
t.Logf("icon for %s found: %s", binName, iconPath)
|
||||
}
|
|
@ -1,4 +1,4 @@
|
|||
package profile
|
||||
package icons
|
||||
|
||||
import (
|
||||
"errors"
|
||||
|
@ -42,7 +42,8 @@ func (t IconType) sortOrder() int {
|
|||
}
|
||||
}
|
||||
|
||||
func sortAndCompactIcons(icons []Icon) []Icon {
|
||||
// SortAndCompact sorts and compacts a list of icons.
|
||||
func SortAndCompact(icons []Icon) []Icon {
|
||||
// Sort.
|
||||
slices.SortFunc[[]Icon, Icon](icons, func(a, b Icon) int {
|
||||
aOrder := a.Type.sortOrder()
|
|
@ -1,4 +1,4 @@
|
|||
package profile
|
||||
package icons
|
||||
|
||||
import (
|
||||
"crypto"
|
||||
|
@ -13,18 +13,21 @@ import (
|
|||
"github.com/safing/portbase/api"
|
||||
)
|
||||
|
||||
var profileIconStoragePath = ""
|
||||
// ProfileIconStoragePath defines the location where profile icons are stored.
|
||||
// Must be set before anything else from this package is called.
|
||||
// Must not be changed once set.
|
||||
var ProfileIconStoragePath = ""
|
||||
|
||||
// GetProfileIcon returns the profile icon with the given ID and extension.
|
||||
func GetProfileIcon(name string) (data []byte, err error) {
|
||||
// Check if enabled.
|
||||
if profileIconStoragePath == "" {
|
||||
if ProfileIconStoragePath == "" {
|
||||
return nil, errors.New("api icon storage not configured")
|
||||
}
|
||||
|
||||
// Build storage path.
|
||||
iconPath := filepath.Clean(
|
||||
filepath.Join(profileIconStoragePath, name),
|
||||
filepath.Join(ProfileIconStoragePath, name),
|
||||
)
|
||||
|
||||
iconPath, err = filepath.Abs(iconPath)
|
||||
|
@ -34,7 +37,7 @@ func GetProfileIcon(name string) (data []byte, err error) {
|
|||
|
||||
// Do a quick check if we are still within the right directory.
|
||||
// This check is not entirely correct, but is sufficient for this use case.
|
||||
if filepath.Dir(iconPath) != profileIconStoragePath {
|
||||
if filepath.Dir(iconPath) != ProfileIconStoragePath {
|
||||
return nil, api.ErrorWithStatus(errors.New("invalid icon"), http.StatusBadRequest)
|
||||
}
|
||||
|
||||
|
@ -72,7 +75,7 @@ func UpdateProfileIcon(data []byte, ext string) (filename string, err error) {
|
|||
|
||||
// Save to disk.
|
||||
filename = sum + "." + ext
|
||||
return filename, os.WriteFile(filepath.Join(profileIconStoragePath, filename), data, 0o0644) //nolint:gosec
|
||||
return filename, os.WriteFile(filepath.Join(ProfileIconStoragePath, filename), data, 0o0644) //nolint:gosec
|
||||
}
|
||||
|
||||
// TODO: Clean up icons regularly.
|
68
profile/icons/locations_linux.go
Normal file
68
profile/icons/locations_linux.go
Normal file
|
@ -0,0 +1,68 @@
|
|||
package icons
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
)
|
||||
|
||||
// IconLocation describes an icon location.
|
||||
type IconLocation struct {
|
||||
Directory string
|
||||
Type IconLocationType
|
||||
PathArg PathArg
|
||||
}
|
||||
|
||||
// IconLocationType describes an icon location type.
|
||||
type IconLocationType uint8
|
||||
|
||||
// Icon Location Types.
|
||||
const (
|
||||
FlatDir IconLocationType = iota
|
||||
XDGIcons
|
||||
)
|
||||
|
||||
// PathArg describes an icon location path argument.
|
||||
type PathArg uint8
|
||||
|
||||
// Path Args.
|
||||
const (
|
||||
NoPathArg PathArg = iota
|
||||
Home
|
||||
BinName
|
||||
)
|
||||
|
||||
var (
|
||||
iconLocations = []IconLocation{
|
||||
{Directory: "/usr/share/pixmaps", Type: FlatDir},
|
||||
{Directory: "/usr/share", Type: XDGIcons},
|
||||
{Directory: "%s/.local/share", Type: XDGIcons, PathArg: Home},
|
||||
{Directory: "%s/.local/share/flatpak/exports/share", Type: XDGIcons, PathArg: Home},
|
||||
{Directory: "/usr/share/%s", Type: XDGIcons, PathArg: BinName},
|
||||
}
|
||||
|
||||
xdgIconPaths = []string{
|
||||
// UI currently uses 48x48, so 256x256 should suffice for the future, even at 2x. (12.2023)
|
||||
"icons/hicolor/256x256/apps",
|
||||
"icons/hicolor/192x192/apps",
|
||||
"icons/hicolor/128x128/apps",
|
||||
"icons/hicolor/96x96/apps",
|
||||
"icons/hicolor/72x72/apps",
|
||||
"icons/hicolor/64x64/apps",
|
||||
"icons/hicolor/48x48/apps",
|
||||
"icons/hicolor/512x512/apps",
|
||||
}
|
||||
)
|
||||
|
||||
// GetPath returns the path of an icon.
|
||||
func (il IconLocation) GetPath(binName string, homeDir string) string {
|
||||
switch il.PathArg {
|
||||
case NoPathArg:
|
||||
return il.Directory
|
||||
case Home:
|
||||
if homeDir != "" {
|
||||
return fmt.Sprintf(il.Directory, homeDir)
|
||||
}
|
||||
case BinName:
|
||||
return fmt.Sprintf(il.Directory, binName)
|
||||
}
|
||||
return ""
|
||||
}
|
|
@ -7,6 +7,7 @@ import (
|
|||
"time"
|
||||
|
||||
"github.com/safing/portbase/database/record"
|
||||
"github.com/safing/portmaster/profile/icons"
|
||||
)
|
||||
|
||||
// MergeProfiles merges multiple profiles into a new one.
|
||||
|
@ -52,12 +53,12 @@ func MergeProfiles(name string, primary *Profile, secondaries ...*Profile) (newP
|
|||
}
|
||||
|
||||
// Collect all icons.
|
||||
newProfile.Icons = make([]Icon, 0, len(secondaries)+1) // Guess the needed space.
|
||||
newProfile.Icons = make([]icons.Icon, 0, len(secondaries)+1) // Guess the needed space.
|
||||
newProfile.Icons = append(newProfile.Icons, primary.Icons...)
|
||||
for _, sp := range secondaries {
|
||||
newProfile.Icons = append(newProfile.Icons, sp.Icons...)
|
||||
}
|
||||
newProfile.Icons = sortAndCompactIcons(newProfile.Icons)
|
||||
newProfile.Icons = icons.SortAndCompact(newProfile.Icons)
|
||||
|
||||
// Collect all fingerprints.
|
||||
newProfile.Fingerprints = make([]Fingerprint, 0, len(primary.Fingerprints)+len(secondaries)) // Guess the needed space.
|
||||
|
|
|
@ -11,6 +11,7 @@ import (
|
|||
"github.com/safing/portbase/log"
|
||||
"github.com/safing/portbase/modules"
|
||||
_ "github.com/safing/portmaster/core/base"
|
||||
"github.com/safing/portmaster/profile/icons"
|
||||
"github.com/safing/portmaster/updates"
|
||||
)
|
||||
|
||||
|
@ -52,7 +53,7 @@ func prep() error {
|
|||
if err := iconsDir.Ensure(); err != nil {
|
||||
return fmt.Errorf("failed to create/check icons directory: %w", err)
|
||||
}
|
||||
profileIconStoragePath = iconsDir.Path
|
||||
icons.ProfileIconStoragePath = iconsDir.Path
|
||||
|
||||
return nil
|
||||
}
|
||||
|
|
|
@ -18,6 +18,7 @@ import (
|
|||
"github.com/safing/portbase/utils/osdetail"
|
||||
"github.com/safing/portmaster/intel/filterlists"
|
||||
"github.com/safing/portmaster/profile/endpoints"
|
||||
"github.com/safing/portmaster/profile/icons"
|
||||
)
|
||||
|
||||
// ProfileSource is the source of the profile.
|
||||
|
@ -68,9 +69,9 @@ type Profile struct { //nolint:maligned // not worth the effort
|
|||
// See IconType for more information.
|
||||
Icon string
|
||||
// Deprecated: IconType describes the type of the Icon property.
|
||||
IconType IconType
|
||||
IconType icons.IconType
|
||||
// Icons holds a list of icons to represent the application.
|
||||
Icons []Icon
|
||||
Icons []icons.Icon
|
||||
|
||||
// Deprecated: LinkedPath used to point to the executableis this
|
||||
// profile was created for.
|
||||
|
@ -505,7 +506,7 @@ func (profile *Profile) updateMetadata(binaryPath string) (changed bool) {
|
|||
|
||||
// updateMetadataFromSystem updates the profile metadata with data from the
|
||||
// operating system and saves it afterwards.
|
||||
func (profile *Profile) updateMetadataFromSystem(ctx context.Context) error {
|
||||
func (profile *Profile) updateMetadataFromSystem(ctx context.Context, md MatchingData) error {
|
||||
var changed bool
|
||||
|
||||
// This function is only valid for local profiles.
|
||||
|
@ -531,6 +532,22 @@ func (profile *Profile) updateMetadataFromSystem(ctx context.Context) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
// Get icon if path matches presentation path.
|
||||
var newIcon *icons.Icon
|
||||
if profile.PresentationPath == md.Path() {
|
||||
// Get home from ENV.
|
||||
var home string
|
||||
if env := md.Env(); env != nil {
|
||||
home = env["HOME"]
|
||||
}
|
||||
var err error
|
||||
newIcon, err = icons.FindIcon(ctx, profile.PresentationPath, home)
|
||||
if err != nil {
|
||||
log.Warningf("profile: failed to find icon for %s: %s", profile.PresentationPath, err)
|
||||
newIcon = nil
|
||||
}
|
||||
}
|
||||
|
||||
// Apply new data to profile.
|
||||
func() {
|
||||
// Lock profile for applying metadata.
|
||||
|
@ -542,6 +559,16 @@ func (profile *Profile) updateMetadataFromSystem(ctx context.Context) error {
|
|||
profile.Name = newName
|
||||
changed = true
|
||||
}
|
||||
|
||||
// Apply new icon if found.
|
||||
if newIcon != nil {
|
||||
if len(profile.Icons) == 0 {
|
||||
profile.Icons = []icons.Icon{*newIcon}
|
||||
} else {
|
||||
profile.Icons = append(profile.Icons, *newIcon)
|
||||
profile.Icons = icons.SortAndCompact(profile.Icons)
|
||||
}
|
||||
}
|
||||
}()
|
||||
|
||||
// If anything changed, save the profile.
|
||||
|
|
Loading…
Add table
Reference in a new issue