[BACK]Return to auth2-chall.c CVS log [TXT][DIR] Up to [local] / src / usr.bin / ssh

Diff for /src/usr.bin/ssh/auth2-chall.c between version 1.2.2.4 and 1.2.2.5

version 1.2.2.4, 2001/05/07 21:09:26 version 1.2.2.5, 2001/09/27 00:15:41
Line 1 
Line 1 
 /*  /*
  * Copyright (c) 2001 Markus Friedl.  All rights reserved.   * Copyright (c) 2001 Markus Friedl.  All rights reserved.
    * Copyright (c) 2001 Per Allansson.  All rights reserved.
  *   *
  * Redistribution and use in source and binary forms, with or without   * Redistribution and use in source and binary forms, with or without
  * modification, are permitted provided that the following conditions   * modification, are permitted provided that the following conditions
Line 29 
Line 30 
 #include "packet.h"  #include "packet.h"
 #include "xmalloc.h"  #include "xmalloc.h"
 #include "dispatch.h"  #include "dispatch.h"
   #include "auth.h"
 #include "log.h"  #include "log.h"
   
 void send_userauth_into_request(Authctxt *authctxt, char *challenge, int echo);  static int auth2_challenge_start(Authctxt *);
 void input_userauth_info_response(int type, int plen, void *ctxt);  static int send_userauth_info_request(Authctxt *);
   static void input_userauth_info_response(int, int, void *);
   
   #ifdef BSD_AUTH
   extern KbdintDevice bsdauth_device;
   #else
   #ifdef SKEY
   extern KbdintDevice skey_device;
   #endif
   #endif
   
   KbdintDevice *devices[] = {
   #ifdef BSD_AUTH
           &bsdauth_device,
   #else
   #ifdef SKEY
           &skey_device,
   #endif
   #endif
           NULL
   };
   
   typedef struct KbdintAuthctxt KbdintAuthctxt;
   struct KbdintAuthctxt
   {
           char *devices;
           void *ctxt;
           KbdintDevice *device;
   };
   
   static KbdintAuthctxt *
   kbdint_alloc(const char *devs)
   {
           KbdintAuthctxt *kbdintctxt;
           int i;
           char buf[1024];
   
           kbdintctxt = xmalloc(sizeof(KbdintAuthctxt));
           if (strcmp(devs, "") == 0) {
                   buf[0] = '\0';
                   for (i = 0; devices[i]; i++) {
                           if (i != 0)
                                   strlcat(buf, ",", sizeof(buf));
                           strlcat(buf, devices[i]->name, sizeof(buf));
                   }
                   debug("kbdint_alloc: devices '%s'", buf);
                   kbdintctxt->devices = xstrdup(buf);
           } else {
                   kbdintctxt->devices = xstrdup(devs);
           }
           kbdintctxt->ctxt = NULL;
           kbdintctxt->device = NULL;
   
           return kbdintctxt;
   }
   static void
   kbdint_reset_device(KbdintAuthctxt *kbdintctxt)
   {
           if (kbdintctxt->ctxt) {
                   kbdintctxt->device->free_ctx(kbdintctxt->ctxt);
                   kbdintctxt->ctxt = NULL;
           }
           kbdintctxt->device = NULL;
   }
   static void
   kbdint_free(KbdintAuthctxt *kbdintctxt)
   {
           if (kbdintctxt->device)
                   kbdint_reset_device(kbdintctxt);
           if (kbdintctxt->devices) {
                   xfree(kbdintctxt->devices);
                   kbdintctxt->devices = NULL;
           }
           xfree(kbdintctxt);
   }
   /* get next device */
   static int
   kbdint_next_device(KbdintAuthctxt *kbdintctxt)
   {
           size_t len;
           char *t;
           int i;
   
           if (kbdintctxt->device)
                   kbdint_reset_device(kbdintctxt);
           do {
                   len = kbdintctxt->devices ?
                       strcspn(kbdintctxt->devices, ",") : 0;
   
                   if (len == 0)
                           break;
                   for (i = 0; devices[i]; i++)
                           if (strncmp(kbdintctxt->devices, devices[i]->name, len) == 0)
                                   kbdintctxt->device = devices[i];
                   t = kbdintctxt->devices;
                   kbdintctxt->devices = t[len] ? xstrdup(t+len+1) : NULL;
                   xfree(t);
                   debug2("kbdint_next_device: devices %s", kbdintctxt->devices ?
                      kbdintctxt->devices : "<empty>");
           } while (kbdintctxt->devices && !kbdintctxt->device);
   
           return kbdintctxt->device ? 1 : 0;
   }
   
 /*  /*
  * try challenge-reponse, return -1 (= postponed) if we have to   * try challenge-reponse, set authctxt->postponed if we have to
  * wait for the response.   * wait for the response.
  */   */
 int  int
 auth2_challenge(Authctxt *authctxt, char *devs)  auth2_challenge(Authctxt *authctxt, char *devs)
 {  {
         char *challenge;          debug("auth2_challenge: user=%s devs=%s",
               authctxt->user ? authctxt->user : "<nouser>",
               devs ? devs : "<no devs>");
   
         if (!authctxt->valid || authctxt->user == NULL)          if (authctxt->user == NULL || !devs)
                 return 0;                  return 0;
         if ((challenge = get_challenge(authctxt, devs)) == NULL)          if (authctxt->kbdintctxt == NULL)
                   authctxt->kbdintctxt = kbdint_alloc(devs);
           return auth2_challenge_start(authctxt);
   }
   
   /* side effect: sets authctxt->postponed if a reply was sent*/
   static int
   auth2_challenge_start(Authctxt *authctxt)
   {
           KbdintAuthctxt *kbdintctxt = authctxt->kbdintctxt;
   
           debug2("auth2_challenge_start: devices %s",
               kbdintctxt->devices ?  kbdintctxt->devices : "<empty>");
   
           if (kbdint_next_device(kbdintctxt) == 0) {
                   kbdint_free(kbdintctxt);
                   authctxt->kbdintctxt = NULL;
                 return 0;                  return 0;
         send_userauth_into_request(authctxt, challenge, 0);          }
           debug("auth2_challenge_start: trying authentication method '%s'",
               kbdintctxt->device->name);
   
           if ((kbdintctxt->ctxt = kbdintctxt->device->init_ctx(authctxt)) == NULL) {
                   kbdint_free(kbdintctxt);
                   authctxt->kbdintctxt = NULL;
                   return 0;
           }
           if (send_userauth_info_request(authctxt) == 0) {
                   kbdint_free(kbdintctxt);
                   authctxt->kbdintctxt = NULL;
                   return 0;
           }
         dispatch_set(SSH2_MSG_USERAUTH_INFO_RESPONSE,          dispatch_set(SSH2_MSG_USERAUTH_INFO_RESPONSE,
             &input_userauth_info_response);              &input_userauth_info_response);
   
         authctxt->postponed = 1;          authctxt->postponed = 1;
         return 0;          return 0;
 }  }
   
 void  static int
 send_userauth_into_request(Authctxt *authctxt, char *challenge, int echo)  send_userauth_info_request(Authctxt *authctxt)
 {  {
         int nprompts = 1;          KbdintAuthctxt *kbdintctxt;
           char *name, *instr, **prompts;
           int i;
           u_int numprompts, *echo_on;
   
           kbdintctxt = authctxt->kbdintctxt;
           if (kbdintctxt->device->query(kbdintctxt->ctxt,
               &name, &instr, &numprompts, &prompts, &echo_on))
                   return 0;
   
         packet_start(SSH2_MSG_USERAUTH_INFO_REQUEST);          packet_start(SSH2_MSG_USERAUTH_INFO_REQUEST);
         /* name, instruction and language are unused */          packet_put_cstring(name);
         packet_put_cstring("");          packet_put_cstring(instr);
         packet_put_cstring("");          packet_put_cstring("");         /* language not used */
         packet_put_cstring("");          packet_put_int(numprompts);
         packet_put_int(nprompts);          for (i = 0; i < numprompts; i++) {
         packet_put_cstring(challenge);                  packet_put_cstring(prompts[i]);
         packet_put_char(echo);                  packet_put_char(echo_on[i]);
           }
         packet_send();          packet_send();
         packet_write_wait();          packet_write_wait();
   
           for (i = 0; i < numprompts; i++)
                   xfree(prompts[i]);
           xfree(prompts);
           xfree(echo_on);
           xfree(name);
           xfree(instr);
           return 1;
 }  }
   
 void  static void
 input_userauth_info_response(int type, int plen, void *ctxt)  input_userauth_info_response(int type, int plen, void *ctxt)
 {  {
         Authctxt *authctxt = ctxt;          Authctxt *authctxt = ctxt;
         int authenticated = 0;          KbdintAuthctxt *kbdintctxt;
         u_int nresp, rlen;          int i, authenticated = 0, res, len;
         char *response, *method = "challenge-reponse";          u_int nresp;
           char **response = NULL, *method;
   
         if (authctxt == NULL)          if (authctxt == NULL)
                 fatal("input_userauth_info_response: no authctxt");                  fatal("input_userauth_info_response: no authctxt");
           kbdintctxt = authctxt->kbdintctxt;
           if (kbdintctxt == NULL || kbdintctxt->ctxt == NULL)
                   fatal("input_userauth_info_response: no kbdintctxt");
           if (kbdintctxt->device == NULL)
                   fatal("input_userauth_info_response: no device");
   
         authctxt->postponed = 0;        /* reset */          authctxt->postponed = 0;        /* reset */
         nresp = packet_get_int();          nresp = packet_get_int();
         if (nresp == 1) {          if (nresp > 0) {
                 response = packet_get_string(&rlen);                  response = xmalloc(nresp * sizeof(char*));
                 packet_done();                  for (i = 0; i < nresp; i++)
                 if (strlen(response) == 0) {                          response[i] = packet_get_string(NULL);
                         /*          }
                          * if we received an empty response, resend challenge          packet_done();
                          * with echo enabled  
                          */          if (authctxt->valid) {
                         char *challenge = get_challenge(authctxt, NULL);                  res = kbdintctxt->device->respond(kbdintctxt->ctxt,
                         if (challenge != NULL) {                      nresp, response);
                                 send_userauth_into_request(authctxt,          } else {
                                     challenge, 1);                  res = -1;
                                 authctxt->postponed = 1;          }
                         }  
                 } else if (authctxt->valid) {          for (i = 0; i < nresp; i++) {
                         authenticated = verify_response(authctxt, response);                  memset(response[i], 'r', strlen(response[i]));
                         memset(response, 'r', rlen);                  xfree(response[i]);
                 }          }
           if (response)
                 xfree(response);                  xfree(response);
   
           switch (res) {
           case 0:
                   /* Success! */
                   authenticated = 1;
                   break;
           case 1:
                   /* Authentication needs further interaction */
                   authctxt->postponed = 1;
                   if (send_userauth_info_request(authctxt) == 0) {
                           authctxt->postponed = 0;
                   }
                   break;
           default:
                   /* Failure! */
                   break;
         }          }
         /* unregister callback */  
         if (!authctxt->postponed)          len = strlen("keyboard-interactive") + 2 +
                   strlen(kbdintctxt->device->name);
           method = xmalloc(len);
           method[0] = '\0';
           strlcat(method, "keyboard-interactive", len);
           strlcat(method, "/", len);
           strlcat(method, kbdintctxt->device->name, len);
   
           if (!authctxt->postponed) {
                   /* unregister callback */
                 dispatch_set(SSH2_MSG_USERAUTH_INFO_RESPONSE, NULL);                  dispatch_set(SSH2_MSG_USERAUTH_INFO_RESPONSE, NULL);
   
                   if (authenticated) {
                           kbdint_free(kbdintctxt);
                           authctxt->kbdintctxt = NULL;
                   } else {
                           /* start next device */
                           /* may set authctxt->postponed */
                           auth2_challenge_start(authctxt);
                   }
           }
         userauth_finish(authctxt, authenticated, method);          userauth_finish(authctxt, authenticated, method);
           xfree(method);
 }  }

Legend:
Removed from v.1.2.2.4  
changed lines
  Added in v.1.2.2.5