diff --git a/core/Piwik.php b/core/Piwik.php
index 12a091d6a29f3ef785a959e72771011e56b47f13..b187e442e7a603981ee7b29141b9b3fba66ee0f0 100644
--- a/core/Piwik.php
+++ b/core/Piwik.php
@@ -2595,4 +2595,38 @@ class Piwik
 		$db = Zend_Registry::get('db');
 		return $db->fetchOne($sql, array($lockName)) == '1';
 	}
+	
+	/**
+	 * Cached result of isLockprivilegeGranted function.
+	 * 
+	 * Public so tests can simulate the situation where the lock tables privilege isn't granted.
+	 * 
+	 * @var bool
+	 */
+	static public $lockPrivilegeGranted = null;
+	
+	/**
+	 * Checks whether the database user is allowed to lock tables.
+	 * 
+	 * @return bool
+	 */
+	static public function isLockPrivilegeGranted()
+	{
+		if (is_null(self::$lockPrivilegeGranted))
+		{
+			try
+			{
+				Piwik_LockTables(Piwik_Common::prefixTable('log_visit'));
+				Piwik_UnlockAllTables();
+				
+				self::$lockPrivilegeGranted = true;
+			}
+			catch (Exception $ex)
+			{
+				self::$lockPrivilegeGranted = false;
+			}
+		}
+		
+		return self::$lockPrivilegeGranted;
+	}
 }
diff --git a/core/PluginsFunctions/Sql.php b/core/PluginsFunctions/Sql.php
index 039178217127cc96c6e378448e45fe13b27881a6..c2f63e9a9c67c4e854c79eb04b334623d7e9508c 100644
--- a/core/PluginsFunctions/Sql.php
+++ b/core/PluginsFunctions/Sql.php
@@ -181,7 +181,7 @@ class Piwik_Sql
 	 * @param string|array  $tablesToWrite  The table or tables to obtain 'write' locks on.
 	 * @return Zend_Db_Statement
 	 */
-	static public function lockTables( $tablesToRead, $tablesToWrite )
+	static public function lockTables( $tablesToRead, $tablesToWrite = array() )
 	{
 		if (!is_array($tablesToRead))
 		{
@@ -354,7 +354,7 @@ function Piwik_DropTables( $tables )
  * @param string|array  $tablesToWrite  The table or tables to obtain 'write' locks on.
  * @return Zend_Db_Statement
  */
-function Piwik_LockTables( $tablesToRead, $tablesToWrite )
+function Piwik_LockTables( $tablesToRead, $tablesToWrite = array() )
 {
 	return Piwik_Sql::lockTables($tablesToRead, $tablesToWrite);
 }
diff --git a/plugins/Installation/FormDatabaseSetup.php b/plugins/Installation/FormDatabaseSetup.php
index ed1edaf4ab77777b70b0f5a34b264c0ba5dda771..d5aed1174a8d3c22d102b93bc1ee672f316e4a76 100644
--- a/plugins/Installation/FormDatabaseSetup.php
+++ b/plugins/Installation/FormDatabaseSetup.php
@@ -148,7 +148,6 @@ class Piwik_Installation_FormDatabaseSetup extends Piwik_QuickForm2
  * - INSERT
  * - UPDATE
  * - DELETE
- * - LOCK TABLES
  * - DROP
  * - CREATE TEMPORARY TABLES
  * 
@@ -244,7 +243,8 @@ class Piwik_Installation_FormDatabaseSetup_Rule_checkUserPrivileges extends HTML
 	 * array maps privilege names with one or more SQL queries that can be used to test
 	 * if the current user has the privilege.
 	 * 
-	 * NOTE: LOAD DATA INFILE privilege is not **required** so its not checked.
+	 * NOTE: LOAD DATA INFILE & LOCK TABLES privileges are not **required** so they're
+	 * not checked.
 	 * 
 	 * @return array
 	 */
@@ -263,7 +263,6 @@ class Piwik_Installation_FormDatabaseSetup_Rule_checkUserPrivileges extends HTML
 			'INSERT' => 'INSERT INTO '.self::TEST_TABLE_NAME.' (value) VALUES (123)',
 			'UPDATE' => 'UPDATE '.self::TEST_TABLE_NAME.' SET value = 456 WHERE id = 1',
 			'DELETE' => 'DELETE FROM '.self::TEST_TABLE_NAME.' WHERE id = 1',
-			'LOCK TABLES' => array('LOCK TABLES '.self::TEST_TABLE_NAME.' WRITE', 'UNLOCK TABLES'),
 			'DROP' => 'DROP TABLE '.self::TEST_TABLE_NAME,
 			'CREATE TEMPORARY TABLES' => 'CREATE TEMPORARY TABLE '.self::TEST_TEMP_TABLE_NAME.' (
 											id INT AUTO_INCREMENT,
diff --git a/plugins/PrivacyManager/LogDataPurger.php b/plugins/PrivacyManager/LogDataPurger.php
index 68f657c15d71c07409d0eb6c8866e34a0df56fc6..d16cadae5ace4930a3ff345d8824bfbb68c1cf8e 100755
--- a/plugins/PrivacyManager/LogDataPurger.php
+++ b/plugins/PrivacyManager/LogDataPurger.php
@@ -73,8 +73,16 @@ class Piwik_PrivacyManager_LogDataPurger
 			}
 		}
 		
-		// delete unused actions from the log_action table
-		$this->purgeUnusedLogActions();
+		// delete unused actions from the log_action table (but only if we can lock tables)
+		if (Piwik::isLockPrivilegeGranted())
+		{
+			$this->purgeUnusedLogActions();
+		}
+		else
+		{
+			$logMessage = get_class($this).": LOCK TABLES privilege not granted; skipping unused actions purge";
+			Piwik::log($logMessage);
+		}
 		
 		// optimize table overhead after deletion
 		Piwik_OptimizeTables($logTables);
diff --git a/plugins/PrivacyManager/tests/PrivacyManager.test.php b/plugins/PrivacyManager/tests/PrivacyManager.test.php
index 8e2ac39c7ebf3867ea840a2aa3045bc1ae967197..adca1b4de9904bfe75f1dfbfd55b7914ae042cf3 100755
--- a/plugins/PrivacyManager/tests/PrivacyManager.test.php
+++ b/plugins/PrivacyManager/tests/PrivacyManager.test.php
@@ -41,6 +41,7 @@ class Test_Piwik_PrivacyManager extends Test_Integration
 		parent::setUp();
 		
 		Piwik_TablePartitioning::$tablesAlreadyInstalled = null;
+		Piwik::$lockPrivilegeGranted = null;
 		
 		// purging depends upon today's date, so 'older_than' parts must be dependent upon today
 		$today = Piwik_Date::factory('today');
@@ -78,6 +79,8 @@ class Test_Piwik_PrivacyManager extends Test_Integration
 	{
 		parent::tearDown();
 		
+		Piwik::$lockPrivilegeGranted = null;
+		
 		// remove archive tables (integration test teardown will only truncate)
 		$archiveTables = $this->getArchiveTableNames();
 		$archiveTables = array_merge($archiveTables['numeric'], $archiveTables['blob']);
@@ -528,6 +531,23 @@ class Test_Piwik_PrivacyManager extends Test_Integration
 		$this->checkReportsAndMetricsPurged($janBlobsRemaining = 6); // 1 segmented blob + 5 day blobs
 	}
 	
+	/** Tests that log actions are not purged when the lock privilege is not granted.
+	 * (Simulates absence of lock privilege.)
+	 */
+	public function test_purgeData_deleteLogsWithoutLockPrivilege()
+	{
+		Piwik::$lockPrivilegeGranted = false;
+		
+		$this->addLogData();
+		
+		// purge data
+		$this->setTimeToRun();
+		$this->instance->deleteLogData();
+		
+		// perform checks
+		$this->checkLogDataPurged($actionsPurged = false);
+	}
+	
 	// --- utility functions follow ---
 	
 	private function addLogData()
@@ -717,7 +737,7 @@ class Test_Piwik_PrivacyManager extends Test_Integration
 		$this->assertEqual(self::FEB_METRIC_ARCHIVE_COUNT + 1, $this->getTableCount($archiveTables['blob'][1])); // February
 	}
 	
-	private function checkLogDataPurged()
+	private function checkLogDataPurged( $actionsPurged = true )
 	{
 		// 3 days removed by purge, so 3 visits, 6 conversions, 6 visit actions, 3 e-commerce orders
 		// & 6 actions removed
@@ -725,7 +745,14 @@ class Test_Piwik_PrivacyManager extends Test_Integration
 		$this->assertEqual(16, $this->getTableCount('log_conversion'));
 		$this->assertEqual(16, $this->getTableCount('log_link_visit_action'));
 		$this->assertEqual(8, $this->getTableCount('log_conversion_item'));
-		$this->assertEqual(21, $this->getTableCount('log_action'));
+		if ($actionsPurged)
+		{
+			$this->assertEqual(21, $this->getTableCount('log_action'));
+		}
+		else
+		{
+			$this->assertEqual(27, $this->getTableCount('log_action'));
+		}
 	}
 	
 	/**