mirror of
https://github.com/phpbb/phpbb.git
synced 2025-01-16 21:58:17 +01:00
Merge branch '3.3.x'
This commit is contained in:
commit
222ca643b5
@ -65,6 +65,13 @@ services:
|
|||||||
- '@controller.helper'
|
- '@controller.helper'
|
||||||
- '@event_dispatcher'
|
- '@event_dispatcher'
|
||||||
|
|
||||||
|
posting.lock:
|
||||||
|
class: phpbb\lock\posting
|
||||||
|
shared: false
|
||||||
|
arguments:
|
||||||
|
- '@cache.driver'
|
||||||
|
- '@config'
|
||||||
|
|
||||||
viewonline_helper:
|
viewonline_helper:
|
||||||
class: phpbb\viewonline_helper
|
class: phpbb\viewonline_helper
|
||||||
arguments:
|
arguments:
|
||||||
|
77
phpBB/phpbb/lock/posting.php
Normal file
77
phpBB/phpbb/lock/posting.php
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* This file is part of the phpBB Forum Software package.
|
||||||
|
*
|
||||||
|
* @copyright (c) phpBB Limited <https://www.phpbb.com>
|
||||||
|
* @license GNU General Public License, version 2 (GPL-2.0)
|
||||||
|
*
|
||||||
|
* For full copyright and license information, please see
|
||||||
|
* the docs/CREDITS.txt file.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace phpbb\lock;
|
||||||
|
|
||||||
|
use phpbb\cache\driver\driver_interface as cache_interface;
|
||||||
|
use phpbb\config\config;
|
||||||
|
|
||||||
|
class posting
|
||||||
|
{
|
||||||
|
/** @var cache_interface */
|
||||||
|
private $cache;
|
||||||
|
|
||||||
|
/** @var config */
|
||||||
|
private $config;
|
||||||
|
|
||||||
|
/** @var string */
|
||||||
|
private $lock_name = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor for posting lock
|
||||||
|
*
|
||||||
|
* @param cache_interface $cache
|
||||||
|
* @param config $config
|
||||||
|
*/
|
||||||
|
public function __construct(cache_interface $cache, config $config)
|
||||||
|
{
|
||||||
|
$this->cache = $cache;
|
||||||
|
$this->config = $config;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set lock name
|
||||||
|
*
|
||||||
|
* @param int $creation_time Creation time of form, must be checked already
|
||||||
|
* @param string $form_token Form token used for form, must be checked already
|
||||||
|
*
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function set_lock_name(int $creation_time, string $form_token): void
|
||||||
|
{
|
||||||
|
$this->lock_name = sha1(((string) $creation_time) . $form_token) . '_posting_lock';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Acquire lock for current posting form submission
|
||||||
|
*
|
||||||
|
* @param int $creation_time Creation time of form, must be checked already
|
||||||
|
* @param string $form_token Form token used for form, must be checked already
|
||||||
|
*
|
||||||
|
* @return bool True if lock could be acquired, false if not
|
||||||
|
*/
|
||||||
|
public function acquire(int $creation_time, string $form_token): bool
|
||||||
|
{
|
||||||
|
$this->set_lock_name($creation_time, $form_token);
|
||||||
|
|
||||||
|
// Lock is held for session, cannot acquire it unless special flag for testing is set
|
||||||
|
if ($this->cache->_exists($this->lock_name) && !$this->config->offsetExists('ci_tests_no_lock_posting'))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->cache->put($this->lock_name, true, $this->config['flood_interval']);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
@ -1427,7 +1427,14 @@ if ($submit || $preview || $refresh)
|
|||||||
// Store message, sync counters
|
// Store message, sync counters
|
||||||
if (!count($error) && $submit)
|
if (!count($error) && $submit)
|
||||||
{
|
{
|
||||||
if ($submit)
|
/** @var \phpbb\lock\posting $posting_lock */
|
||||||
|
$posting_lock = $phpbb_container->get('posting.lock');
|
||||||
|
|
||||||
|
// Get creation time and form token, must be already checked at this point
|
||||||
|
$creation_time = abs($request->variable('creation_time', 0));
|
||||||
|
$form_token = $request->variable('form_token', '');
|
||||||
|
|
||||||
|
if ($posting_lock->acquire($creation_time, $form_token))
|
||||||
{
|
{
|
||||||
// Lock/Unlock Topic
|
// Lock/Unlock Topic
|
||||||
$change_topic_status = $post_data['topic_status'];
|
$change_topic_status = $post_data['topic_status'];
|
||||||
@ -1618,6 +1625,11 @@ if ($submit || $preview || $refresh)
|
|||||||
|
|
||||||
redirect($redirect_url);
|
redirect($redirect_url);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Posting was already locked before, hence form submission was already attempted once and is now invalid
|
||||||
|
$error[] = $language->lang('FORM_INVALID');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -337,6 +337,29 @@ $('[data-ajax]').each(function() {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Prevent accidental double submission of form
|
||||||
|
$('[data-prevent-flood] input[type=submit]').click(function(event) {
|
||||||
|
const $submitButton = $(this); // Store the button element
|
||||||
|
const $form = $submitButton.closest('form');
|
||||||
|
|
||||||
|
// Always add the disabled class for visual feedback
|
||||||
|
$submitButton.addClass('disabled');
|
||||||
|
|
||||||
|
// Submit form if it hasn't been submitted yet
|
||||||
|
if (!$form.prop('data-form-submitted')) {
|
||||||
|
$form.prop('data-form-submitted', true);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prevent default submission for subsequent clicks within 5 seconds
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
$form.prop('removeProp', 'data-form-submitted');
|
||||||
|
$submitButton.removeClass('disabled'); // Re-enable after 5 seconds
|
||||||
|
}, 5000);
|
||||||
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This simply appends #preview to the action of the
|
* This simply appends #preview to the action of the
|
||||||
|
@ -97,7 +97,7 @@
|
|||||||
<!-- IF not S_SHOW_DRAFTS and not $SIG_EDIT eq 1 -->
|
<!-- IF not S_SHOW_DRAFTS and not $SIG_EDIT eq 1 -->
|
||||||
<div class="panel bg2">
|
<div class="panel bg2">
|
||||||
<div class="inner">
|
<div class="inner">
|
||||||
<fieldset class="submit-buttons">
|
<fieldset class="submit-buttons" data-prevent-flood>
|
||||||
{S_HIDDEN_ADDRESS_FIELD}
|
{S_HIDDEN_ADDRESS_FIELD}
|
||||||
{S_HIDDEN_FIELDS}
|
{S_HIDDEN_FIELDS}
|
||||||
<!-- EVENT posting_editor_submit_buttons -->
|
<!-- EVENT posting_editor_submit_buttons -->
|
||||||
|
56
tests/lock/posting_test.php
Normal file
56
tests/lock/posting_test.php
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* This file is part of the phpBB Forum Software package.
|
||||||
|
*
|
||||||
|
* @copyright (c) phpBB Limited <https://www.phpbb.com>
|
||||||
|
* @license GNU General Public License, version 2 (GPL-2.0)
|
||||||
|
*
|
||||||
|
* For full copyright and license information, please see
|
||||||
|
* the docs/CREDITS.txt file.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
use phpbb\cache\driver\file as file_cache;
|
||||||
|
use phpbb\config\config;
|
||||||
|
use phpbb\lock\posting;
|
||||||
|
|
||||||
|
class phpbb_lock_posting_test extends phpbb_test_case
|
||||||
|
{
|
||||||
|
/** @var \phpbb\cache\driver\file */
|
||||||
|
protected $cache;
|
||||||
|
|
||||||
|
/** @var config */
|
||||||
|
protected $config;
|
||||||
|
|
||||||
|
/** @var posting */
|
||||||
|
protected $lock;
|
||||||
|
|
||||||
|
public function setUp(): void
|
||||||
|
{
|
||||||
|
$this->cache = new file_cache(__DIR__ . '/../tmp/');
|
||||||
|
$this->cache->purge(); // ensure cache is clean
|
||||||
|
$this->config = new config([
|
||||||
|
'flood_interval' => 15,
|
||||||
|
]);
|
||||||
|
$this->lock = new posting($this->cache, $this->config);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_lock_acquire()
|
||||||
|
{
|
||||||
|
$this->assertTrue($this->lock->acquire(100, 'foo'));
|
||||||
|
$this->assertFalse($this->lock->acquire(100, 'foo'));
|
||||||
|
|
||||||
|
$this->assertTrue($this->cache->_exists(sha1('100foo') . '_posting_lock'));
|
||||||
|
$this->assertFalse($this->lock->acquire(100, 'foo'));
|
||||||
|
$this->cache->put(sha1('100foo') . '_posting_lock', 'foo', -30);
|
||||||
|
|
||||||
|
$this->assertTrue($this->lock->acquire(100, 'foo'));
|
||||||
|
$this->assertTrue($this->cache->_exists(sha1('100foo') . '_posting_lock'));
|
||||||
|
$this->config->offsetSet('ci_tests_no_lock_posting', true);
|
||||||
|
$this->assertTrue($this->lock->acquire(100, 'foo'));
|
||||||
|
$this->assertTrue($this->cache->_exists(sha1('100foo') . '_posting_lock'));
|
||||||
|
// Multiple acquires possible due to special ci test flag
|
||||||
|
$this->assertTrue($this->lock->acquire(100, 'foo'));
|
||||||
|
}
|
||||||
|
}
|
@ -106,6 +106,14 @@ class phpbb_functional_test_case extends phpbb_test_case
|
|||||||
|
|
||||||
$db = $this->get_db();
|
$db = $this->get_db();
|
||||||
|
|
||||||
|
// Special flag for testing without possibility to run into lock scenario.
|
||||||
|
// Unset entry and add it back if lock behavior for posting should be tested.
|
||||||
|
// Unset ci_tests_no_lock_posting from config
|
||||||
|
$db->sql_return_on_error(true);
|
||||||
|
$sql = 'INSERT INTO ' . CONFIG_TABLE . " (config_name, config_value) VALUES ('ci_tests_no_lock_posting', '1')";
|
||||||
|
$this->db->sql_query($sql);
|
||||||
|
$db->sql_return_on_error(false);
|
||||||
|
|
||||||
foreach (static::setup_extensions() as $extension)
|
foreach (static::setup_extensions() as $extension)
|
||||||
{
|
{
|
||||||
$this->purge_cache();
|
$this->purge_cache();
|
||||||
@ -130,6 +138,11 @@ class phpbb_functional_test_case extends phpbb_test_case
|
|||||||
|
|
||||||
if ($this->db instanceof \phpbb\db\driver\driver_interface)
|
if ($this->db instanceof \phpbb\db\driver\driver_interface)
|
||||||
{
|
{
|
||||||
|
// Unset ci_tests_no_lock_posting from config
|
||||||
|
$sql = 'DELETE FROM ' . CONFIG_TABLE . "
|
||||||
|
WHERE config_name = 'ci_tests_no_lock_posting'";
|
||||||
|
$this->db->sql_query($sql);
|
||||||
|
|
||||||
// Close the database connections again this test
|
// Close the database connections again this test
|
||||||
$this->db->sql_close();
|
$this->db->sql_close();
|
||||||
}
|
}
|
||||||
@ -212,6 +225,9 @@ class phpbb_functional_test_case extends phpbb_test_case
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return \phpbb\db\driver\driver_interface
|
||||||
|
*/
|
||||||
protected function get_db()
|
protected function get_db()
|
||||||
{
|
{
|
||||||
global $phpbb_root_path, $phpEx;
|
global $phpbb_root_path, $phpEx;
|
||||||
|
@ -136,7 +136,6 @@ class phpbb_test_case_helpers
|
|||||||
{
|
{
|
||||||
$config = array();
|
$config = array();
|
||||||
|
|
||||||
|
|
||||||
if (extension_loaded('sqlite3'))
|
if (extension_loaded('sqlite3'))
|
||||||
{
|
{
|
||||||
$config = array_merge($config, array(
|
$config = array_merge($config, array(
|
||||||
|
Loading…
x
Reference in New Issue
Block a user