PostgreSqlLockManager.php 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485
  1. <?php
  2. use Wikimedia\Rdbms\DBError;
  3. /**
  4. * PostgreSQL version of DBLockManager that supports shared locks.
  5. * All locks are non-blocking, which avoids deadlocks.
  6. *
  7. * @ingroup LockManager
  8. * @phan-file-suppress PhanUndeclaredConstant Phan doesn't read constants in LockManager
  9. * when accessed via self::
  10. */
  11. class PostgreSqlLockManager extends DBLockManager {
  12. /** @var array Mapping of lock types to the type actually used */
  13. protected $lockTypeMap = [
  14. self::LOCK_SH => self::LOCK_SH,
  15. self::LOCK_UW => self::LOCK_SH,
  16. self::LOCK_EX => self::LOCK_EX
  17. ];
  18. protected function doGetLocksOnServer( $lockSrv, array $paths, $type ) {
  19. $status = StatusValue::newGood();
  20. if ( $paths === [] ) {
  21. return $status; // nothing to lock
  22. }
  23. $db = $this->getConnection( $lockSrv ); // checked in isServerUp()
  24. $bigints = array_unique( array_map(
  25. function ( $key ) {
  26. return Wikimedia\base_convert( substr( $key, 0, 15 ), 16, 10 );
  27. },
  28. array_map( [ $this, 'sha1Base16Absolute' ], $paths )
  29. ) );
  30. // Try to acquire all the locks...
  31. $fields = [];
  32. foreach ( $bigints as $bigint ) {
  33. $fields[] = ( $type == self::LOCK_SH )
  34. ? "pg_try_advisory_lock_shared({$db->addQuotes( $bigint )}) AS K$bigint"
  35. : "pg_try_advisory_lock({$db->addQuotes( $bigint )}) AS K$bigint";
  36. }
  37. $res = $db->query( 'SELECT ' . implode( ', ', $fields ), __METHOD__ );
  38. $row = $res->fetchRow();
  39. if ( in_array( 'f', $row ) ) {
  40. // Release any acquired locks if some could not be acquired...
  41. $fields = [];
  42. foreach ( $row as $kbigint => $ok ) {
  43. if ( $ok === 't' ) { // locked
  44. $bigint = substr( $kbigint, 1 ); // strip off the "K"
  45. $fields[] = ( $type == self::LOCK_SH )
  46. ? "pg_advisory_unlock_shared({$db->addQuotes( $bigint )})"
  47. : "pg_advisory_unlock({$db->addQuotes( $bigint )})";
  48. }
  49. }
  50. if ( count( $fields ) ) {
  51. $db->query( 'SELECT ' . implode( ', ', $fields ), __METHOD__ );
  52. }
  53. foreach ( $paths as $path ) {
  54. $status->fatal( 'lockmanager-fail-acquirelock', $path );
  55. }
  56. }
  57. return $status;
  58. }
  59. /**
  60. * @see QuorumLockManager::releaseAllLocks()
  61. * @return StatusValue
  62. */
  63. protected function releaseAllLocks() {
  64. $status = StatusValue::newGood();
  65. foreach ( $this->conns as $lockDb => $db ) {
  66. try {
  67. $db->query( "SELECT pg_advisory_unlock_all()", __METHOD__ );
  68. } catch ( DBError $e ) {
  69. $status->fatal( 'lockmanager-fail-db-release', $lockDb );
  70. }
  71. }
  72. return $status;
  73. }
  74. }