Merge commit 'origin/platal-0.10.0'
[platal.git] / include / googleapps.inc.php
index abfa384..71853c4 100644 (file)
@@ -1,6 +1,6 @@
 <?php
 /***************************************************************************
- *  Copyright (C) 2003-2008 Polytechnique.org                              *
+ *  Copyright (C) 2003-2009 Polytechnique.org                              *
  *  http://opensource.polytechnique.org/                                   *
  *                                                                         *
  *  This program is free software; you can redistribute it and/or modify   *
  *  59 Temple Place, Suite 330, Boston, MA  02111-1307  USA                *
  ***************************************************************************/
 
-// Checks the admin status of the @p account_name.
-function is_google_apps_administrator($account_name) {
-    static $last_account_name = null;
-    static $last_result = null;
-
-    if ($last_account_name == $account_name) {
-        return $last_result;
-    }
-
-    $res = XDB::query(
-        "SELECT  g_admin
-           FROM  gapps_accounts
-          WHERE  g_account_name = {?} AND g_status = 'active'",
-        $account_name);
-    $last_account_name = $account_name;
-    $last_result = ($res->numRows() > 0 ? (bool)$res->fetchOneRow() : false);
-    return $last_result;
-}
-
-// Post-queue job cleanup functions; they are used to update the plat/al database
-// when a specific Google Apps queue job enters 'success' state.
+// Post-processes the successful Google Apps account creation queue job.
 function post_queue_u_create($job) {
     global $globals;
 
-    // Retrieves the user parameters (userid and forlife).
+    // Retrieves the user parameters (GoogleApps username and user_id).
     $parameters = json_decode($job['j_parameters'], true);
-    $forlife = isset($parameters['username']) ? $parameters['username'] : null;
-    $userid = $job['q_recipient_id'];
-    if (!$forlife || !$userid) {
+    $username = isset($parameters['username']) ? $parameters['username'] : null;
+    if (!($user = User::getSilent($username))) {
         return;
     }
 
-    // Adds a redirection to the Google Apps delivery address.
-    $account = new GoogleAppsAccount($userid, $forlife);
+    // Adds a redirection to the Google Apps delivery address, if requested by
+    // the user at creation time.
+    $account = new GoogleAppsAccount($user);
     if ($account->activate_mail_redirection) {
         require_once('emails.inc.php');
-        $storage = new MailStorageGoogleApps($userid);
-        $storage->enable();
+        $storage = new EmailStorage($user, 'googleapps');
+        $storage->activate();
     }
 
-    // Sends an email to the account owner.
+    // Sends the 'account created' email to the user, with basic documentation.
     $res = XDB::query(
         "SELECT  FIND_IN_SET('femme', u.flags), prenom
            FROM  auth_user_md5 AS u
-     INNER JOIN  aliases AS a ON (a.id = u.user_id)
-          WHERE  a.alias = {?}",
-        $forlife);
+          WHERE  u.user_id = {?}", $user->id());
     list($sexe, $prenom) = $res->fetchOneRow();
 
     $mailer = new PlMailer('googleapps/create.mail.tpl');
     $mailer->assign('account', $account);
-    $mailer->assign('email', $forlife . '@' . $globals->mail->domain);
+    $mailer->assign('email', $user->bestEmail());
     $mailer->assign('googleapps_domain', $globals->mailstorage->googleapps_domain);
     $mailer->assign('prenom', $prenom);
     $mailer->assign('sexe', $sexe);
     $mailer->send();
 }
 
+// Post-processes the successful Google Apps account update queue job.
 function post_queue_u_update($job) {
     global $globals;
 
@@ -84,34 +63,31 @@ function post_queue_u_update($job) {
     // to the Google Apps delivery address, provided the account is active (it might
     // have been deleted between the unsuspension and the post-queue processing).
     $parameters = json_decode($job['j_parameters'], true);
-    $forlife = isset($parameters['username']) ? $parameters['username'] : null;
-    $userid = $job['q_recipient_id'];
-    if (!$forlife || !$userid) {
+    $username = isset($parameters['username']) ? $parameters['username'] : null;
+    if (!($user = User::getSilent($username))) {
         return;
     }
 
     if (isset($parameters['suspended']) && $parameters['suspended'] == false) {
         require_once('emails.inc.php');
-        $account = new GoogleAppsAccount($userid, $forlife);
-        if ($account->g_status == 'active') {
+        $account = new GoogleAppsAccount($user);
+        if ($account->active()) {
             // Re-adds the email redirection (if the user did request it).
             if ($account->activate_mail_redirection) {
-                $storage = new MailStorageGoogleApps($userid);
-                $storage->enable();
+                $storage = new EmailStorage($user, 'googleapps');
+                $storage->activate();
             }
 
             // Sends an email to the account owner.
             $res = XDB::query(
                 "SELECT  FIND_IN_SET('femme', u.flags), prenom
                    FROM  auth_user_md5 AS u
-             INNER JOIN  aliases AS a ON (a.id = u.user_id)
-                  WHERE  a.alias = {?}",
-                $forlife);
+                  WHERE  u.user_id = {?}", $user->id());
             list($sexe, $prenom) = $res->fetchOneRow();
 
             $mailer = new PlMailer('googleapps/unsuspend.mail.tpl');
             $mailer->assign('account', $account);
-            $mailer->assign('email', $forlife . '@' . $globals->mail->domain);
+            $mailer->assign('email', $user->bestEmail());
             $mailer->assign('prenom', $prenom);
             $mailer->assign('sexe', $sexe);
             $mailer->send();
@@ -120,13 +96,21 @@ function post_queue_u_update($job) {
 }
 
 // Reprensentation of an SQL-stored Google Apps account.
+// This class is the interface with the gappsd SQL tables: gappsd is the python
+// daemon which deals with Google Apps provisioning APIs.
+// TODO(vincent.zanotti): add the url of gappsd, when available.
 class GoogleAppsAccount
 {
-    private $uid;
+    // User identification: user id, and hruid.
+    private $user;
     public $g_account_name;
 
+    // Local account parameters.
     public $sync_password;
     public $activate_mail_redirection;
+
+    // Account status, obtained from Google Apps provisioning & reporting APIs.
+    public $g_account_id;
     public $g_status;
     public $g_suspension;
     public $r_disk_usage;
@@ -135,6 +119,7 @@ class GoogleAppsAccount
     public $r_last_webmail;
     public $reporting_date;
 
+    // Pending requests in the gappsd job queue (cf. top note).
     public $pending_create;
     public $pending_delete;
     public $pending_update;
@@ -143,26 +128,35 @@ class GoogleAppsAccount
     public $pending_update_password;
     public $pending_update_suspension;
 
+    // Pending requests in plat/al validation queue.
     public $pending_validation_unsuspend;
 
-    public function __construct($uid, $account_name)
+    // Constructs the account object, by retrieving all informations from the
+    // GApps account table, from GApps job queue, and from plat/al validation queue.
+    public function __construct(User &$user)
     {
-        $this->uid = $uid;
-        $this->g_account_name = $account_name;
+        $this->user = &$user;
+        if (!$this->user || !$this->user->login()) {
+            return;
+        }
+
+        // TODO: switch to multi-domain Google Apps, and use $this->user->forlifeEmail()
+        // as Google Apps idenfiant (requires changes in gappsd).
+        $this->g_account_name = $this->user->login();
         $this->g_status = NULL;
 
         $res = XDB::query(
             "SELECT  l_sync_password, l_activate_mail_redirection,
-                     g_account_name, g_status, g_suspension, r_disk_usage,
+                     g_account_name, g_account_id, g_status, g_suspension, r_disk_usage,
                      UNIX_TIMESTAMP(r_creation) as r_creation,
                      UNIX_TIMESTAMP(r_last_login) as r_last_login,
                      UNIX_TIMESTAMP(r_last_webmail) as r_last_webmail
                FROM  gapps_accounts
-              WHERE  g_account_name = {?}",
-            $account_name);
+              WHERE  g_account_name = {?}", $this->g_account_name);
         if ($account = $res->fetchOneAssoc()) {
             $this->sync_password = $account['l_sync_password'];
             $this->activate_mail_redirection = $account['l_activate_mail_redirection'];
+            $this->g_account_id = $account['g_account_id'];
             $this->g_status = $account['g_status'];
             $this->g_suspension = $account['g_suspension'];
             $this->r_disk_usage = $account['r_disk_usage'];
@@ -181,11 +175,10 @@ class GoogleAppsAccount
         }
     }
 
-    // Account object initialization methods.
+    // Determines if changes to the Google Account are currently waiting in the
+    // GApps job queue, and initializes the local values accordingly.
     private function load_pending_counts()
     {
-        // Determines if changes to the Google Account are currently waiting
-        // in the Google Apps queue.
         $res = XDB::query(
             "SELECT  SUM(j_type = 'u_create') AS pending_create,
                      SUM(j_type = 'u_update') AS pending_update,
@@ -193,8 +186,7 @@ class GoogleAppsAccount
                FROM  gapps_queue
               WHERE  q_recipient_id = {?} AND
                      p_status IN ('idle', 'active', 'softfail')
-           GROUP BY  j_type",
-            $this->uid);
+           GROUP BY  j_type", $this->user->id());
         $pending = $res->fetchOneAssoc();
         $this->pending_create = $pending['pending_create'];
         $this->pending_update = $pending['pending_update'];
@@ -206,24 +198,26 @@ class GoogleAppsAccount
         $this->pending_update_suspension = false;
     }
 
+    // Checks for unsuspend requests waiting for validation in plat/al
+    // validation queue.
     private function load_pending_validations()
     {
         require_once('validations.inc.php');
         $this->pending_validation_unsuspend =
-            Validate::get_typed_requests_count($this->uid, 'gapps-unsuspend');
+            Validate::get_typed_requests_count($this->user->id(), 'gapps-unsuspend');
     }
 
+    // Retrieves all the pending update job in the gappsd queue for the current
+    // user, and analyzes the scope of the update (ie. the fields in the user
+    // account which are going to be updated).
     private function load_pending_updates()
     {
-        // If updates are pending, determines their nature (more specifically:
-        // determines which part of the account is concerned).
         $res = XDB::iterator(
             "SELECT  j_parameters
                FROM  gapps_queue
               WHERE  q_recipient_id = {?} AND
                      p_status IN ('idle', 'active', 'softfail') AND
-                     j_type = 'u_update'",
-            $this->uid);
+                     j_type = 'u_update'", $this->user->id());
         while ($update = $res->next()) {
             $update_data = json_decode($update["j_parameters"], true);
 
@@ -240,36 +234,62 @@ class GoogleAppsAccount
     }
 
     // Creates a queue job of the @p type, for the user represented by this
-    // GoogleAppsAccount object, using @p parameters.
+    // GoogleAppsAccount object, using @p parameters. @p parameters is supposed
+    // to be a one-dimension array of key-value mappings.
+    // The created job as a 'immediate' priority, and is scheduled for immediate
+    // execution.
     private function create_queue_job($type, $parameters) {
         $parameters["username"] = $this->g_account_name;
         XDB::execute(
             "INSERT  INTO gapps_queue
                 SET  q_owner_id = {?}, q_recipient_id = {?},
                      p_entry_date = NOW(), p_notbefore_date = NOW(),
-                     p_priority = 'normal',
+                     p_priority = 'immediate',
                      j_type = {?}, j_parameters = {?}",
             S::v('uid'),
-            $this->uid,
+            $this->user->id(),
             $type,
             json_encode($parameters));
     }
 
+
+    // Returns true if the account is currently active.
+    public function active()
+    {
+        return $this->g_status == 'active';
+    }
+
+    // Returns true if the account exists in Google Apps.
+    public function provisioned()
+    {
+        return $this->g_status == 'active' or $this->g_status == 'disabled';
+    }
+
+    // Returns true if the account exists, but cannot be used (user-requested
+    // suspension, or Google-requested suspension).
+    public function suspended()
+    {
+        return $this->g_status == 'disabled';
+    }
+
+
     // Changes the GoogleApps password.
     public function set_password($password) {
-        if ($this->g_status == NULL || $this->g_status == 'unprovisioned') {
+        if (!$this->provisioned()) {
             return;
         }
 
         if (!$this->pending_update_password) {
             $this->create_queue_job('u_update', array('password' => $password));
+            $this->pending_update_password = true;
         }
     }
 
+
     // Changes the password synchronization status ("sync = true" means that the
     // Polytechnique.org password will be replicated to the Google Apps account).
     public function set_password_sync($sync) {
-        if ($this->g_status == NULL || $this->g_status == 'unprovisioned') {
+        if (!$this->provisioned()) {
             return;
         }
 
@@ -284,19 +304,24 @@ class GoogleAppsAccount
 
     // Suspends the Google Apps account.
     public function suspend() {
-        if ($this->g_status == NULL || $this->g_status == 'unprovisioned') {
+        if (!$this->provisioned()) {
             return;
         }
 
         if (!$this->pending_update_suspension) {
             $this->create_queue_job('u_update', array('suspended' => true));
             $this->pending_update_suspension = true;
+            XDB::execute(
+                "UPDATE  gapps_accounts
+                    SET  g_status = 'disabled'
+                  WHERE  g_account_name = {?} AND g_status = 'active'",
+                $this->g_account_name);
         }
     }
 
     // Adds an unsuspension request to the validation queue (used on user-request).
     public function unsuspend($activate_mail_redirection = NULL) {
-        if ($this->g_status == NULL || $this->g_status == 'unprovisioned') {
+        if (!$this->provisioned()) {
             return;
         }
         if ($activate_mail_redirection !== NULL) {
@@ -305,12 +330,13 @@ class GoogleAppsAccount
                 "UPDATE  gapps_accounts
                     SET  l_activate_mail_redirection = {?}
                   WHERE  g_account_name = {?}",
-                $activate_mail_redirection);
+                $activate_mail_redirection,
+                $this->g_account_name);
         }
 
         if (!$this->pending_update_suspension && !$this->pending_validation_unsuspend) {
             require_once('validations.inc.php');
-            $unsuspend = new GoogleAppsUnsuspendReq($this->uid);
+            $unsuspend = new GoogleAppsUnsuspendReq($this->user);
             $unsuspend->submit();
             $this->pending_validation_unsuspend = true;
         }
@@ -319,7 +345,7 @@ class GoogleAppsAccount
     // Unsuspends the Google Apps account (used on admin-request, or on validation of
     // an user-request).
     public function do_unsuspend() {
-        if ($this->g_status == NULL || $this->g_status == 'unprovisioned') {
+        if (!$this->provisioned()) {
             return;
         }
 
@@ -328,8 +354,7 @@ class GoogleAppsAccount
                 $res = XDB::query(
                     "SELECT  password
                        FROM  auth_user_md5
-                      WHERE  user_id = {?}",
-                    $this->uid);
+                      WHERE  user_id = {?}", $this->user->id());
                 $password = ($res->numRows() > 0 ? $res->fetchOneCell() : false);
             } else {
                 $password = false;
@@ -346,7 +371,7 @@ class GoogleAppsAccount
         return false;
     }
 
-    // Adds a creation request in the job queue.
+    // Creates a new Google Apps account with the @p local parameters.
     public function create($password_sync, $password, $redirect_mails) {
         if ($this->g_status != NULL) {
             return;
@@ -357,11 +382,10 @@ class GoogleAppsAccount
             $res = XDB::query(
                 "SELECT  nom, nom_usage, prenom
                    FROM  auth_user_md5
-                  WHERE  user_id = {?}",
-                $this->uid);
+                  WHERE  user_id = {?}", $this->user->id());
             list($nom, $nom_usage, $prenom) = $res->fetchOneRow();
 
-            // Adds an entry in the gapps_accounts table.
+            // Adds an 'unprovisioned' entry in the gapps_accounts table.
             XDB::execute(
                 "INSERT  INTO gapps_accounts
                     SET  l_userid = {?},
@@ -371,7 +395,7 @@ class GoogleAppsAccount
                          g_first_name = {?},
                          g_last_name = {?},
                          g_status = 'unprovisioned'",
-                $this->uid,
+                $this->user->id(),
                 $password_sync,
                 $redirect_mails,
                 $this->g_account_name,
@@ -389,9 +413,29 @@ class GoogleAppsAccount
                 ));
 
             // Updates the GoogleAppsAccount status.
-            $this->__construct($this->uid, $this->g_account_name);
+            $this->__construct($this->user);
         }
     }
+
+
+    // Returns the status of the Google Apps account for @p user, or false
+    // when no account exists.
+    static public function account_status($uid) {
+        $res = XDB::query(
+            "SELECT  g_status
+               FROM  gapps_accounts
+              WHERE  l_userid = {?}", $uid);
+        return ($res->numRows() > 0 ? $res->fetchOneCell() : false);
+    }
+
+    // Returns true if the @p user is an administrator of the Google Apps domain.
+    static public function is_administrator($uid) {
+        $res = XDB::query(
+            "SELECT  g_admin
+               FROM  gapps_accounts
+              WHERE  l_userid = {?} AND g_status = 'active'", $uid);
+        return ($res->numRows() > 0 ? (bool)$res->fetchOneCell() : false);
+    }
 }
 
 // vim:set et sw=4 sts=4 sws=4 foldmethod=marker enc=utf-8: