How to troubleshoot and optimize the Full Reindex in Jira Data Center

お困りですか?

アトラシアン コミュニティをご利用ください。

コミュニティに質問

プラットフォームについて: Data Center - この記事は、Data Center プラットフォームのアトラシアン製品に適用されます。

このナレッジベース記事は製品の Data Center バージョン用に作成されています。Data Center 固有ではない機能の Data Center ナレッジベースは、製品のサーバー バージョンでも動作する可能性はありますが、テストは行われていません。サーバー*製品のサポートは 2024 年 2 月 15 日に終了しました。サーバー製品を利用している場合は、アトラシアンのサーバー製品のサポート終了のお知らせページにて移行オプションをご確認ください。

*Fisheye および Crucible は除く

 

要約

This article is meant to be a follow-up to How to increase the speed of full reindex in Jira Server and Data Center.

It offers some insights and techniques Admins can use to:

  1. Understand why Jira's Full Reindex is slow
  2. Optimize it so it takes full advantage of the infrastructure and parallelism to Reindex all the Issues as quickly as possible.

When a Jira node is performing a Full Reindex, it reports "maintenance" to the /status URL probe and doesn't process end-users requests, so it's interesting to have finish as soon as possible using as much resource available as possible.

This article was written based off experience from the Atlassian Support and is provided as-is.

Going beyond this to further optimize the Full Reindex is the expertise and scope of Atlassian certified Solution Partners.

Known Reindex performance issues

Optimal reindex speed is observed on Jira 9.4.latest and 9.12.6.

課題要約AffectsFixed on

JRASERVER-74787

May increase up to 4× the Full Reindex time observed on Jira 8.20.

9.0.0 ~ 9.4.5

9.5.0 ~ 9.6.99

9.4.6 ~ 9.4.99

9.7.0 ~ 9.99.99

JRASERVER-77548

May double the Full Reindex time observed on 9.4.6.

9.5.0 — 9.12.5

9.13.0 — 9.14.99

9.12.6 — 9.12.99

9.15.0 — 9.99.99

The ".99" version is to illustrate "all that came after on this range", so "9.4.6 – 9.4.99" includes 9.4.8, 9.4.10, 9.4.16; and "9.0.0 – 9.4.5" includes 9.1.1, 9.3.2, 9.4.1, etc.


環境

All versions of Jira Data Center 8 and 9. Both Jira Software and Jira Service Management.


ソリューション

Jira's Full Reindex is very I/O intensive: many Threads read a lot of data from the DB at the same time and write it to memory and disk. The main factors that can contribute to a Full Reindex low performance are:

  • Custom Field suboptimal configuration (expensive global context fields)
  • Memory shortage on the Jira node (JVM Heap)
  • DB underperformance
  • Local storage underperformance (Index folders on the local Jira home folder)


To address that, we may follow these general steps:

  1. Check the relevant logs
  2. Confirm there are no memory (Heap) shortage on the JVM
  3. Assess and optimize Custom Fields contexts
  4. Increase the Index Threads and DB connections as needed
  5. Adjust Lucene RAM Buffer
  6. Check for filesystem underperformance
  7. Check for DB underperformance
  8. Increase the Reindex batch sizes
  9. Limiting factors


While these are general guidelines, your instance details (custom fields types, amount and distribution of data, etc) and infra-structure may be such that you may have to revert some changes (because they weren't beneficial or were even harmful) or rely on just some advices instead of all of them. This requires several iterations of Full Reindex, analysis of data, config changes and repeat.

It is possible to further optimize beyond what's described here, but the config changes start affecting operations beyond the Full Reindex and may not grant as many benefits as the complexity added to the instance. Atlassian certified Solution Partners may be able to advise further through their professional services, though.


1. Relevant logs

After a Full Reindex there are a few log entries very useful for the performance assessment and tuning:

grep -E -a -h " main | JiraTaskExecutionThread-[0-9]+ " application-logs/atlassian-jira.log* | grep -E "Reindex All starting|Reindex All COMPLETED|Canned Response.*FOREGROUND reindex all|ReindexAll took|Index backup started" | sort

(change the file path to the correct one. application-logs/atlassian-jira.log is the path of the unzipped Support Zip)

2023-12-27 12:46:44,589-0500 JiraTaskExecutionThread-1 ··· Reindex All starting...
2023-12-27 12:46:44,693-0500 JiraTaskExecutionThread-1 ··· Canned Response 'REINDEX ALL EVENT' is about to start FOREGROUND reindex all
2023-12-27 12:46:44,884-0500 JiraTaskExecutionThread-1 ··· Canned Response REINDEX ALL EVENT has finished FOREGROUND reindex all
2023-12-27 18:19:43,620-0500 JiraTaskExecutionThread-1 ··· ReindexAll took: 19978969 ms in foreground, index size is 14 GB
2023-12-27 18:19:47,878-0500 JiraTaskExecutionThread-1 ··· Index backup started. Requesting node: ···, currentNode: ···
2023-12-27 18:20:00,152-0500 JiraTaskExecutionThread-1 ··· Reindex All COMPLETED without any errors. Total time: 19990862ms. Reindex run: 453

(··· is redacted info stripped out for better legibility)

The bulk of the Reindex, and what this article aims at improving, is the time between the "Canned Response finished" (line 3) and the "ReindexAll took" (line 4).

The time between line 4 and 5 is what it takes to reindex "shared entities" like filters, dashboards, etc. Some customers with thousands of Filters and Dashboards may have a considerable time on this, but the bulk of the Reindex time should be between lines 3 and 4.


2. Eliminate memory pressure

The Full Reindex reads from the DB, parses the data and stores it into the Lucene memory (which is shortly after flushed to disk). This means the Full Reindex makes intensive use of the "Eden" portion of the JVM Heap — a lot of very short lived objects.

We should load the GC logs into tools like GC Viewer (Best practices for performance troubleshooting tools) to confirm there's no Heap shortage and GC throughput is high (>= 97%) even during Full Reindex — and no Full GC is ever observed. A Full GC event during the Full Reindex will compromise the data we'd analyze and we need to get rid of it first:

  • Restart a Jira node and kick off a Full Reindex right after it's online (even better if you can leave it out of the LB and access it through an alternate Tomcat port)
  • If a Full GC still occurs, we may need to increase the Heap memory (first make sure there's the -XX:UseG1GC JVM Opt)

If we have Thread dumps during the Full Reindex, check if there are no GC Threads with high CPU — this indicates GC overhead and needs to be addressed first before we can assertively move forward.


3. Optimize Custom Fields context

Perhaps the most important evidence of the Full Reindex is the "field indexing cost" report printed in the atlassian-jira.log after each successful Full Reindex:

This lists the most expensive fields to reindex and provides two important infos: the count (the number of Issues that this field was indexed on) and the avg time of reindex:


>>> Click here to expand a command and output example <<<
Linux log grep example
grep -E "\[indexing-stats\] field indexing cost" application-logs/atlassian-jira.log* -m30 | awk -v OFS='\t' 'BEGIN {print "Order", "Count", "Avg in ms", "Custom Field"}; {match($0, /order:[0-9]+/); myorder=substr($0, RSTART+6, RLENGTH-6); match($0, /name:.*, isKnown:/); myfield=substr($0, RSTART+5, RLENGTH-15); match($0, /avg:[0-9\.,]+/); myavg=substr($0, RSTART+4, RLENGTH-4); match($0, /count:[0-9]+/); mycount=substr($0, RSTART+6, RLENGTH-6);; print myorder, mycount, myavg, myfield}' | column -t -s $'\t'

The -m30 on the first grep limits the parsing to the first fields, they're ordered from most expensive to least.

Sample output
Order  Count    Avg in ms  Custom Field
1      1414575  182.3      customfield_14502 (Person Active)
2      1414575  182.3      customfield_14503 (User Cost Center)
3      1414575  182.3      customfield_14501 (Employee ID)
4      1414575  182.2      customfield_14504 (User Company Code)
5      1509084  11.5       customfield_10100 (Approvals)
6      1509084  10.8       customfield_11300 (Epic Link)
7      1509084  9.9        customfield_17204 (Original story points)
8      1509084  9.9        customfield_17202 (Target start)
9      1509084  9.9        customfield_17203 (Target end)
10     1509084  8.4        customfield_11304 (Sprint)
11     1509084  8.2        customfield_26600 (Acknowledgement)
12     1509084  7.0        lastUpdatedBy customfield_10300 (issueFunction)
13     1414575  7.3        customfield_26206 (Last comment)
14     1509084  6.8        portfolioParent customfield_10300 (issueFunction)
15     1509084  5.0        customfield_10701 (Resolution Time)
16     1509084  4.6        customfield_11305 (Rank)
17     1509084  3.5        comments customfield_10300 (issueFunction)
18     1509084  3.4        remoteLinks customfield_10300 (issueFunction)
19     1509084  3.4        customfield_11800 (Problem)
20     1509084  3.3        versions
21     1509084  3.2        fixVersions
22     1509084  3.2        components
23     1509084  3.2        attachment
24     1509084  3.2        subtasks
25     1509084  3.2        voter
26     1509084  3.1        watcher
27     1509084  3.1        labels
28     1509084  1.1        customfield_12300 (Days Open)
29     1509084  1.1        customfield_11901 (Elapsed Time)
30     1414575  1.1        customfield_14500 (IPR State)

On this example, we see 2 valuable infos: some fields have very high avg times and many are present in all 1.4–1.5M Issues — probably fields with Global or a too broad context.


Even if the Field's not used in the Project, it's still being indexed — and if it's a calculated, dynamic or scripted field (each 3rd party app names them differently), they'll greatly impact reindex time regardless.

You may learn what custom fields are these by querying the DB:

select * from customfield where id in (14501, 14502, 14503, 14504);

Or running the query from this article:

If there are "native" Jira fields ranking high, we may suspect of an underlying underperformance in the filesystem or DB (addressed on the next items).

アクション

You may benefit from cleaning up this instance — specially reducing the context of the most expensive custom fields first. These articles below may help you on this effort:

Archiving Issues is always a good option to decrease the Full Reindex duration and even improve performance on other regular end-user operations in Jira that rely on Index searches.


4. Index Threads and DB connections

Jira's Full Reindex (and since Jira 9, the Index delta catch up on startup) are parallelized among a number of Index Threads we can configure. Background Reindex, on the other hand, is single-threaded.

Since the Full Reindex is not CPU intensive, we may benefit from having more Index Threads reading from the DB at once and writing to memory and disk (Lucene disk flush is single-threaded, though). The more we can parallelize, the better.

As a starting point, you can check the number of CPU cores available to Jira and set the Reindex Threads to double that amount:

<available-processors>24</available-processors>

This <available-processors> line is present inside the Support Zip: /application-properties/application.xml.

We could set the Indexer Threads to 50 based on this. We'd edit the jira-config.properties file located in the Jira home folder:

jira.index.issue.threads = 50

If the file doesn't exist, you can create it. If the property exist, update it or add it in a new line if it's not there. See Edit the jira-config.properties file in Jira server for more.


Now we need enough DB connections to support these many Index Threads. On the dbconfig.xml file in Jira home folder, update the pool-max-size to at least the double of Index Threads:

<pool-max-size>100</pool-max-size>

Remember since Jira 8.0 we advise having at least 40 for the pool-max-size. It's OK if you have more, though you may run into bottlenecks if you work with less.


Depending on the instance (custom field types and all), we could even go beyond 2× the CPU cores, but generally this — along with the other tuning items below — is enough to considerably drop the Reindex time. Going beyond 2× the CPU requires a more detailed Thread dump and CPU load analysis.


5. Lucene RAM Buffer

Each Index Threads creates and stores a lot of Lucene Documents in memory during Reindex, and Lucene flushes this data to the disk every 5 minutes or when it's Ram Buffer's full.

Lucene's RAM Buffer is 1024MB by default since Jira 8.0 and this may not be enough depending on how many Custom Fields you have and how much data you have stored in them.

If you notice on the atlassian-jira.log messages like this during the Full Reindex (they're only printing during Full Reindex, actually), you should double it's size:

2023-12-17 02:15:52,700-0500 ClusterMessageHandlerServiceThread:thread-1 WARN      [c.a.jira.index.MonitoringIndexWriter] [lucene-stats] Detected frequent flushes (every 14 millis) of lucene index which is below warning limit: jira.index.warn.flush.min.interval.millis=1000 millis. This may affect the foreground indexing performance. Please visit https://confluence.atlassian.com/x/w0VwOQ for more information.
>>> Click here to expand a command and output example <<<
Linux example command
$ grep "Detected frequent flushes" application-logs/atlassian-jira.log* -h | awk -v OFS='\t' 'BEGIN {print "Timestamp", "Flush rate"};{match($0, /every [0-9]+ millis/); flushRate=substr($0, RSTART+6, RLENGTH-13); print $1" "$2, flushRate}' | column -t -s $'\t'
Example output
Timestamp                     Flush rate
2023-12-16 20:00:04,545-0500  6
2023-12-16 20:23:55,546-0500  93
2023-12-16 20:47:22,534-0500  155
2023-12-16 20:52:34,535-0500  12
2023-12-16 20:58:10,536-0500  16
2023-12-16 21:03:10,539-0500  9
2023-12-16 21:08:25,882-0500  8
2023-12-16 21:15:55,534-0500  92

On this output, at 20:00, the average flush rate of Lucene to disk was 6 milliseconds for a whole 5 minute period. This is overwhelming.


This message's printed if the average flush rate on the 5-minute interval was below 1,000 milliseconds. The presence of this log message may indicate the RAM Buffer's not enough for the Reindex and this may be compromising disk performance.

We can increase it by updating or adding this to the jira-config.properties:

jira.index.batch.maxrambuffermb = 2048

This Buffer's taken from the JVM Heap — this is why it's important to assert the Heap's not already under too much pressure (topic #2).

Reducing the fields context or archiving Issues can also considerably contribute to easing the pressure on the Lucene RAM Buffer.

Increasing the number of Index Threads may also increase the pressure on Lucene RAM Buffer (as there will be more Threads filling up the Buffer at the same time).


6. Filesystem underperformance

Jira uses the local home folder for it's Indexes and the Full Reindex is very disk intensive. Improving the technology of the local storage will not only benefit the Full Reindex but also general performance in Jira (every search, gadget, board, WebHook and many Automations) rely on the Index as data source.

This article on Troubleshooting performance with Jira Stats can help us on this.

We'll look into the "timeToAddMillis" metric and we expect it to be around 1ms. If it's above that, it indicates a potential filesystem underperformance.

We can also download the JAR from Test disk access speed for a Java application and run it to the local Index folder and compare with the benchmark on the article.

Here'a command and sample output of parsing the atlassian-jira.log for some relevant JIRA-STATS:


>>> Click here to expand a command and output example <<<
Linux example command
grep -E ' \[JIRA-STATS\] \[LOCALQ\] \[VIA-INVALIDATION\] .* snapshot stats:' application-logs/atlassian-jira.log* -h -a | awk '
BEGIN {
OFS="\t";
};
{
match($0, /timeToAddMillis":{"count":[0-9]+,"min":[0-9]+,"max":[0-9]+,"sum":[0-9]+,"avg":[0-9]+/);
stat_match=substr($0, RSTART, RLENGTH);
match(stat_match, /[0-9]+$/);
stat_avg=substr(stat_match, RSTART, RLENGTH);
timeToAddMillis[$1" "substr($2,1,8)]=stat_avg;
};
{
match($0, /timeToSendMillis":{"count":[0-9]+,"min":[0-9]+,"max":[0-9]+,"sum":[0-9]+,"avg":[0-9]+/);
stat_match=substr($0, RSTART, RLENGTH);
match(stat_match, /[0-9]+$/);
stat_avg=substr(stat_match, RSTART, RLENGTH);
timeToSendMillis[$1" "substr($2,1,8)]=stat_avg;
};
{
match($0, /queueSize":[0-9]+/);
stat_match=substr($0, RSTART, RLENGTH);
match(stat_match, /[0-9]+$/);
stat_avg=substr(stat_match, RSTART, RLENGTH);
queueSize[$1" "substr($2,1,8)]=stat_avg;
};
END {
print "Timestamp", "timeToAddMillis", "timeToSendMillis", "queueSize";
for (s in timeToAddMillis) {print s, timeToAddMillis[s], timeToSendMillis[s], queueSize[s]}
}
' | column -t -s $'\t' | sort -r
Sample output
Timestamp            timeToAddMillis  timeToSendMillis  queueSize
2023-12-16 19:44:36  5                0                 0
2023-12-16 19:39:36  5                0                 0
2023-12-16 19:34:36  5                0                 0
2023-12-16 19:29:36  5                0                 0
2023-12-16 19:24:36  5                0                 1
2023-12-16 19:19:36  5                0                 0
2023-12-16 19:14:36  4                0                 1
2023-12-16 19:09:36  5                0                 1
2023-12-16 19:04:36  4                0                 1
2023-12-16 18:59:36  4                0                 0
2023-12-16 18:54:36  4                0                 0
2023-12-16 18:49:36  4                0                 0

Notice how the timeToAddMillis is consistently above 1ms. These metrics are printed every 5 minutes to the log.


7. Database underperformance

Database underperformance is a very common cause of Reindex underperformance, too.

First thing, the DBA team should monitor the DB's CPU during the Full Reindex and assess if there are any maintenance or optimization tasks that could improve Jira's performance.

Again the JIRA-STATS may help us identify potential DB issues:


>>> Click here to expand a command and output example <<<
Linux command example for DB read in millis
grep -E ' \[JIRA-STATS\] \[VERSIONING\] snapshot stats:' application-logs/atlassian-jira.log* -h -a | awk '
BEGIN {
OFS="\t";
};
{
match($0, /getIssueVersionMillis":{"count":[0-9]+,"min":[0-9]+,"max":[0-9]+,"sum":[0-9]+,"avg":[0-9]+/);
stat_match=substr($0, RSTART, RLENGTH);
match(stat_match, /[0-9]+$/);
stat_avg=substr(stat_match, RSTART, RLENGTH);
getIssueVersionMillis[$1" "substr($2,1,8)]=stat_avg;
};
{
match($0, /incrementIssueVersionMillis":{"count":[0-9]+,"min":[0-9]+,"max":[0-9]+,"sum":[0-9]+,"avg":[0-9]+/);
stat_match=substr($0, RSTART, RLENGTH);
match(stat_match, /[0-9]+$/);
stat_avg=substr(stat_match, RSTART, RLENGTH);
incrementIssueVersionMillis[$1" "substr($2,1,8)]=stat_avg;
};
END {
print "Timestamp", "getIssueVersionMillis", "incrementIssueVersionMillis";
for (s in getIssueVersionMillis) {print s, getIssueVersionMillis[s], incrementIssueVersionMillis[s]}
}
' | column -t -s $'\t' | sort -r
Linux command example for DB latency
TOKEN="latencyNanos"; DATEPATTERN="202.-..-.."; egrep "^$DATEPATTERN.*?$TOKEN" -h application-logs/atlassian-jira-perf.log | while read -r line ; do PERF_TIME=`echo $line | cut -d" " -f1,2 | sed -e 's/,[0-9]{3}//g'`; echo $line | grep -Eo ''$TOKEN'\S*"(,"value"){0,1}:"{0,1}[0-9]*"{0,1}' | while read -r linee; do PERF_VALUE=`echo $linee | sed 's/.*://g' | sed 's/"//g'`; echo "$PERF_TIME $PERF_VALUE"; done; done | awk -v OFS='\t' '($3/1000000 >= 3) {print $1" "$2, $3, $3/1000000"ms"}' | column -t -s $'\t';


We’d expect these stats to be around:

  • getIssueVersionMillis: 5ms

  • incrementIssueVersionMillis: 10ms

  • latencyNanos: as low as possible (3ms is already suspicious)

When DB performance impacts Jira Reindex, we won't see any single query take much CPU or run for long — but instead, it's an increase of very few milliseconds that cascades into a considerable duration increase overall. This article dives into the DB latency impact on Jira operations:

All these stats take into account: Java time + network time + DB time + network time + Java time. The latencyNanos stat is collected every minute and the query is select * from productlicense, which returns a small and stable amount of rows and data. So network time may interfere as well, and also Full GC cycles (the stat may report longer because the Thread was interrupted so the JVM could free up Heap memory).


If Jira's running on Postgres or MS SQL Server, you'll need to perform some DB optimization tasks regularly:

For Postgres
VACUUM FULL;
ANALYZE VERBOSE;
REINDEX DATABASE <jira-db-name>;

See Optimize and Improve PostgreSQL Performance with VACUUM, ANALYZE, and REINDEX.


For MS SQL
Daily for hot tables - UPDATE STATISTICS <table.name>;
Weekly for all Jira DB - UPDATE STATISTICS with fullscan;

Please don't underestimate or disregard this — we keep having consistent reports of Full Reindex times dropping below 50% or more when these exact commands are executed (even if cases with autovacuum enabled).


8. Increase the Reindex batch sizes

Jira Full Reindex works by having a single Thread load a batch of Issues from the DB to the memory (just their Ids), and spawn a number of Index Threads to consume this batch in parallel (going to the DB for data and creating the Index documents). The Index Threads consume this batch by fetching a number of Issues from it at every iteration (50 is the default value), until there's no more Issues left in the batch. Then they die or sleep until the batch is replenished again.

By design, the single "producer" Thread waits until the loaded batch is depleted and all Index Threads have completed their work — this may lead to an excessive waiting time while we could've had the Index Threads working for longer if we had a bigger batch loaded every time. i.e. have this waiting time every 40,000 Issues, not every 4,000.

This waiting time happens every so often, and we can make a better use of it by bumping the batch sizes to benefit more from the parallel phase of the Reindex, and rely less on the synchronized or single-threaded phase.

We can update or add these lines into the jira-config.properties:

jira.index.background.batch.size = 40000
jira.index.issue.maxqueuesize = 40000
jira.index.sharedentity.maxqueuesize = 40000

These 3 configs should always have the same number. Their default is 4000.

You can start with 10× that, 40000, and bump it to 80000 or more until you notice no more benefit from increasing it (see the exact formula below). The tradeoff is more Heap memory will be consumed during Full Reindex and the pressure on the DB and filesystem (Lucene RAM Buffer) will also increase, as there will be more Index Threads fetching data and processing them in parallel.

A good starting point is 40,000 (10× the default) and you can increase it by tweaking the "iteration" variable on this equation:

Index Threads × 50 × Iterations = Batch size

The more iterations we allow, the more we'll rely on the Index Threads parallel processing and the more we'll demand from all resources: CPU, Heap memory, disk, DB CPU and network.

The defaults on Jira 8 and 9, as an example, are: 20 × 50 × 4 = 4000. 4 iterations seems to be too low for most instances (we have JRASERVER-76819 to increase this) so you can start off with 10 or 20 iterations and adjust from there at every Full Reindex execution.


9. Limiting factors

While we may want the Full Reindex to finish as quickly as possible, optimizing it this much will likely put pressure on the the node CPU, JVM Heap memory, local storage, Database load and network even (node to database). If you start noticing some bottlenecks or perfomance degradation as a side effect on other nodes or in the Full Reindex duration itself, for example, you may revert the most recent changes and try others — if unable to improve the infra-structure.


最終更新日: 2025 年 1 月 9 日

この内容はお役に立ちましたか?

はい
いいえ
この記事についてのフィードバックを送信する
Powered by Confluence and Scroll Viewport.