Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix favicon "not found" error on non-homepage pages #617

Merged
merged 6 commits into from
Jan 20, 2025

Conversation

vidipsingh
Copy link
Contributor

This PR addresses the issue where favicons were not found on non-homepage pages, particularly on Firefox browser. The problem was traced to an incorrect favicon path that was not being handled properly across different pages.

Changes Made:

  • Updated the favicon link in _layouts/base.html to use a consistent, absolute path.
  • Modified js/main.js to dynamically update the favicon based on a random color scheme.
  • Implemented error handling in the JavaScript to revert to a default favicon if the specific one is not found.

Issue Link:

Closes #507

How to Test:

  1. Load the homepage and any non-homepage page (e.g., /about-us or /press) on Firefox.
  2. Check if the favicon is displayed correctly across all pages.
  3. Ensure that a fallback favicon is shown if the dynamic one is missing.

Additional Notes:

  • This issue was specifically observed in the Firefox browser.
  • The fix ensures consistent favicon display on all pages, improving the site's visual consistency and user experience.

OS and Browser Information:

  • Operating System: Linux
  • Browser: Firefox

@quozl
Copy link
Contributor

quozl commented Dec 29, 2024

You have a commit to add author name display to blog, yet this pull request is for the favicon problem. Please remove the commit 9c1b988.

Also, you have placed in the pull request opening comment text that should be in the commit message of 53cb483. Please do read our guide to making commits carefully.

- Updated `_layouts/base.html` to use a consistent favicon path.
- Modified `js/main.js` to handle dynamic favicon changes with error handling.
- Added fallback to the default favicon if a specific favicon is not found.
@vidipsingh vidipsingh force-pushed the fix/favicon-not-found-507 branch from 53cb483 to 58177ea Compare December 30, 2024 05:11
@vidipsingh
Copy link
Contributor Author

@quozl, I have removed the commit 9c1b988 and will make sure to follow the guide for future contributions.
Thank you for the feedback!

Copy link
Contributor

@quozl quozl left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not convinced this is the right approach. Details are in the feedback below.

  • rewrite commit messages as suggested earlier.

@@ -4,7 +4,7 @@
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<link id="defaultIcon1" rel="shortcut icon" href="{{ site.baseurl }}assets/favicon_06.png" />
<link id="defaultIcon1" rel="shortcut icon" href="/assets/favicon.png" />
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is site.baseurl removed?

Why is there no file assets/favicon.png yet you link to it here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will revert the changes in _layouts/base.html to the previous version.
Since there is no /favicon.png, should I refer to /assets/favicon_06.png here instead as before?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't mind either way, just that it should refer to something that exists.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, then I will change it to {{ site.baseurl }}/assets/favicon_06.png.

js/main.js Outdated
};
img.onerror = function() {
// Fallback to default favicon
defaultIcon.href = '/assets/favicon.png';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no file assets/favicon.png

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since there is no /assets/favicon.png, should I refer to /assets/favicon_06.png here instead?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As above.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok sure.

js/main.js Outdated
// Use absolute path for favicon
var faviconPath = '/assets/favicon_' + logoID + '.png';

// Test if favicon exists before setting
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't see why we need to test if it exists. We know exactly how many files we have, what their names are, and we know the code above this section in js/main.js.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, I'll remove the favicon test since we already know the number and names of the images.

@@ -16,7 +16,24 @@ $(document).ready(function () {
if (logoID < 10) {
logoID = "0" + logoID;
}
document.querySelector('#defaultIcon1').href = 'https://www.sugarlabs.org/assets/favicon_' + logoID + '.png';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code you are removing has nothing to do with the issue; the issue and my tests of master branch just now show the error to be a reference to press/assets/favicon_06.png and this line you are removing is an absolute link without press.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have simplified the favicon logic by making the path relative to /press/, removing the full domain URL, and eliminating unnecessary existence checks, as shown in the following code:

var defaultIcon = document.querySelector('#defaultIcon1');
if (defaultIcon) {
    var logoID = colorIndex + 1;
    if (logoID < 10) {
        logoID = "0" + logoID;
    }
    defaultIcon.href = '/press/assets/favicon_' + logoID + '.png';
}

Also, I have removed the testing and existence checks. Is this approach fine?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You've lost me here. Why would the path be made absolute from /press/?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, I think relative path will be better for this like:

var defaultIcon = document.querySelector('#defaultIcon1');
if (defaultIcon) {
    var logoID = colorIndex + 1;
    if (logoID < 10) {
        logoID = "0" + logoID;
    }
    defaultIcon.href = 'assets/favicon_' + logoID + '.png';
}

Now it's pointing to the assets/ directory relative to the current location, which should be more flexible.

@vidipsingh
Copy link
Contributor Author

I'm not convinced this is the right approach. Details are in the feedback below.

  • rewrite commit messages as suggested earlier.

Hi @quozl,

Thanks for the feedback. I’ll work on the changes as suggested.
Could you please point out what specifically is wrong with my current commit message? It would help me improve from next time.

@quozl
Copy link
Contributor

quozl commented Dec 31, 2024

Sure. When I went to your commit message and read it, assuming the persona of a future developer trying to figure out why git bisect identified the commit as the cause of a problem being investigated, I could not find the information we expect;

  • explain the problem that is solved; you've put this in the pull request opening comment instead of the commit message,
  • when the problem was introduced by a previous commit, mention the hash; this shows your full understanding and helps with review, letting us see how the problem began and evaluate if your solution is the right one,
  • when the problem is in an issue or ticket, as it does, end with "Fixes #507", so that GitHub will automatically close the issue.

It is alarming when a pull request opening comment has such a huge amount of detail that is left out of the commit message.

Remember that the git commit message is kept in perpetuity, and pull requests are deleted eventually. The git repository clone on a developer's system is where everything necessary to development must be present. GitHub is only a public hosting point for the repository.

@vidipsingh
Copy link
Contributor Author

Sure. When I went to your commit message and read it, assuming the persona of a future developer trying to figure out why git bisect identified the commit as the cause of a problem being investigated, I could not find the information we expect;

  • explain the problem that is solved; you've put this in the pull request opening comment instead of the commit message,
  • when the problem was introduced by a previous commit, mention the hash; this shows your full understanding and helps with review, letting us see how the problem began and evaluate if your solution is the right one,
  • when the problem is in an issue or ticket, as it does, end with "Fixes #507", so that GitHub will automatically close the issue.

It is alarming when a pull request opening comment has such a huge amount of detail that is left out of the commit message.

Remember that the git commit message is kept in perpetuity, and pull requests are deleted eventually. The git repository clone on a developer's system is where everything necessary to development must be present. GitHub is only a public hosting point for the repository.

Thanks for the feedback, @quozl! I’ll make sure to incorporate these practices in future commits and ensure all relevant details are included directly in the commit message.

@quozl
Copy link
Contributor

quozl commented Jan 2, 2025

Thanks for the feedback, @quozl! I’ll make sure to incorporate these practices in future commits and ensure all relevant details are included directly in the commit message.

So you're saying you won't fix it for this pull request? Why not? Our purpose for review includes managing the history.

You have just one commit. All you have to do is git commit --amend followed by git push --force. If you had more than one commit, you would git rebase -i, select reword, then git push --force.

See our guide for reviewers to understand our review practices, and please learn to review your own work and others. We need more reviewers.

@vidipsingh
Copy link
Contributor Author

Thanks for the feedback, @quozl! I’ll make sure to incorporate these practices in future commits and ensure all relevant details are included directly in the commit message.

So you're saying you won't fix it for this pull request? Why not? Our purpose for review includes managing the history.

You have just one commit. All you have to do is git commit --amend followed by git push --force. If you had more than one commit, you would git rebase -i, select reword, then git push --force.

See our guide for reviewers to understand our review practices, and please learn to review your own work and others. We need more reviewers.

Sorry about that, @quozl! I’ll make the necessary changes to the recent commit and ensure to follow these practices for all future commits as well.

@quozl quozl requested a review from pikurasa January 3, 2025 20:12
@quozl
Copy link
Contributor

quozl commented Jan 3, 2025

@pikurasa ready for your review; this pull request has finished technical review and is on the way toward commit message goodness, just need to know you are happy with the solution to the issue.

js/main.js Outdated
if (logoID < 10) {
logoID = "0" + logoID;
}
defaultIcon.href = 'assets/favicon_' + logoID + '.png';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this still taking a relative path?

I ask because I still see things like this in the logs:

Screenshot from 2025-01-04 16-25-26

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, is use of relative path. Oops.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this still taking a relative path?

I ask because I still see things like this in the logs:

Screenshot from 2025-01-04 16-25-26

Yes, it is now taking relative paths as intended. Previously, the code was referencing the favicon like this:

document.querySelector('#defaultIcon1').href = 'https://www.sugarlabs.org/assets/favicon_' + logoID + '.png';

However, I have updated it to use relative paths, as shown here:

defaultIcon.href = 'assets/favicon_' + logoID + '.png';

After this change, I am no longer seeing any errors in my logs, whether on Firefox or other browsers.

image

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, relative or absolute, doesn't matter, as long as the directory that contains the icons is accessed on both the root and lower pages. It works for the root page, but as @pikurasa said it doesn't work on lower pages, such as about-us/. That's what you need to fix now.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(a relative access from about-us/ would be to ../assets/)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, relative or absolute, doesn't matter, as long as the directory that contains the icons is accessed on both the root and lower pages. It works for the root page, but as @pikurasa said it doesn't work on lower pages, such as about-us/. That's what you need to fix now.

In the _layouts/base.html, the previous line was:

<link id="defaultIcon1" rel="shortcut icon" href="{{ site.baseurl }}assets/favicon_06.png" /> 

I changed it to:

<link id="defaultIcon1" rel="shortcut icon" href="{{ site.baseurl }}/assets/favicon_06.png" /> 

The issue was that there was no forward slash between {{ site.baseurl }} and assets, which caused incorrect path resolution when navigating between pages like /about-us/. Now, the problem of the favicon "not found" error on non-homepage pages is not there in Firefox.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I know, but I was not talking about that part of the change. I was talking about the part of the change that this discussion thread is linked to; js/main.js line 26, which loads a relative path into the element href, without dereferencing the current document's relative path, which is why the browser then reports failures to find the png files.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I know, but I was not talking about that part of the change. I was talking about the part of the change that this discussion thread is linked to; js/main.js line 26, which loads a relative path into the element href, without dereferencing the current document's relative path, which is why the browser then reports failures to find the png files.

@quozl The change that I made for this is as follows:

var defaultIcon = document.querySelector('#defaultIcon1');
if (defaultIcon) {
    var logoID = colorIndex + 1;
    if (logoID < 10) {
        logoID = "0" + logoID;
    }

    // Extract the base path from the current favicon href
    var currentHref = defaultIcon.getAttribute('href');
    var basePath = currentHref.substring(0, currentHref.lastIndexOf('/') + 1);
    
    // Update only the filename portion while keeping the original path
    defaultIcon.href = basePath + 'favicon_' + logoID + '.png';
}

And in _layouts/base.html:

<link id="defaultIcon1" rel="shortcut icon" href="{{ site.baseurl }}/assets/favicon_06.png" />

This approach resolves the issue of subdirectories by dynamically extracting the base path from the current favicon and then updating only the filename. By preserving the base path that starts from {{ site.baseurl }}, this ensures that the favicon works across all pages, regardless of how deep they are within the site's directory structure.

Would this be a good approach to resolve the issue? I’d also appreciate any suggestions you might have to improve it.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hope so. Commit the change and push to the pull request branch so that I can see it in context and we can test.

@quozl
Copy link
Contributor

quozl commented Jan 4, 2025

Noted edited commit message on 1939ac9

Review comments;

  • the first line is too long, see how GitHub renders it by breaking in the middle of a word, which is them telling you to fix it,
  • remove the # before the commit hash so it is properly rendered by GitHub, like this 58177ea,
  • put the "fixes" and issue number at the very end.

- Fixed "favicon not found" error on non-homepage pages, particularly in Firefox.
- Problem traced to incorrect favicon path introduced in commit 58177ea.
- Updated path from absolute to relative to resolve the issue.
- Ensures favicons load correctly across all pages.

Fixes sugarlabs#507
@vidipsingh vidipsingh force-pushed the fix/favicon-not-found-507 branch from 1939ac9 to 15f9065 Compare January 5, 2025 11:22
- Fixed issue with favicon path on subdirectory pages by dynamically extracting and updating favicon filename.
- Problem traced to previous commit 15f9065 which introduced static favicon path.
- Ensured favicon loads correctly across all pages in firefox.

Fixes sugarlabs#507
@quozl
Copy link
Contributor

quozl commented Jan 10, 2025

Tested 245ea8c with Firefox in a container using --host.

  • does fix Favicon gets "not found" error on non-homepage pages #507,
  • causes double GET of favicon;
    • often the initial GET is aborted asynchronously; is there a way to avoid that, such as by making the href change inside the page rather than in external script?
    • otherwise the initial GET completes and then the js/main.js triggers another GET; is there a way to avoid two GETs?
  • removing defer makes no difference,
  • your branch diverged from master a few weeks ago, so all the things we've since fixed come back, please keep your pull request branch up to date.

@vidipsingh
Copy link
Contributor Author

Tested 245ea8c with Firefox in a container using --host.

  • does fix Favicon gets "not found" error on non-homepage pages #507,

  • causes double GET of favicon;

    • often the initial GET is aborted asynchronously; is there a way to avoid that, such as by making the href change inside the page rather than in external script?
    • otherwise the initial GET completes and then the js/main.js triggers another GET; is there a way to avoid two GETs?
  • removing defer makes no difference,

  • your branch diverged from master a few weeks ago, so all the things we've since fixed come back, please keep your pull request branch up to date.

Hi @quozl,
Thanks for your feedback.

  • I'm also running the setup with Firefox in a container using --host, but I haven’t been able to observe the double GET request for the favicon in the console or elsewhere. Could you kindly elaborate on how and where exactly the two GET requests for the favicon are being triggered? It would be very helpful if you can provide more details or steps to reproduce the issue.
  • As for the defer attribute, since removing it doesn't seem to make any difference in the site's functionality or error logging, should I go ahead and remove it?
  • I'll make sure to update my branch with the latest changes from master.

@quozl
Copy link
Contributor

quozl commented Jan 11, 2025

I'm also running the setup with Firefox in a container using --host, but I haven’t been able to observe the double GET request for the favicon in the console or elsewhere.

jekyll does not appear to have an option to provide a server side request log. It is possible to use tcpdump or wireshark but takes longer to analyse.

Could you kindly elaborate on how and where exactly the two GET requests for the favicon are being triggered? It would be very helpful if you can provide more details or steps to reproduce the issue.

Sure. Method used;

  • Start Jekyll,
  • Start Firefox and open the site using the URL provided by Jekyll,
  • Tools → Browser Tools → Web Developer Tools,
  • Network tab,
  • Reload,
  • Filter "favicon".

Observed one of three scenarios, which is unsurprising given this is a race condition between the browser's built-in loading of favicon and our script in js/main.js;

Scenario 0 - browser begins GET but cancels due to script

fi0

Scenario 1 - browser GET followed by script GET

fi1

Scenario 2 - browser GET but the script chooses the same random icon

fi2

As for the defer attribute, since removing it doesn't seem to make any difference in the site's functionality or error logging, should I go ahead and remove it?

Why would you do that?

@vidipsingh
Copy link
Contributor Author

I'm also running the setup with Firefox in a container using --host, but I haven’t been able to observe the double GET request for the favicon in the console or elsewhere.

jekyll does not appear to have an option to provide a server side request log. It is possible to use tcpdump or wireshark but takes longer to analyse.

Could you kindly elaborate on how and where exactly the two GET requests for the favicon are being triggered? It would be very helpful if you can provide more details or steps to reproduce the issue.

Sure. Method used;

  • Start Jekyll,
  • Start Firefox and open the site using the URL provided by Jekyll,
  • Tools → Browser Tools → Web Developer Tools,
  • Network tab,
  • Reload,
  • Filter "favicon".

Observed one of three scenarios, which is unsurprising given this is a race condition between the browser's built-in loading of favicon and our script in js/main.js;

Scenario 0 - browser begins GET but cancels due to script

fi0

Scenario 1 - browser GET followed by script GET

fi1

Scenario 2 - browser GET but the script chooses the same random icon

fi2

As for the defer attribute, since removing it doesn't seem to make any difference in the site's functionality or error logging, should I go ahead and remove it?

Why would you do that?

I think the problem is:

  1. When the browser loads the page, it first detects the default favicon in the HTML:
<link id="defaultIcon1" rel="shortcut icon" href="{{ site.baseurl }}/assets/favicon_06.png" />
  1. The JavaScript in main.js then attempts to randomly change the favicon:
var defaultIcon = document.querySelector('#defaultIcon1');
if (defaultIcon) {
    var logoID = colorIndex + 1;
    if (logoID < 10) {
        logoID = "0" + logoID;
    }
    // Update favicon path
    defaultIcon.href = basePath + 'favicon_' + logoID + '.png';
}

This creates a race condition which is given in the above three scenarios.

One solution I can think of is that we can:

  • Moving favicon generation to before the DOM content loads
  • Removing the static favicon from HTML
  • Dynamically creating and injecting the favicon element using JavaScript
  • Using a meta tag to store the base path

However, the issue only appears during the first request, and subsequent GET requests for the favicon work fine in Firefox for me.

As for the deferattribute I think it should be there only as it is helpful because it loads the script in the background without blocking the parsing of the HTML, and only executes it when the DOM has been completely constructed.

@quozl
Copy link
Contributor

quozl commented Jan 13, 2025

Thanks.

Removing the static favicon would probably cause the browser to attempt favicon.ico as the default. Let me know what you find.

My guess is that changing the href in a page script may happen sooner than js/main.js

Yes, the issue is only there on first request of a page, but our target environment has been users with bad internet. I'd like to minimse off-site and repeated GETs.

We have Google Analytics loading as well, but if nobody is using the results we don't need to do that.

@vidipsingh
Copy link
Contributor Author

Hi @quozl,

Referring to the favicon issue fix, it’s now working perfectly in Firefox! The favicon loads correctly on sub-URLs, ensuring the icon is displayed consistently across different page depths.

Here’s the full updated code in main.js:

var defaultIcon = document.querySelector('#defaultIcon1');
    if (defaultIcon) {
        var logoID = colorIndex + 1;
        if (logoID < 10) {
            logoID = "0" + logoID;
        }
        
        // Get the site's root URL by looking at the <base> tag or constructing from window.location
        var rootUrl = '';
        var baseTag = document.querySelector('base');
        if (baseTag && baseTag.href) {
            rootUrl = baseTag.href;
        } else {
            // Construct the root URL from window.location
            rootUrl = window.location.protocol + '//' + window.location.host + '/';
        }
        
        // Use absolute path for favicon
        defaultIcon.href = rootUrl + 'assets/favicon_' + logoID + '.png';
    }

Please share some feedback or a review on the fix.

@quozl
Copy link
Contributor

quozl commented Jan 16, 2025

As it is only a change to js/main.js, I can't see how it would change the browser default loading of what is specified in base.html

Perhaps you might push your changes to the branch so I can test them?

- Updated favicon handling to dynamically set the favicon path based on the root URL.
- Fixed the issue in the commit  245ea8c where favicon would not load correctly on subdirectory pages by calculating the site root URL using window location.

Fixes sugarlabs#507
@pikurasa
Copy link
Contributor

I tested vidipsingh@7c5c491, and it seems to work. (Previous commits did not solve the issue for me.)

@vidipsingh @quozl is there anything else that we're waiting on here?

@quozl
Copy link
Contributor

quozl commented Jan 20, 2025

Thanks for testing.

@quozl quozl merged commit b71fa69 into sugarlabs:master Jan 20, 2025
@vidipsingh
Copy link
Contributor Author

Thank you @pikurasa and @quozl for your feedback, and for merging the PR!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Favicon gets "not found" error on non-homepage pages
3 participants