Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
Q
question2answer
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
outils
question2answer
Commits
1c21830c
Commit
1c21830c
authored
Jan 24, 2018
by
Simon Champion
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Refactor the recalc functions into a class.
parent
f9f9d3bb
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
876 additions
and
15 deletions
+876
-15
recalc.php
qa-include/ajax/recalc.php
+6
-4
recalc.php
qa-include/app/recalc.php
+4
-0
RecalcMain.php
qa-include/app/recalc/RecalcMain.php
+853
-0
install.php
qa-include/db/install.php
+7
-6
admin-recalc.php
qa-include/pages/admin/admin-recalc.php
+5
-4
qa-app-recalc.php
qa-include/qa-app-recalc.php
+1
-1
No files found.
qa-include/ajax/recalc.php
View file @
1c21830c
...
@@ -20,7 +20,7 @@
...
@@ -20,7 +20,7 @@
*/
*/
require_once
QA_INCLUDE_DIR
.
'app/users.php'
;
require_once
QA_INCLUDE_DIR
.
'app/users.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc
/RecalcMain
.php'
;
if
(
qa_get_logged_in_level
()
>=
QA_USER_LEVEL_ADMIN
)
{
if
(
qa_get_logged_in_level
()
>=
QA_USER_LEVEL_ADMIN
)
{
...
@@ -29,14 +29,16 @@ if (qa_get_logged_in_level() >= QA_USER_LEVEL_ADMIN) {
...
@@ -29,14 +29,16 @@ if (qa_get_logged_in_level() >= QA_USER_LEVEL_ADMIN) {
$message
=
qa_lang
(
'misc/form_security_reload'
);
$message
=
qa_lang
(
'misc/form_security_reload'
);
}
else
{
}
else
{
$
state
=
qa_post_text
(
'state'
);
$
recalc
=
new
Q2A_App_Recalc_Main
(
qa_post_text
(
'state'
)
);
$stoptime
=
time
()
+
3
;
$stoptime
=
time
()
+
3
;
while
(
qa_recalc_perform_step
(
$state
)
&&
time
()
<
$stoptime
)
{
while
(
$recalc
->
performStep
(
)
&&
time
()
<
$stoptime
)
{
// wait
// wait
}
}
$message
=
qa_recalc_get_message
(
$state
);
$message
=
$recalc
->
getMessage
();
$state
=
$recalc
->
getState
();
}
}
}
else
{
}
else
{
...
...
qa-include/app/recalc.php
View file @
1c21830c
...
@@ -60,6 +60,10 @@ if (!defined('QA_VERSION')) { // don't allow this page to be requested directly
...
@@ -60,6 +60,10 @@ if (!defined('QA_VERSION')) { // don't allow this page to be requested directly
exit
;
exit
;
}
}
if
(
defined
(
'QA_DEBUG_PERFORMANCE'
)
&&
QA_DEBUG_PERFORMANCE
)
{
trigger_error
(
'Included file '
.
basename
(
__FILE__
)
.
' is deprecated'
);
}
require_once
QA_INCLUDE_DIR
.
'db/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'db/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'db/post-create.php'
;
require_once
QA_INCLUDE_DIR
.
'db/post-create.php'
;
require_once
QA_INCLUDE_DIR
.
'db/points.php'
;
require_once
QA_INCLUDE_DIR
.
'db/points.php'
;
...
...
qa-include/app/recalc/RecalcMain.php
0 → 100644
View file @
1c21830c
<?php
/*
Question2Answer by Gideon Greenspan and contributors
http://www.question2answer.org/
File: qa-include/Q2A/Util/Usage.php
Description: Debugging stuff, currently used for tracking resource usage
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
More about this license: http://www.question2answer.org/license.php
*/
/*
A full list of redundant (non-normal) information in the database that can be recalculated:
Recalculated in doreindexcontent:
================================
^titlewords (all): index of words in titles of posts
^contentwords (all): index of words in content of posts
^tagwords (all): index of words in tags of posts (a tag can contain multiple words)
^posttags (all): index tags of posts
^words (all): list of words used for indexes
^options (title=cache_*): cached values for various things (e.g. counting questions)
Recalculated in dorecountposts:
==============================
^posts (upvotes, downvotes, netvotes, hotness, acount, amaxvotes, flagcount): number of votes, hotness, answers, answer votes, flags
Recalculated in dorecalcpoints:
===============================
^userpoints (all except bonus): points calculation for all users
^options (title=cache_userpointscount):
Recalculated in dorecalccategories:
===================================
^posts (categoryid): assign to answers and comments based on their antecedent question
^posts (catidpath1, catidpath2, catidpath3): hierarchical path to category ids (requires QA_CATEGORY_DEPTH=4)
^categories (qcount): number of (visible) questions in each category
^categories (backpath): full (backwards) path of slugs to that category
Recalculated in dorebuildupdates:
=================================
^sharedevents (all): per-entity event streams (see big comment in /qa-include/db/favorites.php)
^userevents (all): per-subscriber event streams
[but these are not entirely redundant since they can contain historical information no longer in ^posts]
*/
if
(
!
defined
(
'QA_VERSION'
))
{
// don't allow this page to be requested directly from browser
header
(
'Location: ../../'
);
exit
;
}
require_once
QA_INCLUDE_DIR
.
'db/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'db/post-create.php'
;
require_once
QA_INCLUDE_DIR
.
'db/points.php'
;
require_once
QA_INCLUDE_DIR
.
'db/selects.php'
;
require_once
QA_INCLUDE_DIR
.
'db/admin.php'
;
require_once
QA_INCLUDE_DIR
.
'db/users.php'
;
require_once
QA_INCLUDE_DIR
.
'app/options.php'
;
require_once
QA_INCLUDE_DIR
.
'app/post-create.php'
;
require_once
QA_INCLUDE_DIR
.
'app/post-update.php'
;
class
Q2A_App_Recalc_Main
{
protected
$state
;
protected
$operation
;
protected
$length
;
protected
$next
;
protected
$done
;
/**
* Initialize the counts of resource usage.
*/
public
function
__construct
(
$state
)
{
$this
->
state
=
$state
;
list
(
$this
->
operation
,
$this
->
length
,
$this
->
next
,
$this
->
done
)
=
explode
(
"
\t
"
,
$state
.
"
\t\t\t\t
"
);
}
public
function
getState
()
{
return
$this
->
state
;
}
public
function
performStep
()
{
if
(
!
method_exists
(
$this
,
$this
->
operation
))
{
$this
->
state
=
''
;
return
false
;
}
$continue
=
$this
->
{
$this
->
operation
}();
if
(
$continue
)
{
$this
->
state
=
$this
->
operation
.
"
\t
"
.
$this
->
length
.
"
\t
"
.
$this
->
next
.
"
\t
"
.
$this
->
done
;
}
return
$continue
&&
$this
->
done
<
$this
->
length
;
}
private
function
DoReindexContent
()
{
$this
->
transition
(
'doreindexcontent_pagereindex'
);
}
private
function
DoReindexContent_PageReindex
()
{
$pages
=
qa_db_pages_get_for_reindexing
(
$this
->
next
,
10
);
if
(
!
count
(
$pages
))
{
$this
->
transition
(
'doreindexcontent_postcount'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/format.php'
;
$lastpageid
=
max
(
array_keys
(
$pages
));
foreach
(
$pages
as
$pageid
=>
$page
)
{
if
(
!
(
$page
[
'flags'
]
&
QA_PAGE_FLAGS_EXTERNAL
))
{
$searchmodules_u
=
qa_load_modules_with
(
'search'
,
'unindex_page'
);
foreach
(
$searchmodules_u
as
$searchmodule
)
{
$searchmodule
->
unindex_page
(
$pageid
);
}
$searchmodules_i
=
qa_load_modules_with
(
'search'
,
'index_page'
);
if
(
count
(
$searchmodules_i
))
{
$indextext
=
qa_viewer_text
(
$page
[
'content'
],
'html'
);
foreach
(
$searchmodules_i
as
$searchmodule
)
{
$searchmodule
->
index_page
(
$pageid
,
$page
[
'tags'
],
$page
[
'heading'
],
$page
[
'content'
],
'html'
,
$indextext
);
}
}
}
}
$this
->
next
=
1
+
$lastpageid
;
$this
->
done
+=
count
(
$pages
);
return
true
;
}
private
function
DoReindexContent_PostCount
()
{
qa_db_qcount_update
();
qa_db_acount_update
();
qa_db_ccount_update
();
$this
->
transition
(
'doreindexcontent_postreindex'
);
return
false
;
}
private
function
DoReindexContent_PostReindex
()
{
$posts
=
qa_db_posts_get_for_reindexing
(
$this
->
next
,
10
);
if
(
!
count
(
$posts
))
{
qa_db_truncate_indexes
(
$this
->
next
);
$this
->
transition
(
'doreindexposts_wordcount'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/format.php'
;
$lastpostid
=
max
(
array_keys
(
$posts
));
qa_db_prepare_for_reindexing
(
$this
->
next
,
$lastpostid
);
qa_suspend_update_counts
();
foreach
(
$posts
as
$postid
=>
$post
)
{
qa_post_unindex
(
$postid
);
qa_post_index
(
$postid
,
$post
[
'type'
],
$post
[
'questionid'
],
$post
[
'parentid'
],
$post
[
'title'
],
$post
[
'content'
],
$post
[
'format'
],
qa_viewer_text
(
$post
[
'content'
],
$post
[
'format'
]),
$post
[
'tags'
],
$post
[
'categoryid'
]);
}
$this
->
next
=
1
+
$lastpostid
;
$this
->
done
+=
count
(
$posts
);
return
true
;
}
private
function
DoReindexPosts_WordCount
()
{
$wordids
=
qa_db_words_prepare_for_recounting
(
$this
->
next
,
1000
);
if
(
!
count
(
$wordids
))
{
qa_db_tagcount_update
();
// this is quick so just do it here
$this
->
transition
(
'doreindexposts_complete'
);
return
false
;
}
$lastwordid
=
max
(
$wordids
);
qa_db_words_recount
(
$this
->
next
,
$lastwordid
);
$this
->
next
=
1
+
$lastwordid
;
$this
->
done
+=
count
(
$wordids
);
return
true
;
}
private
function
DoRecountPosts
()
{
$this
->
transition
(
'dorecountposts_postcount'
);
return
false
;
}
private
function
DoRecountPosts_PostCount
()
{
qa_db_qcount_update
();
qa_db_acount_update
();
qa_db_ccount_update
();
qa_db_unaqcount_update
();
qa_db_unselqcount_update
();
$this
->
transition
(
'dorecountposts_votecount'
);
return
false
;
}
private
function
DoRecountPosts_VoteCount
()
{
$postids
=
qa_db_posts_get_for_recounting
(
$this
->
next
,
1000
);
if
(
!
count
(
$postids
))
{
$this
->
transition
(
'dorecountposts_acount'
);
return
false
;
}
$lastpostid
=
max
(
$postids
);
qa_db_posts_votes_recount
(
$this
->
next
,
$lastpostid
);
$this
->
next
=
1
+
$lastpostid
;
$this
->
done
+=
count
(
$postids
);
return
true
;
}
private
function
DoRecountPosts_Acount
()
{
$postids
=
qa_db_posts_get_for_recounting
(
$this
->
next
,
1000
);
if
(
count
(
$postids
))
{
qa_db_unupaqcount_update
();
$this
->
transition
(
'dorecountposts_complete'
);
return
false
;
}
$lastpostid
=
max
(
$postids
);
qa_db_posts_answers_recount
(
$this
->
next
,
$lastpostid
);
$this
->
next
=
1
+
$lastpostid
;
$this
->
done
+=
count
(
$postids
);
return
true
;
}
private
function
DoRecalcPoints
()
{
$this
->
transition
(
'dorecalcpoints_usercount'
);
return
false
;
}
private
function
DoRecalcPoints_UserCount
()
{
qa_db_userpointscount_update
();
// for progress update - not necessarily accurate
qa_db_uapprovecount_update
();
// needs to be somewhere and this is the most appropriate place
$this
->
transition
(
'dorecalcpoints_recalc'
);
return
false
;
}
private
function
DoRecalcPoints_Recalc
()
{
$default_recalccount
=
10
;
$userids
=
qa_db_users_get_for_recalc_points
(
$this
->
next
,
$default_recalccount
+
1
);
// get one extra so we know where to start from next
$gotcount
=
count
(
$userids
);
$recalccount
=
min
(
$default_recalccount
,
$gotcount
);
// can't recalc more than we got
if
(
$recalccount
>
0
)
{
$lastuserid
=
$userids
[
$recalccount
-
1
];
qa_db_users_recalc_points
(
$this
->
next
,
$lastuserid
);
$this
->
done
+=
$recalccount
;
}
else
{
$lastuserid
=
$this
->
next
;
// for truncation
}
if
(
$gotcount
>
$recalccount
)
{
// more left to do
$this
->
next
=
$userids
[
$recalccount
];
// start next round at first one not recalculated
return
true
;
}
else
{
qa_db_truncate_userpoints
(
$lastuserid
);
qa_db_userpointscount_update
();
// quick so just do it here
$this
->
transition
(
'dorecalcpoints_complete'
);
return
false
;
}
}
private
function
DoRefillEvents
()
{
$this
->
transition
(
'dorefillevents_qcount'
);
return
false
;
}
private
function
DoRefillEvents_Qcount
()
{
qa_db_qcount_update
();
$this
->
transition
(
'dorefillevents_refill'
);
return
false
;
}
private
function
DoRefillEvents_Refill
()
{
$questionids
=
qa_db_qs_get_for_event_refilling
(
$this
->
next
,
1
);
if
(
!
count
(
$questionids
))
{
$this
->
transition
(
'dorefillevents_complete'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/events.php'
;
require_once
QA_INCLUDE_DIR
.
'app/updates.php'
;
require_once
QA_INCLUDE_DIR
.
'util/sort.php'
;
$lastquestionid
=
max
(
$questionids
);
foreach
(
$questionids
as
$questionid
)
{
// Retrieve all posts relating to this question
list
(
$question
,
$childposts
,
$achildposts
)
=
qa_db_select_with_pending
(
qa_db_full_post_selectspec
(
null
,
$questionid
),
qa_db_full_child_posts_selectspec
(
null
,
$questionid
),
qa_db_full_a_child_posts_selectspec
(
null
,
$questionid
)
);
// Merge all posts while preserving keys as postids
$posts
=
array
(
$questionid
=>
$question
);
foreach
(
$childposts
as
$postid
=>
$post
)
{
$posts
[
$postid
]
=
$post
;
}
foreach
(
$achildposts
as
$postid
=>
$post
)
{
$posts
[
$postid
]
=
$post
;
}
// Creation and editing of each post
foreach
(
$posts
as
$postid
=>
$post
)
{
$followonq
=
(
$post
[
'basetype'
]
==
'Q'
)
&&
(
$postid
!=
$questionid
);
if
(
$followonq
)
{
$updatetype
=
QA_UPDATE_FOLLOWS
;
}
elseif
(
$post
[
'basetype'
]
==
'C'
&&
@
$posts
[
$post
[
'parentid'
]][
'basetype'
]
==
'Q'
)
{
$updatetype
=
QA_UPDATE_C_FOR_Q
;
}
elseif
(
$post
[
'basetype'
]
==
'C'
&&
@
$posts
[
$post
[
'parentid'
]][
'basetype'
]
==
'A'
)
{
$updatetype
=
QA_UPDATE_C_FOR_A
;
}
else
{
$updatetype
=
null
;
}
qa_create_event_for_q_user
(
$questionid
,
$postid
,
$updatetype
,
$post
[
'userid'
],
@
$posts
[
$post
[
'parentid'
]][
'userid'
],
$post
[
'created'
]);
if
(
isset
(
$post
[
'updated'
])
&&
!
$followonq
)
{
qa_create_event_for_q_user
(
$questionid
,
$postid
,
$post
[
'updatetype'
],
$post
[
'lastuserid'
],
$post
[
'userid'
],
$post
[
'updated'
]);
}
}
// Tags and categories of question
qa_create_event_for_tags
(
$question
[
'tags'
],
$questionid
,
null
,
$question
[
'userid'
],
$question
[
'created'
]);
qa_create_event_for_category
(
$question
[
'categoryid'
],
$questionid
,
null
,
$question
[
'userid'
],
$question
[
'created'
]);
// Collect comment threads
$parentidcomments
=
array
();
foreach
(
$posts
as
$postid
=>
$post
)
{
if
(
$post
[
'basetype'
]
==
'C'
)
{
$parentidcomments
[
$post
[
'parentid'
]][
$postid
]
=
$post
;
}
}
// For each comment thread, notify all previous comment authors of each comment in the thread (could get slow)
foreach
(
$parentidcomments
as
$parentid
=>
$comments
)
{
$keyuserids
=
array
();
qa_sort_by
(
$comments
,
'created'
);
foreach
(
$comments
as
$comment
)
{
foreach
(
$keyuserids
as
$keyuserid
=>
$dummy
)
{
if
(
$keyuserid
!=
$comment
[
'userid'
]
&&
$keyuserid
!=
@
$posts
[
$parentid
][
'userid'
])
{
qa_db_event_create_not_entity
(
$keyuserid
,
$questionid
,
$comment
[
'postid'
],
QA_UPDATE_FOLLOWS
,
$comment
[
'userid'
],
$comment
[
'created'
]);
}
}
if
(
isset
(
$comment
[
'userid'
]))
{
$keyuserids
[
$comment
[
'userid'
]]
=
true
;
}
}
}
}
$this
->
next
=
1
+
$lastquestionid
;
$this
->
done
+=
count
(
$questionids
);
return
true
;
}
private
function
DoRecalcCategories
()
{
$this
->
transition
(
'dorecalccategories_postcount'
);
return
false
;
}
private
function
DoRecalcCategories_PostCount
()
{
qa_db_acount_update
();
qa_db_ccount_update
();
$this
->
transition
(
'dorecalccategories_postupdate'
);
return
false
;
}
private
function
DoRecalcCategories_PostUpdate
()
{
$postids
=
qa_db_posts_get_for_recategorizing
(
$this
->
next
,
100
);
if
(
!
count
(
$postids
))
{
$this
->
transition
(
'dorecalccategories_recount'
);
return
false
;
}
$lastpostid
=
max
(
$postids
);
qa_db_posts_recalc_categoryid
(
$this
->
next
,
$lastpostid
);
qa_db_posts_calc_category_path
(
$this
->
next
,
$lastpostid
);
$this
->
next
=
1
+
$lastpostid
;
$this
->
done
+=
count
(
$postids
);
return
true
;
}
private
function
DoRecalcCategories_Recount
()
{
$categoryids
=
qa_db_categories_get_for_recalcs
(
$this
->
next
,
10
);
if
(
!
count
(
$categoryids
))
{
$this
->
transition
(
'dorecalccategories_backpaths'
);
return
false
;
}
$lastcategoryid
=
max
(
$categoryids
);
foreach
(
$categoryids
as
$categoryid
)
{
qa_db_ifcategory_qcount_update
(
$categoryid
);
}
$this
->
next
=
1
+
$lastcategoryid
;
$this
->
done
+=
count
(
$categoryids
);
return
true
;
}
private
function
DoRecalcCategories_BackPaths
()
{
$categoryids
=
qa_db_categories_get_for_recalcs
(
$this
->
next
,
10
);
if
(
!
count
(
$categoryids
))
{
$this
->
transition
(
'dorecalccategories_complete'
);
return
false
;
}
$lastcategoryid
=
max
(
$categoryids
);
qa_db_categories_recalc_backpaths
(
$this
->
next
,
$lastcategoryid
);
$this
->
next
=
1
+
$lastcategoryid
;
$this
->
done
+=
count
(
$categoryids
);
return
true
;
}
private
function
DoDeleteHidden
()
{
$this
->
transition
(
'dodeletehidden_comments'
);
return
false
;
}
private
function
DoDeleteHidden_Comments
()
{
$posts
=
qa_db_posts_get_for_deleting
(
'C'
,
$this
->
next
,
1
);
if
(
!
count
(
$posts
))
{
$this
->
transition
(
'dodeletehidden_answers'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/posts.php'
;
$postid
=
$posts
[
0
];
qa_post_delete
(
$postid
);
$this
->
next
=
1
+
$postid
;
$this
->
done
++
;
return
true
;
}
private
function
DoDeleteHidden_Answers
()
{
$posts
=
qa_db_posts_get_for_deleting
(
'A'
,
$this
->
next
,
1
);
if
(
!
count
(
$posts
))
{
$this
->
transition
(
'dodeletehidden_questions'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/posts.php'
;
$postid
=
$posts
[
0
];
qa_post_delete
(
$postid
);
$this
->
next
=
1
+
$postid
;
$this
->
done
++
;
return
true
;
}
private
function
dodeletehidden_questions
()
{
$posts
=
qa_db_posts_get_for_deleting
(
'Q'
,
$this
->
next
,
1
);
if
(
!
count
(
$posts
))
{
$this
->
transition
(
'dodeletehidden_complete'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/posts.php'
;
$postid
=
$posts
[
0
];
qa_post_delete
(
$postid
);
$this
->
next
=
1
+
$postid
;
$this
->
done
++
;
return
true
;
}
private
function
DoBlobsToDisk
()
{
$this
->
transition
(
'doblobstodisk_move'
);
return
false
;
}
private
function
DoBlobsToDisk_Move
()
{
$blob
=
qa_db_get_next_blob_in_db
(
$this
->
next
);
if
(
!
isset
(
$blob
))
{
$this
->
transition
(
'doblobstodisk_complete'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/blobs.php'
;
require_once
QA_INCLUDE_DIR
.
'db/blobs.php'
;
if
(
qa_write_blob_file
(
$blob
[
'blobid'
],
$blob
[
'content'
],
$blob
[
'format'
]))
{
qa_db_blob_set_content
(
$blob
[
'blobid'
],
null
);
}
$this
->
next
=
1
+
$blob
[
'blobid'
];
$this
->
done
++
;
return
true
;
}
private
function
DoBlobsToDB
()
{
$this
->
transition
(
'doblobstodb_move'
);
return
false
;
}
private
function
DoBlobsToDB_Move
()
{
$blob
=
qa_db_get_next_blob_on_disk
(
$this
->
next
);
if
(
!
isset
(
$blob
))
{
$this
->
transition
(
'doblobstodb_complete'
);
return
false
;
}
require_once
QA_INCLUDE_DIR
.
'app/blobs.php'
;
require_once
QA_INCLUDE_DIR
.
'db/blobs.php'
;
$content
=
qa_read_blob_file
(
$blob
[
'blobid'
],
$blob
[
'format'
]);
qa_db_blob_set_content
(
$blob
[
'blobid'
],
$content
);
qa_delete_blob_file
(
$blob
[
'blobid'
],
$blob
[
'format'
]);
$this
->
next
=
1
+
$blob
[
'blobid'
];
$this
->
done
++
;
return
true
;
}
private
function
DoCacheTrim
()
{
$this
->
transition
(
'docachetrim_process'
);
return
false
;
}
private
function
DoCacheClear
()
{
$this
->
transition
(
'docacheclear_process'
);
return
false
;
}
private
function
DoCacheTrim_Process
()
{
return
$this
->
docacheclear_process
();
}
private
function
DoCacheClear_Process
()
{
$cacheDriver
=
Q2A_Storage_CacheFactory
::
getCacheDriver
();
$cacheStats
=
$cacheDriver
->
getStats
();
$limit
=
min
(
$cacheStats
[
'files'
],
20
);
if
(
!
(
$cacheStats
[
'files'
]
>
0
&&
$this
->
next
<=
$this
->
length
))
{
$this
->
transition
(
'docacheclear_complete'
);
return
false
;
}
$deleted
=
$cacheDriver
->
clear
(
$limit
,
$this
->
next
,
(
$this
->
operation
===
'docachetrim_process'
));
$this
->
done
+=
$deleted
;
$this
->
next
+=
$limit
-
$deleted
;
// skip files that weren't deleted on next iteration
return
true
;
}
/**
* Change the $state to represent the beginning of a new $operation
* @param $newOoperation
*/
public
function
transition
(
$newOperation
)
{
$this
->
operation
=
$newOperation
;
$this
->
length
=
$this
->
stageLength
();
$this
->
next
=
(
QA_FINAL_EXTERNAL_USERS
&&
(
$newOperation
==
'dorecalcpoints_recalc'
))
?
''
:
0
;
$this
->
done
=
0
;
$this
->
state
=
$newOperation
.
"
\t
"
.
$this
->
length
.
"
\t
"
.
$this
->
next
.
"
\t
"
.
$this
->
done
;
}
/**
* Return how many steps there will be in recalculation $operation
* @return int
*/
private
function
stageLength
()
{
switch
(
$this
->
operation
)
{
case
'doreindexcontent_pagereindex'
:
return
qa_db_count_pages
();
case
'doreindexcontent_postreindex'
:
return
qa_opt
(
'cache_qcount'
)
+
qa_opt
(
'cache_acount'
)
+
qa_opt
(
'cache_ccount'
);
case
'doreindexposts_wordcount'
:
return
qa_db_count_words
();
case
'dorecalcpoints_recalc'
:
return
qa_opt
(
'cache_userpointscount'
);
case
'dorecountposts_votecount'
:
case
'dorecountposts_acount'
:
case
'dorecalccategories_postupdate'
:
return
qa_db_count_posts
();
case
'dorefillevents_refill'
:
return
qa_opt
(
'cache_qcount'
)
+
qa_db_count_posts
(
'Q_HIDDEN'
);
case
'dorecalccategories_recount'
:
case
'dorecalccategories_backpaths'
:
return
qa_db_count_categories
();
case
'dodeletehidden_comments'
:
return
count
(
qa_db_posts_get_for_deleting
(
'C'
));
case
'dodeletehidden_answers'
:
return
count
(
qa_db_posts_get_for_deleting
(
'A'
));
case
'dodeletehidden_questions'
:
return
count
(
qa_db_posts_get_for_deleting
(
'Q'
));
case
'doblobstodisk_move'
:
return
qa_db_count_blobs_in_db
();
case
'doblobstodb_move'
:
return
qa_db_count_blobs_on_disk
();
case
'docachetrim_process'
:
case
'docacheclear_process'
:
$cacheDriver
=
Q2A_Storage_CacheFactory
::
getCacheDriver
();
$cacheStats
=
$cacheDriver
->
getStats
();
return
$cacheStats
[
'files'
];
}
return
0
;
}
/**
* Return the translated language ID string replacing the progress and total in it.
* @access private
* @param string $langId Language string ID that contains 2 placeholders (^1 and ^2)
* @param int $progress Amount of processed elements
* @param int $total Total amount of elements
*
* @return string Returns the language string ID with their placeholders replaced with
* the formatted progress and total numbers
*/
private
function
progressLang
(
$langId
,
$progress
,
$total
)
{
return
strtr
(
qa_lang
(
$langId
),
array
(
'^1'
=>
qa_format_number
(
$progress
),
'^2'
=>
qa_format_number
(
$total
),
));
}
/**
* Return a string which gives a user-viewable version of $state
* @return string
*/
public
function
getMessage
()
{
require_once
QA_INCLUDE_DIR
.
'app/format.php'
;
$this
->
done
=
(
int
)
$this
->
done
;
$this
->
length
=
(
int
)
$this
->
length
;
switch
(
$this
->
operation
)
{
case
'doreindexcontent_postcount'
:
case
'dorecountposts_postcount'
:
case
'dorecalccategories_postcount'
:
case
'dorefillevents_qcount'
:
$message
=
qa_lang
(
'admin/recalc_posts_count'
);
break
;
case
'doreindexcontent_pagereindex'
:
$message
=
$this
->
progressLang
(
'admin/reindex_pages_reindexed'
,
$this
->
done
,
$this
->
length
);
break
;
case
'doreindexcontent_postreindex'
:
$message
=
$this
->
progressLang
(
'admin/reindex_posts_reindexed'
,
$this
->
done
,
$this
->
length
);
break
;
case
'doreindexposts_complete'
:
$message
=
qa_lang
(
'admin/reindex_posts_complete'
);
break
;
case
'doreindexposts_wordcount'
:
$message
=
$this
->
progressLang
(
'admin/reindex_posts_wordcounted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecountposts_votecount'
:
$message
=
$this
->
progressLang
(
'admin/recount_posts_votes_recounted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecountposts_acount'
:
$message
=
$this
->
progressLang
(
'admin/recount_posts_as_recounted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecountposts_complete'
:
$message
=
qa_lang
(
'admin/recount_posts_complete'
);
break
;
case
'dorecalcpoints_usercount'
:
$message
=
qa_lang
(
'admin/recalc_points_usercount'
);
break
;
case
'dorecalcpoints_recalc'
:
$message
=
$this
->
progressLang
(
'admin/recalc_points_recalced'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecalcpoints_complete'
:
$message
=
qa_lang
(
'admin/recalc_points_complete'
);
break
;
case
'dorefillevents_refill'
:
$message
=
$this
->
progressLang
(
'admin/refill_events_refilled'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorefillevents_complete'
:
$message
=
qa_lang
(
'admin/refill_events_complete'
);
break
;
case
'dorecalccategories_postupdate'
:
$message
=
$this
->
progressLang
(
'admin/recalc_categories_updated'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecalccategories_recount'
:
$message
=
$this
->
progressLang
(
'admin/recalc_categories_recounting'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecalccategories_backpaths'
:
$message
=
$this
->
progressLang
(
'admin/recalc_categories_backpaths'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dorecalccategories_complete'
:
$message
=
qa_lang
(
'admin/recalc_categories_complete'
);
break
;
case
'dodeletehidden_comments'
:
$message
=
$this
->
progressLang
(
'admin/hidden_comments_deleted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dodeletehidden_answers'
:
$message
=
$this
->
progressLang
(
'admin/hidden_answers_deleted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dodeletehidden_questions'
:
$message
=
$this
->
progressLang
(
'admin/hidden_questions_deleted'
,
$this
->
done
,
$this
->
length
);
break
;
case
'dodeletehidden_complete'
:
$message
=
qa_lang
(
'admin/delete_hidden_complete'
);
break
;
case
'doblobstodisk_move'
:
case
'doblobstodb_move'
:
$message
=
$this
->
progressLang
(
'admin/blobs_move_moved'
,
$this
->
done
,
$this
->
length
);
break
;
case
'doblobstodisk_complete'
:
case
'doblobstodb_complete'
:
$message
=
qa_lang
(
'admin/blobs_move_complete'
);
break
;
case
'docachetrim_process'
:
case
'docacheclear_process'
:
$message
=
$this
->
progressLang
(
'admin/caching_delete_progress'
,
$this
->
done
,
$this
->
length
);
break
;
case
'docacheclear_complete'
:
$message
=
qa_lang
(
'admin/caching_delete_complete'
);
break
;
default
:
$message
=
''
;
break
;
}
return
$message
;
}
}
qa-include/db/install.php
View file @
1c21830c
...
@@ -78,7 +78,7 @@ function qa_db_table_definitions()
...
@@ -78,7 +78,7 @@ function qa_db_table_definitions()
* In MySQL versions prior to 5.0.3, VARCHAR(x) columns will be silently converted to TEXT where x>255
* In MySQL versions prior to 5.0.3, VARCHAR(x) columns will be silently converted to TEXT where x>255
* See box at top of /qa-include/app/recalc.php for a list of redundant (non-normal) information in the database
* See box at top of /qa-include/app/recalc
/RecalcMain
.php for a list of redundant (non-normal) information in the database
* Starting in version 1.2, we explicitly name keys and foreign key constraints, instead of allowing MySQL
* Starting in version 1.2, we explicitly name keys and foreign key constraints, instead of allowing MySQL
to name these by default. Our chosen names match the default names that MySQL would have assigned, and
to name these by default. Our chosen names match the default names that MySQL would have assigned, and
...
@@ -769,7 +769,7 @@ function qa_db_default_userfields_sql()
...
@@ -769,7 +769,7 @@ function qa_db_default_userfields_sql()
*/
*/
function
qa_db_upgrade_tables
()
function
qa_db_upgrade_tables
()
{
{
require_once
QA_INCLUDE_DIR
.
'app/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc
/RecalcMain
.php'
;
$definitions
=
qa_db_table_definitions
();
$definitions
=
qa_db_table_definitions
();
$keyrecalc
=
array
();
$keyrecalc
=
array
();
...
@@ -1611,16 +1611,17 @@ function qa_db_upgrade_tables()
...
@@ -1611,16 +1611,17 @@ function qa_db_upgrade_tables()
// Perform any necessary recalculations, as determined by upgrade steps
// Perform any necessary recalculations, as determined by upgrade steps
foreach
(
$keyrecalc
as
$state
=>
$dummy
)
{
foreach
(
array_keys
(
$keyrecalc
)
as
$state
)
{
while
(
$state
)
{
$recalc
=
new
Q2A_App_Recalc_Main
(
$state
);
while
(
$recalc
->
getState
())
{
set_time_limit
(
60
);
set_time_limit
(
60
);
$stoptime
=
time
()
+
2
;
$stoptime
=
time
()
+
2
;
while
(
qa_recalc_perform_step
(
$state
)
&&
(
time
()
<
$stoptime
))
while
(
$recalc
->
performStep
(
)
&&
(
time
()
<
$stoptime
))
;
;
qa_db_upgrade_progress
(
qa_recalc_get_message
(
$state
));
qa_db_upgrade_progress
(
$recalc
->
getMmessage
(
));
}
}
}
}
}
}
...
...
qa-include/pages/admin/admin-recalc.php
View file @
1c21830c
...
@@ -25,7 +25,7 @@ if (!defined('QA_VERSION')) { // don't allow this page to be requested directly
...
@@ -25,7 +25,7 @@ if (!defined('QA_VERSION')) { // don't allow this page to be requested directly
}
}
require_once
QA_INCLUDE_DIR
.
'app/admin.php'
;
require_once
QA_INCLUDE_DIR
.
'app/admin.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc
/RecalcMain
.php'
;
// Check we have administrative privileges
// Check we have administrative privileges
...
@@ -71,15 +71,16 @@ if ($recalcnow) {
...
@@ -71,15 +71,16 @@ if ($recalcnow) {
<?php
<?php
while
(
$state
)
{
$recalc
=
new
Q2A_App_Recalc_Main
(
$state
);
while
(
$recalc
->
getState
())
{
set_time_limit
(
60
);
set_time_limit
(
60
);
$stoptime
=
time
()
+
2
;
// run in lumps of two seconds...
$stoptime
=
time
()
+
2
;
// run in lumps of two seconds...
while
(
qa_recalc_perform_step
(
$state
)
&&
time
()
<
$stoptime
)
while
(
$recalc
->
performStep
(
)
&&
time
()
<
$stoptime
)
;
;
echo
qa_html
(
qa_recalc_get_message
(
$state
)
)
.
str_repeat
(
' '
,
1024
)
.
"<br>
\n
"
;
echo
qa_html
(
$recalc
->
getMessage
)
.
str_repeat
(
' '
,
1024
)
.
"<br>
\n
"
;
flush
();
flush
();
sleep
(
1
);
// ... then rest for one
sleep
(
1
);
// ... then rest for one
...
...
qa-include/qa-app-recalc.php
View file @
1c21830c
...
@@ -12,4 +12,4 @@ if (defined('QA_DEBUG_PERFORMANCE') && QA_DEBUG_PERFORMANCE) {
...
@@ -12,4 +12,4 @@ if (defined('QA_DEBUG_PERFORMANCE') && QA_DEBUG_PERFORMANCE) {
trigger_error
(
'Included file '
.
basename
(
__FILE__
)
.
' is deprecated'
);
trigger_error
(
'Included file '
.
basename
(
__FILE__
)
.
' is deprecated'
);
}
}
require_once
QA_INCLUDE_DIR
.
'app/recalc.php'
;
require_once
QA_INCLUDE_DIR
.
'app/recalc
/RecalcMain
.php'
;
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment