mirror of
https://github.com/miniflux/v2.git
synced 2025-08-26 18:21:01 +00:00
445 lines
15 KiB
Go
445 lines
15 KiB
Go
// SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package icon // import "miniflux.app/v2/internal/reader/icon"
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/base64"
|
|
"image"
|
|
"strings"
|
|
"testing"
|
|
|
|
"miniflux.app/v2/internal/model"
|
|
)
|
|
|
|
func TestParseImageDataURL(t *testing.T) {
|
|
iconURL := ""
|
|
icon, err := parseImageDataURL(iconURL)
|
|
if err != nil {
|
|
t.Fatalf(`We should be able to parse valid data URL: %v`, err)
|
|
}
|
|
|
|
if icon.MimeType != "image/webp" {
|
|
t.Fatal(`Invalid mime type parsed`)
|
|
}
|
|
|
|
if icon.Hash == "" {
|
|
t.Fatal(`Image hash should be computed`)
|
|
}
|
|
}
|
|
|
|
func TestParseImageDataURLWithNoEncoding(t *testing.T) {
|
|
iconURL := `data:image/webp,%3Ch1%3EHello%2C%20World%21%3C%2Fh1%3E`
|
|
icon, err := parseImageDataURL(iconURL)
|
|
if err != nil {
|
|
t.Fatalf(`We should be able to parse valid data URL: %v`, err)
|
|
}
|
|
|
|
if icon.MimeType != "image/webp" {
|
|
t.Fatal(`Invalid mime type parsed`)
|
|
}
|
|
|
|
if string(icon.Content) == "Hello, World!" {
|
|
t.Fatal(`Value should be URL-decoded`)
|
|
}
|
|
|
|
if icon.Hash == "" {
|
|
t.Fatal(`Image hash should be computed`)
|
|
}
|
|
}
|
|
|
|
func TestParseImageWithRawSVGEncodedInUTF8(t *testing.T) {
|
|
iconURL := `data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 456 456'><circle></circle></svg>`
|
|
icon, err := parseImageDataURL(iconURL)
|
|
if err != nil {
|
|
t.Fatalf(`We should be able to parse valid data URL: %v`, err)
|
|
}
|
|
|
|
if icon.MimeType != "image/svg+xml" {
|
|
t.Fatal(`Invalid mime type parsed`)
|
|
}
|
|
|
|
if icon.Hash == "" {
|
|
t.Fatal(`Image hash should be computed`)
|
|
}
|
|
|
|
if string(icon.Content) != `<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 456 456'><circle></circle></svg>` {
|
|
t.Fatal(`Invalid SVG content`)
|
|
}
|
|
}
|
|
|
|
func TestParseImageDataURLWithNoMediaTypeAndNoEncoding(t *testing.T) {
|
|
iconURL := `data:,Hello%2C%20World%21`
|
|
_, err := parseImageDataURL(iconURL)
|
|
if err == nil {
|
|
t.Fatal(`We should detect invalid mime type`)
|
|
}
|
|
}
|
|
|
|
func TestParseInvalidImageDataURLWithBadMimeType(t *testing.T) {
|
|
_, err := parseImageDataURL("data:text/plain;base64,blob")
|
|
if err == nil {
|
|
t.Fatal(`We should detect invalid mime type`)
|
|
}
|
|
}
|
|
|
|
func TestParseInvalidImageDataURLWithUnsupportedEncoding(t *testing.T) {
|
|
_, err := parseImageDataURL("data:image/png;base32,blob")
|
|
if err == nil {
|
|
t.Fatal(`We should detect unsupported encoding`)
|
|
}
|
|
}
|
|
|
|
func TestParseInvalidImageDataURLWithNoData(t *testing.T) {
|
|
_, err := parseImageDataURL("data:image/png;base64,")
|
|
if err == nil {
|
|
t.Fatal(`We should detect invalid encoded data`)
|
|
}
|
|
}
|
|
|
|
func TestParseInvalidImageDataURL(t *testing.T) {
|
|
_, err := parseImageDataURL("">
|
|
<link rel="shortcut icon" href="",
|
|
"/regular-icon.ico",
|
|
"data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg'></svg>",
|
|
}
|
|
|
|
if len(iconURLs) != len(expected) {
|
|
t.Fatalf("Expected %d icon URLs, got %d", len(expected), len(iconURLs))
|
|
}
|
|
|
|
for i, expectedURL := range expected {
|
|
if iconURLs[i] != expectedURL {
|
|
t.Errorf("Expected icon URL %d to be %q, got %q", i, expectedURL, iconURLs[i])
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestFindIconURLsFromHTMLDocument_RelativeAndAbsoluteURLs(t *testing.T) {
|
|
html := `<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<link rel="icon" href="/absolute-path.ico">
|
|
<link rel="icon" href="relative-path.ico">
|
|
<link rel="icon" href="../parent-dir.ico">
|
|
<link rel="icon" href="https://example.com/external.ico">
|
|
<link rel="icon" href="//cdn.example.com/protocol-relative.ico">
|
|
</head>
|
|
</html>`
|
|
|
|
iconURLs, err := findIconURLsFromHTMLDocument(strings.NewReader(html), "text/html")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
expected := []string{
|
|
"/absolute-path.ico",
|
|
"relative-path.ico",
|
|
"../parent-dir.ico",
|
|
"https://example.com/external.ico",
|
|
"//cdn.example.com/protocol-relative.ico",
|
|
}
|
|
|
|
if len(iconURLs) != len(expected) {
|
|
t.Fatalf("Expected %d icon URLs, got %d", len(expected), len(iconURLs))
|
|
}
|
|
|
|
for i, expectedURL := range expected {
|
|
if iconURLs[i] != expectedURL {
|
|
t.Errorf("Expected icon URL %d to be %q, got %q", i, expectedURL, iconURLs[i])
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestFindIconURLsFromHTMLDocument_InvalidHTML(t *testing.T) {
|
|
html := `<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<link rel="icon" href="/valid-before-error.ico">
|
|
<link rel="icon" href="/unclosed-tag.ico"
|
|
<link rel="shortcut icon" href="/valid-after-error.ico">
|
|
</head>
|
|
</html>`
|
|
|
|
iconURLs, err := findIconURLsFromHTMLDocument(strings.NewReader(html), "text/html")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// goquery should handle malformed HTML gracefully
|
|
if len(iconURLs) == 0 {
|
|
t.Fatal("Expected to find some icon URLs even with malformed HTML")
|
|
}
|
|
|
|
// Should at least find the valid ones
|
|
foundValidIcon := false
|
|
for _, url := range iconURLs {
|
|
if url == "/valid-before-error.ico" || url == "/valid-after-error.ico" {
|
|
foundValidIcon = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !foundValidIcon {
|
|
t.Errorf("Expected to find at least one valid icon URL, got: %v", iconURLs)
|
|
}
|
|
}
|
|
|
|
func TestFindIconURLsFromHTMLDocument_EmptyDocument(t *testing.T) {
|
|
iconURLs, err := findIconURLsFromHTMLDocument(strings.NewReader(""), "text/html")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if len(iconURLs) != 0 {
|
|
t.Fatalf("Expected 0 icon URLs from empty document, got %d", len(iconURLs))
|
|
}
|
|
}
|
|
|
|
func TestResizeIconSmallGif(t *testing.T) {
|
|
data, err := base64.StdEncoding.DecodeString("R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
icon := model.Icon{
|
|
Content: data,
|
|
MimeType: "image/gif",
|
|
}
|
|
if !bytes.Equal(icon.Content, resizeIcon(&icon).Content) {
|
|
t.Fatalf("Converted gif smaller than 16x16")
|
|
}
|
|
}
|
|
|
|
func TestResizeIconPng(t *testing.T) {
|
|
data, err := base64.StdEncoding.DecodeString("iVBORw0KGgoAAAANSUhEUgAAACEAAAAhCAYAAABX5MJvAAAALUlEQVR42u3OMQEAAAgDoJnc6BpjDyRgcrcpGwkJCQkJCQkJCQkJCQkJCYmyB7NfUj/Kk4FkAAAAAElFTkSuQmCC")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
icon := model.Icon{
|
|
Content: data,
|
|
MimeType: "image/png",
|
|
}
|
|
resizedIcon := resizeIcon(&icon)
|
|
|
|
if bytes.Equal(data, resizedIcon.Content) {
|
|
t.Fatalf("Didn't convert png of 33x33")
|
|
}
|
|
|
|
config, _, err := image.DecodeConfig(bytes.NewReader(resizedIcon.Content))
|
|
if err != nil {
|
|
t.Fatalf("Couln't decode resulting png: %v", err)
|
|
}
|
|
|
|
if config.Height != 32 || config.Width != 32 {
|
|
t.Fatalf("Was expecting an image of 16x16, got %dx%d", config.Width, config.Height)
|
|
}
|
|
}
|
|
|
|
func TestResizeIconWebp(t *testing.T) {
|
|
data, err := base64.StdEncoding.DecodeString("UklGRkAAAABXRUJQVlA4IDQAAADwAQCdASoBAAEAAQAcJaACdLoB+AAETAAA/vW4f/6aR40jxpHxcP/ugT90CfugT/3NoAAA")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
icon := model.Icon{
|
|
Content: data,
|
|
MimeType: "image/webp",
|
|
}
|
|
|
|
if !bytes.Equal(icon.Content, resizeIcon(&icon).Content) {
|
|
t.Fatalf("Converted webp smaller than 16x16")
|
|
}
|
|
}
|
|
|
|
func TestResizeInvalidImage(t *testing.T) {
|
|
icon := model.Icon{
|
|
Content: []byte("invalid data"),
|
|
MimeType: "image/gif",
|
|
}
|
|
if !bytes.Equal(icon.Content, resizeIcon(&icon).Content) {
|
|
t.Fatalf("Tried to convert an invalid image")
|
|
}
|
|
}
|
|
|
|
func TestMinifySvg(t *testing.T) {
|
|
data := []byte(`<svg path d=" M1 4h-.001 V1h2v.001 M1 2.6 h1v.001"/></svg>`)
|
|
want := []byte(`<svg path="" d="M1 4H.999V1h2v.001M1 2.6h1v.001"/></svg>`)
|
|
icon := model.Icon{Content: data, MimeType: "image/svg+xml"}
|
|
got := resizeIcon(&icon).Content
|
|
if !bytes.Equal(want, got) {
|
|
t.Fatalf("Didn't correctly minify the svg: got %s instead of %s", got, want)
|
|
}
|
|
}
|
|
|
|
func TestMinifySvgWithError(t *testing.T) {
|
|
// Invalid SVG with malformed XML that should cause minification to fail
|
|
data := []byte(`<svg><><invalid-tag<>unclosed`)
|
|
original := make([]byte, len(data))
|
|
copy(original, data)
|
|
|
|
icon := model.Icon{
|
|
Content: data,
|
|
MimeType: "image/svg+xml",
|
|
}
|
|
|
|
result := resizeIcon(&icon)
|
|
|
|
// When minification fails, the original content should be preserved
|
|
if !bytes.Equal(original, result.Content) {
|
|
t.Fatalf("Expected original content to be preserved on minification error, got %s instead of %s", result.Content, original)
|
|
}
|
|
|
|
// MimeType should remain unchanged
|
|
if result.MimeType != "image/svg+xml" {
|
|
t.Fatalf("Expected MimeType to remain image/svg+xml, got %s", result.MimeType)
|
|
}
|
|
}
|