IsmaelMartinez / teams-for-linux

Unofficial Microsoft Teams for Linux client
GNU General Public License v3.0
2.88k stars 235 forks source link

Custom background images? #591

Closed nitro322 closed 1 year ago

nitro322 commented 2 years ago

Hello. I'm using your teams-for-linux client again after switching to the "official" client for a couple of years. With the recent news about Microsoft killing off the Linux client, was delighted to see that this project was still going strong, and even more so to learn that you're supporting background blur and images and various other features Microsoft apparently couldn't be bothered to support themselves on Linux. Very much appreciate what you've been doing here.

I'm trying to add a custom background image. My understanding from some basic searching is that there's supposed to be an Add Image option in the background image selector, but I don't seem to have that. Honestly not sure if this is a bug, known limitation, or if I'm just missing something since I've never done this before.

Can you please provide any guidance on whether this is supported and if so, how to use this feature? Apologies for the basic question here, but couldn't find a clear answer and not sure where else to ask. Thanks.

jijojosephk commented 2 years ago

Are you referring to background during video calls? If so, it's already working I believe.

nitro322 commented 2 years ago

Adding a custom background image during video calls, yes. If that's working for you, mind sharing how to do that? I'm expecting to see some kind of Add Image link or button in the "Show background effects" pane. I have the ability to select blur and various canned images, and get the Preview and Apply options, but nothing related to adding images.

jijojosephk commented 2 years ago

Ok, i understand now. I was referring to predefined backgrounds in the teams app. Never tried uploading image from local machine. Implementing this feature may be difficult. If it's present in web version, it could be possible here as well. I'd have it checked. Thanks.

jijojosephk commented 2 years ago

@nitro322

Can you check in chrome?

https://support.microsoft.com/en-us/office/change-your-background-for-a-teams-meeting-f77a2381-443a-499d-825e-509a140f4780

nitro322 commented 2 years ago

I tested with Vivaldi (not Chrome, but uses the same engine and I've never observed any compatibility issues between it and Chrome) and get the same behavior as the electron app. You bring up a good point about the web version - I was thinking of Teams as a desktop application, but if you're essentially re-packaging the web version then I could see where that feature wouldn't be supported. Checking the link you sent, for example, shows instructions for desktop and mobile clients, but not web.

So, looks like this may not be supported after all. Bummer, but not a big deal. Appreciate the dialog. Feel free to close this if you agree this isn't supported with the web app.

jijojosephk commented 2 years ago

It's sort of hidden. You can use the teams-for-linux app itself.

Screenshot from 2022-09-16 22-27-06 Screenshot from 2022-09-16 22-27-58

nitro322 commented 2 years ago

Right, I can find that part. I'm talking about custom images. Ie., this (from that same microsoft link you posted above):

You can also replace your background with one of the images provided, or with one of your own choosing. To use an image of your own, select Add new and then select one to upload from your computer. Make sure it's a .JPG, .PNG, or .BMP file.

That "Add new" option doesn't exist for me.

jijojosephk commented 2 years ago

Oh, I see. Have you had used the feature on the destop for linux or have you used windows or mac version?

nitro322 commented 2 years ago

It didn't exist in the Linux client. No background effects options existed. No, I've never used the Windows or Mac version, hence my general uncertainty in the OP as I've never actually used this feature myself before. But many of my work colleagues have used it, so I know it works under the right conditions. Seems like those conditions exclude the web client, unfortunately.

jijojosephk commented 2 years ago

Right, I see the images are dynamically loaded from a CDN. We might be able to do some hacks to achieve.

For example: https://static.cdn.microsoft.com/image1 request can be locally served using a different image. By faking the response and make the logic believe that it actually came from a web server. We can experiment.

malanvaneck commented 2 years ago

Hi there @jijojosephk, I really hope there is something we could do. Is there could you possibly point me to what specific URL I can overwrite to fake the images? I see the above DNS does not actually exist.

jijojosephk commented 2 years ago

@malanvaneck that was just a url I added while writing comment. Not an actual one.

Preview: https://statics.teams.cdn.office.net/evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice03_thumb.png?v=0.1

Background: https://statics.teams.cdn.office.net/evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice03.png?v=0.1

malanvaneck commented 2 years ago

Thanks @jijojosephk

Actually faking the URL locally will be a problem as its an HTTPS request:

[19537:1011/141457.927329:ERROR:ssl_client_socket_impl.cc(983)] handshake failed; returned -1, SSL error code 1, net_error -202 [ERROR] Unknown cert issuer for url: https://statics.teams.cdn.office.net/evergreen-assets/backgroundimages/fluentSpaces4.png?v=0.1 [ERROR] Issuer Name: Internet Widgits Pty Ltd [ERROR] The unknown certificate fingerprint is: sha256/qF4veWeE4gVPvPp+5bV0ACX4JxChH39nMhTy+in7s3Q=

malanvaneck commented 2 years ago

Apologies, only just now saw the --customCACertsFingerprints option. Got around the cert issues like this: teams-for-linux --customCACertsFingerprints sha256/dRZpQs2cfMlKCdkKOMvGNHASikwm0WmHRtIhEypr880=

Images are not successfully being pulled from my mini python server: 127.0.0.1 - - [11/Oct/2022 16:49:08] "OPTIONS /evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice01.png?v=0.1 HTTP/1.1" 200 - 127.0.0.1 - - [11/Oct/2022 16:49:08] "GET /evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice01.png?v=0.1 HTTP/1.1" 200 - 127.0.0.1 - - [11/Oct/2022 16:49:08] "OPTIONS /evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice01.png?v=0.1 HTTP/1.1" 200 - 127.0.0.1 - - [11/Oct/2022 16:49:08] "GET /evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice01.png?v=0.1 HTTP/1.1" 200 -

But black image on teams-for-linux

jijojosephk commented 2 years ago

@malanvaneck I think you're pretty close. I also get black background. You're not changing the protocol or scheme. You might be stuck in CORS issue. What do you see in the debugger?

I took a slightly different route. Looks like the approach I'm taking requires some help from electron dev I guess. I left a comment in bug here

What I tried is below with response ERR_UNKNOWN_URL_SCHEME

// In the main file
protocol.registerSchemesAsPrivileged([
    {
        scheme: 'atom',
        privileges: {
            bypassCSP: true,
            corsEnabled: true,
            secure: true,
            standard: true,
            supportFetchAPI: true
        }
    }
]);

// When app is ready
protocol.registerHttpProtocol('atom', (request, callback) => {
        request.url = path.join(__dirname, '../assets/bg/bg1.png');
        callback(request);
});

// Before request
window.webContents.session.webRequest.onBeforeRequest({ urls: ['https://*/*'] }, (details, callback) => {
    if (details.url === 'https://mywebsite.com/bg/bg1.png') {
        callback({
            redirectURL: 'atom://bg1.png'
        });
    } else {
        // Proceed normally
        callback({});
    }
});

net::ERR_UNKNOWN_URL_SCHEME

malanvaneck commented 2 years ago

Hi there @jijojosephk, I see these errors in the logs: 41818:1012/093045.034783:ERROR:ssl_client_socket_impl.cc(983)] handshake failed; returned -1, SSL error code 1, net_error -202

Also, I have noticed that overwriting the statics.teams.cdn.office.net DNS name to point to local causes Teams to sometimes be unresponsive.

Xisiqomelir commented 2 years ago

Hello, I'm extremely interested in this feature. I reboot to the other partition literally only to switch on the company-required background for departmental calls and all the rest of the time I'm peacefully in Linux reading chat. With the previous official client I can confirm it was impossible to switch the background, but for the Windows client it was fairly straightforward to add a custom bg.

Where are the default backgrounds stored? Perhaps one of them could be overwritten?

jijojosephk commented 2 years ago

@Xisiqomelir They're not stored locally, served from a MS CDN. I tried to fake the response from a local file but did not succeed. 32253 seems to be blocker from electron framework.

Xisiqomelir commented 2 years ago

Ahh that’s typically M$ and very annoying.

In any case, thanks for your work in keeping the client alive especially this year that they’re trying to kill it.

cpfeiffer commented 1 year ago

@jijojosephk It works if you serve the image via https with an Access-Control-Allow-Origin header (just setting it to * worked). Redirecting to a data: url might also work.

DerXteMensch commented 1 year ago

@cpfeiffer Could you tell us how exactly this can be used?

What code do we need given that we have a URL of a custom background picture? And where do we need to put that code?

Speaking as a user of teams-for-linux, not a developer.

amenk commented 1 year ago

I tried a data URL (that's some Cat Picutre)

window.webContents.session.webRequest.onBeforeRequest({ urls: ['https://*/*'] }, (details, callback) => {
    if (details.url === 'https://statics.teams.cdn.office.net/evergreen-assets/backgroundimages/teamsBackgroundContemporaryOffice03.png?v=0.1') {
        callback({
            redirectURL: ''
        });
    } else {
        // Proceed normally
        callback({});
    }
});

But I am getting

Uncaught TypeError: Cannot read properties of undefined (reading 'session')
    at <anonymous>:1:20

(I tried this in the dev tools )

EDIT: tried inserting it into onBeforeRequestHandler / app/mainAppWindow/index.js:135

UNSAFE_REDIRECT .. happens also for data urls

kw-pr commented 1 year ago

@amenk UNSAFE_REDIRECT : Maybe you need to add the Access-Control-Allow-Origin: * header

Not sure how... I found this https://stackoverflow.com/a/52243138/956397

amenk commented 1 year ago

I tried a lot with CORS and the request but it all caused UNSAFE_REDIRECTS or made everything stop working.

But now I found another way to change the background to a custom one, using the debugger.

image

  1. Search for the function configureEffect
  2. Open a new meeting in a channel
  3. Click any of the built-in backgrounds
  4. Code should stop now at that function
  5. Paste A.backgroundImageUrl = "data: ....." with your image into the console, press enter (probably a hosted image works as well?)
  6. Continue the code

The custom image should show in the preview and I tested that it is also visible to the other party in a call.

Unfortunately that custom background is not preserved after restarts.

The other question is if this can be made usable in a nice way - i.e. is the Electron app able to inject / patch Javascript code?

cpfeiffer commented 1 year ago

@DerXteMensch Sorry, this is nothing for end-users, yet. I was merely trying to check if it works at all, because all previous attempts in this thread appeared to have been unsuccessful. All I did was patch the onBeforeRequestHandler(), check the URL for one of the default background images and redirect to a custom image of my own. The only "special" thing I had to do was return the Access-Control-Allow-Origin: * policy from my custom server.

To make this usable for end-users, one would

jijojosephk commented 1 year ago

Well I have something interesting :smile: Once the page is loaded. Open debug->console and execute the script below.

Replace args[0]=<daturl> with base64 converted image of your choice. You may use https://www.base64-image.de/ to convert.

Click calendar Meet Now Select background effect Click any image

May be we're very close :smile:

const controller = typeof window.angular !== 'undefined' ? window.angular.element(document.documentElement).controller() : null;
const x = controller.callingService._deviceManagerService.setBackgroundImageOnDevice;
controller.callingService._deviceManagerService.setBackgroundImageOnDevice = (...args) => {
    args[0] = '<daturl>'; x.apply(controller.callingService._deviceManagerService, [...args]);
};
jijojosephk commented 1 year ago

@amenk @cpfeiffer try the above.

amenk commented 1 year ago

@jijojosephk Seems to work pretty well. It replaces all the in-built images with the custom one. Draft pull request incoming ...

jijojosephk commented 1 year ago

Breakthrough :tada: Now I need to stitch everything together as a feature. Hooray!

The car image is hosted in apache. So we need to have a light weight web server starting with application at a custom port. Then things are pretty doable.

https://user-images.githubusercontent.com/11662148/236913080-4f5ee054-bf76-431b-a17f-511a24643ada.mp4

jijojosephk commented 1 year ago

@nitro322 @malanvaneck @Xisiqomelir @cpfeiffer @DerXteMensch @amenk @kw-pr try 1.0.84 pre-release. Read https://github.com/IsmaelMartinez/teams-for-linux/tree/develop/app/config

RolandRosenfeld commented 1 year ago

It seems to work, really great!

But I have some annotations:

My custom_bg.json:

[
    {
        "filetype": "jpg",
        "id": "Custom_bg01",
        "name": "Regatta",
        "src": "/teams-for-linux/custom-bg/regatta.jpg",
        "thumb_src": "/teams-for-linux/custom-bg/regatta-thumb.jpg"
    }
]

results in the following apache log:

::1 - - [09/May/2023:09:05:29 +0200] "GET /regatta-thumb.jpg HTTP/1.1" 200 35755 "-" "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.5249.199 Safari/537.36"
::1 - - [09/May/2023:09:05:34 +0200] "GET /regatta.jpg HTTP/1.1" 200 522403 "-" "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.5249.199 Safari/537.36"

while curl http://localhost/teams-for-linux/custom-bg/regatta.jpg gives:

::1 - - [09/May/2023:09:14:16 +0200] "GET /teams-for-linux/custom-bg/regatta.jpg HTTP/1.1" 200 522347 "-" "curl/7.74.0"

No idea, why teams-for-linux strips the path on request...

RolandRosenfeld commented 1 year ago

Seems that I misinterpreted the documentation. /teams-for-linux/custom-bg in src and and thumb_src of the custom_bg.json seem to be fixed strings that can not be changed, while <image-path> is really the path (not only the filename). So now I changed src to "/teams-for-linux/custom-bg/teams/regatta.jpg" and placed regatta.jpg in directory teams below the webserver root and everything works as expected.

jijojosephk commented 1 year ago

@RolandRosenfeld I guess the comment you're talking about is sample in the readme. They were added for the understanding. JSON file can't have comments. Sorry if that was misleading. And for the url, what's given in the json file /teams-for-linux/custom-bg/ will be replaced withhttp://localhost. So, /teams-for-linux/custom-bg/regatta.jpg becomes http://localhost/regatta.jpg

DerXteMensch commented 1 year ago

Thanks so much for your effort. However it's not yet completely working for me.

I have put two jpgs into folder /var/www/html. apache2 is running. And I can view the jpgs in my browser under http://localhost/Teamshintergrund.jpg and http://localhost/Teamshintergrund_thumb.jpg. I have put a newly created custom_bg.json to /home/user/.config/teams-for-linux/custom_bg.json. The json contains:

[

{
    "filetype": "jpg",
    "id": "Teamshintergrund",
    "name": "Teamshintergrund",
    "src": "/teams-for-linux/custom-bg/Teamshintergrund.jpg",
    "thumb_src": "/teams-for-linux/custom-bg/Teamshintergrund_thumb.jpg"
}

]

When I open teams-for-linux, I see the thumbnail. But when I select it, the preview remains with the beforehand selected image (see screenshot). Also when I enter the meeting, it's still the beforehand selected image, not the image from localhost.

Bildschirmfoto vom 2023-05-09 16-29-49

What am I doing wrong?

RolandRosenfeld commented 1 year ago

Are you sure, that you enabled Access-Control-Allow-Origin: *? Try curl -v -s http://localhost/Teamshintergrund.jpg , which should show it in the server headers. Without this header I only see the thumb but not the background...

DerXteMensch commented 1 year ago

I guess I forgot to stop and restart apache2.

First I got an error while restarting:

Mai 09 17:56:18 malte-ThinkPad-X380-Yoga apachectl[50213]: AH00526: Syntax error on line 171 of /etc/apache2/apache2.conf: Mai 09 17:56:18 malte-ThinkPad-X380-Yoga apachectl[50213]: Invalid command 'Header', perhaps misspelled or defined by a module not included in the server configuration

This did the trick:

sudo a2enmod headers

Thanks everybody! That's really great to have custom background images working!

jijojosephk commented 1 year ago

With right configuration mostly people find this working now. Please follow the instructions on release 1.0.84. Closing this now.

jijojosephk commented 1 year ago

Minor breaking change in 1.0.87, read https://github.com/IsmaelMartinez/teams-for-linux/tree/develop/app/config#custom-backgrounds. custom_bg.json to be moved to server.

BzSpi commented 1 year ago

Hi,

I can have it working with the binary download but not with Snap (1.0.88 - edge channel). Is there a limitation ? I have all permissions enabled.

jijojosephk commented 1 year ago

@BzSpi if things are ok with native, stick with that. We don't know what issues are there with snap. Are you using a local server or a remote one?

BzSpi commented 1 year ago

A local one. I can't see any request coming to the server when using snap and nothing in logs.

jijojosephk commented 1 year ago

Can you run the app from the command line? See if there are errors.

BzSpi commented 1 year ago

No error in the command line output and no request have been made to the HTTP server.

$ $ teams-for-linux --appLogLevels error,warn,info,debug
configPath = /home/xxxx/.config/teams-for-linux
configFile = {
  customBGServiceBaseUrl: 'http://localhost/msteams/',
  minimized: true,
  appLogLevels: 'debug'
}
processArgs [
  '/opt/teams-for-linux/teams-for-linux',
  '--appLogLevels',
  'error,warn,info,debug'
]
[180740:0516/084936.761534:ERROR:browser_main_loop.cc(270)] Gtk: gtk_widget_add_accelerator: assertion 'GTK_IS_ACCEL_GROUP (accel_group)' failed
[180740:0516/084936.761664:ERROR:browser_main_loop.cc(270)] Gtk: gtk_widget_add_accelerator: assertion 'GTK_IS_ACCEL_GROUP (accel_group)' failed
[180740:0516/084936.761711:ERROR:browser_main_loop.cc(270)] Gtk: gtk_widget_add_accelerator: assertion 'GTK_IS_ACCEL_GROUP (accel_group)' failed
[180740:0516/084936.761747:ERROR:browser_main_loop.cc(270)] Gtk: gtk_widget_add_accelerator: assertion 'GTK_IS_ACCEL_GROUP (accel_group)' failed
[180740:0516/084936.761784:ERROR:browser_main_loop.cc(270)] Gtk: gtk_widget_add_accelerator: assertion 'GTK_IS_ACCEL_GROUP (accel_group)' failed
did-finish-load
did-finish-load
jijojosephk commented 1 year ago

@BzSpi you need to open debug tools. From the tray icon menu choose debug and go to network tab. see what response you're getting for http://localhost/msteams/<your-image>

BzSpi commented 1 year ago

Hi @jijojosephk

My bad, I had a deb version of the application along with the snap version. I've uninstalled everything all and reinstalled the snap version and this is working file.

Sorry for the inconvenience.