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

http authentication not disabled when using --auth-type=http_x_remote_user #1119

Open
grunlab opened this issue Nov 5, 2020 · 10 comments
Open
Labels
need:support support/help/contribution is required prio:medium medium priority reverse proxy reverse proxy related webui built-in WebUI
Milestone

Comments

@grunlab
Copy link

grunlab commented Nov 5, 2020

Use case:

I'm trying to configure radicale this way:

  • single user usage (user=adrien)
  • basic auth managed at reverse proxy level with user name forwarded to radicale once authentication done.
  • disable login/passwd prompt at radicale level <-- this is my issue ... see below

Radicale is running into a docker container deployed on top of a kubernetes cluster.
Traefik is used as edge router to access the apps running into the cluster.

Radicale config:

python3 -m radicale --server-hosts=0.0.0.0:5232 --auth-type=http_x_remote_user --storage-filesystem-folder=/mnt/collections --logging-level=info

Authentication type set to http_x_remote_user in order to:

  • get the authenticated user name from X-Remote-User header set by Treafik
  • disable the http authentication at radicale level

Traefik config:

  • Middleware basicAuth radicale-basic-auth configured to manage login/password at Traefik level + forward authenticated user name to radicale with X-Remote-User header:
apiVersion: traefik.containo.us/v1alpha1
kind: Middleware
metadata:
  name: radicale-basic-auth
  namespace: radicale-p
spec:
  basicAuth:
    secret: radicale-basic-auth
    headerField: X-Remote-User
  • IngressRoute configured to access radicale app into the cluster with radicale-basic-auth middleware activated:
apiVersion: traefik.containo.us/v1alpha1
kind: IngressRoute
metadata:
  name: radicale
  namespace: radicale-p
spec:
  entryPoints:
    - https
  routes:
  - match: Host(`<my_url>`)
    kind: Rule
    services:
    - name: radicale
      port: 80
    middlewares:
      - name: radicale-basic-auth
  tls:
    certResolver: default

Expected result:

  • Open the app url --> get Traefik login/passwd prompt : OK
  • Check radicale logs to see if user name adrien has been correctly forwarded by traefik to the app --> OK
kubectl logs radicale-7856dd6fcd-hs76h -f
[2020-11-05 22:24:42 +0100] [1] [INFO] Loaded default config
[2020-11-05 22:24:42 +0100] [1] [INFO] Skipped missing config file '/etc/radicale/config'
[2020-11-05 22:24:42 +0100] [1] [INFO] Skipped missing config file '/home/k8s/.config/radicale/config'
[2020-11-05 22:24:42 +0100] [1] [INFO] Loaded arguments
[2020-11-05 22:24:42 +0100] [1] [INFO] Starting Radicale
[2020-11-05 22:24:42 +0100] [1] [INFO] auth type is 'radicale.auth.http_x_remote_user'
[2020-11-05 22:24:42 +0100] [1] [INFO] storage type is 'radicale.storage.multifilesystem'
[2020-11-05 22:24:42 +0100] [1] [INFO] rights type is 'radicale.rights.owner_only'
[2020-11-05 22:24:42 +0100] [1] [INFO] web type is 'radicale.web.internal'
[2020-11-05 22:24:42 +0100] [1] [INFO] Listening on '[0.0.0.0]:5232'
[2020-11-05 22:24:42 +0100] [1] [INFO] Radicale server ready
[2020-11-05 22:24:55 +0100] [1/Thread-7] [INFO] GET request for '/' received from '10.44.0.0' (forwarded by 10.36.0.0) using 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:82.0) Gecko/20100101 Firefox/82.0'
[2020-11-05 22:24:55 +0100] [1/Thread-7] [INFO] Successful login: 'adrien'
[2020-11-05 22:24:55 +0100] [1/Thread-7] [INFO] GET response status for '/' in 0.062 seconds: 302 Found
[2020-11-05 22:24:55 +0100] [1/Thread-8] [INFO] GET request for '/.web' received from '10.44.0.0' (forwarded by 10.36.0.0) using 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:82.0) Gecko/20100101 Firefox/82.0'
[2020-11-05 22:24:55 +0100] [1/Thread-8] [INFO] Successful login: 'adrien'
[2020-11-05 22:24:55 +0100] [1/Thread-8] [INFO] GET response status for '/.web' in 0.033 seconds: 302 Found
[2020-11-05 22:24:55 +0100] [1/Thread-9] [INFO] GET request for '/.web/css/main.css' received from '10.44.0.0' (forwarded by 10.36.0.0) using 'Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:82.0) Gecko/20100101 Firefox/82.0'
[2020-11-05 22:24:55 +0100] [1/Thread-9] [INFO] Successful login: 'adrien'
[2020-11-05 22:24:55 +0100] [1/Thread-9] [INFO] GET response status for '/.web/css/main.css' in 0.040 seconds: 200 OK
  • check radicale collections if adrien folder exist --> OK
kubectl exec -ti radicale-7856dd6fcd-hs76h -- ls -l /mnt/collections/collection-root
total 4
drwxr-xr-x 2 k8s k8s 4096 nov.   5 21:12 adrien
  • http authentication disabled at radicale level --> KO ... I'm no able to get the radicale login/passwd prompt disabled !

Did i missed something into the configuration or is there a bug somewhere !?

Thank you for your support

@aerusso
Copy link

aerusso commented Dec 26, 2020

I also ran into this problem. I worked around it by just curl-ing the CalDAV and CardDAV commands.

@jtorrex
Copy link

jtorrex commented Jan 24, 2021

Same case here as detailed by @grunlab.

@timakro
Copy link

timakro commented Aug 16, 2021

Seems like you can log in with any username and password. And even though it says "logged in as xyz", the effective user is always the one from X-Remote-User. No security issue here but confusing UX.

@chris2fr
Copy link

chris2fr commented Oct 8, 2022

Yes, confusing GUI.

@gardiol
Copy link

gardiol commented Jan 11, 2024

Agreed, some thing happens here. User seems logged, but web ui requires another login (useless). Works tough. Just confusing and ugly.

@57194
Copy link

57194 commented Aug 5, 2024

I'm not sure exactly where all in the code it would need to be updated, but basically anywhere in the code the login page would be called, I think there just needs to be a check of the config file and just send the user in if they have the config option set and the right header. Would probably need another "passwordless" code path added, since most of the login code looks like it all requires a (username, password) tuple.

@pbiering pbiering added reverse proxy reverse proxy related prio:medium medium priority labels Aug 5, 2024
@pbiering pbiering added this to the 3.2.3 milestone Aug 5, 2024
@pbiering pbiering modified the milestones: 3.2.3, 3.2.x Aug 27, 2024
@pbiering pbiering added the need:reporter feedback feedback from reporter required label Aug 27, 2024
@pbiering
Copy link
Collaborator

Hi all, I need more details to understand what happens when and what is unexpected? Would be good if one can provide related request+response headers.

@pbiering pbiering added webui built-in WebUI need:support support/help/contribution is required and removed need:reporter feedback feedback from reporter required labels Aug 28, 2024
@pbiering
Copy link
Collaborator

I ran some analysis and assume the issue is related to WebUI only which still after passing Basic Auth on a reverse proxy, which protects e.g. complete /radicale/* is displaying the login screen and also requires credentials to work.

This is somehow by nature as the WebUI is client-only Javascript implementation and requires the credentials to run the C*DAV-requests.

@MatthewHana : is there any chance that the WebUI is able to take advantage of already provided BasicAuth of the initial request, so to say "borrow" for it's own requests generated in Javascript the BasicAuth from the browser session?

Final goal would be bypassing the login screen if surrounding BasicAuth is already provided and tested by Javascript successfully in case of "forbidden" fallback to login screen.

A workaround would be excluding the .web from BasicAuth, but I haven't tested whether this is supported by web servers to exclude a sub-path from BasicAuth while the main path is protected. If not supported, one can test internal redirects and expose the WebUI on a different main path outside BasicAuth - but here potentially some base URI must be adjusted.

@gardiol
Copy link

gardiol commented Aug 28, 2024

Could this info be of any use? https://www.authelia.com/integration/proxies/support/#standard

ALso check the NGINX configiration files for authelia here https://www.authelia.com/integration/proxies/nginx/#authelia-authrequestconf and here https://www.authelia.com/integration/proxies/nginx/#authelia-location-basicconf

It already just works with proxy auth, i guess it's only matter of checking the right headers?

@selenith
Copy link

selenith commented Sep 14, 2024

Hi,
I made a quick and dirty hack that gets the username from the server environment variables and submits the form when the page is opened on my server.

If it helps, here's how I did it.

I've created a php page in the same vhost that returns the user name present in the variable $_SERVER['REMOTE_USER'] in json like this: {“user”:“<login_in_http_x_remote_user>”}

My php code :

//file userinfos.php
<?php

$user_data = ['user'=>'not_connected'];
if(!empty($_SERVER["REMOTE_USER"])){

            $user_data['user'] =$_SERVER["REMOTE_USER"];
}

echo(json_encode($user_data));
?>

I modified the fn.js file in two places:

  • In the object LoginScene I've added the following function "on_login_ext" to call onlogin from the outside
function LoginScene() {
/* .....  */
    this.on_login_ext = function(){
        onlogin();
    }
/* .....  */
}
  • I've modified the main function like this :
function main() {



    // Hide startup loading message
    document.getElementById("loadingscene").classList.add("hidden");
    let log_screen = new LoginScene();
    push_scene(log_screen, false);

    // callback executed once the file giving the user name has been requested.
    function get_username_and_connect() {
        let username = this.response.user;
        let html_scene = document.getElementById("loginscene");
        let form = html_scene.querySelector("[data-name=form]");
        let user_form = html_scene.querySelector("[data-name=user]");
        // fill the form and submit
        user_form.value = username;
        log_screen.on_login_ext();
    }

    const req = new XMLHttpRequest();
    req.responseType = 'json';
    req.addEventListener("load", get_username_and_connect);
    req.open("GET", "/userinfos.php");
    req.send();

}

With a little tweaking of the nginx config, I also got it to work with p12 certificates for authentication via HTTPS without password. If anyone is interested I can detail the configuration.

@pbiering pbiering modified the milestones: 3.2.x, 3.3.x Sep 27, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
need:support support/help/contribution is required prio:medium medium priority reverse proxy reverse proxy related webui built-in WebUI
Projects
None yet
Development

No branches or pull requests

9 participants