Scripts, caching, and search speed

Scripts, caching, and search speed

Elasticsearch performs a number of optimizations to make using scripts as fast as possible. One important optimization is a script cache. The compiled script is placed in a cache so that requests that reference the script do not incur a compilation penalty.

Cache sizing is important. Your script cache should be large enough to hold all of the scripts that users need to be accessed concurrently.

If you see a large number of script cache evictions and a rising number of compilations in node stats, your cache might be too small.

All scripts are cached by default so that they only need to be recompiled when updates occur. By default, scripts do not have a time-based expiration. You can change this behavior by using the script.cache.expire setting. Use the script.cache.max_size setting to configure the size of the cache.

The size of scripts is limited to 65,535 bytes. Set the value of script.max_size_in_bytes to increase that soft limit. If your scripts are really large, then consider using a native script engine.

Improving search speed

Scripts are incredibly useful, but can’t use Elasticsearch’s index structures or related optimizations. This relationship can sometimes result in slower search speeds.

If you often use scripts to transform indexed data, you can make search faster by transforming data during ingest instead. However, that often means slower index speeds. Let’s look at a practical example to illustrate how you can increase search speed.

When running searches, it’s common to sort results by the sum of two values. For example, consider an index named my_test_scores that contains test score data. This index includes two fields of type long:

  • math_score
  • verbal_score

You can run a query with a script that adds these values together. There’s nothing wrong with this approach, but the query will be slower because the script valuation occurs as part of the request. The following request returns documents where grad_year equals 2099, and sorts by the results by the valuation of the script.

  1. resp = client.search(
  2. index="my_test_scores",
  3. query={
  4. "term": {
  5. "grad_year": "2099"
  6. }
  7. },
  8. sort=[
  9. {
  10. "_script": {
  11. "type": "number",
  12. "script": {
  13. "source": "doc['math_score'].value + doc['verbal_score'].value"
  14. },
  15. "order": "desc"
  16. }
  17. }
  18. ],
  19. )
  20. print(resp)
  1. response = client.search(
  2. index: 'my_test_scores',
  3. body: {
  4. query: {
  5. term: {
  6. grad_year: '2099'
  7. }
  8. },
  9. sort: [
  10. {
  11. _script: {
  12. type: 'number',
  13. script: {
  14. source: "doc['math_score'].value + doc['verbal_score'].value"
  15. },
  16. order: 'desc'
  17. }
  18. }
  19. ]
  20. }
  21. )
  22. puts response
  1. const response = await client.search({
  2. index: "my_test_scores",
  3. query: {
  4. term: {
  5. grad_year: "2099",
  6. },
  7. },
  8. sort: [
  9. {
  10. _script: {
  11. type: "number",
  12. script: {
  13. source: "doc['math_score'].value + doc['verbal_score'].value",
  14. },
  15. order: "desc",
  16. },
  17. },
  18. ],
  19. });
  20. console.log(response);
  1. GET /my_test_scores/_search
  2. {
  3. "query": {
  4. "term": {
  5. "grad_year": "2099"
  6. }
  7. },
  8. "sort": [
  9. {
  10. "_script": {
  11. "type": "number",
  12. "script": {
  13. "source": "doc['math_score'].value + doc['verbal_score'].value"
  14. },
  15. "order": "desc"
  16. }
  17. }
  18. ]
  19. }

If you’re searching a small index, then including the script as part of your search query can be a good solution. If you want to make search faster, you can perform this calculation during ingest and index the sum to a field instead.

First, we’ll add a new field to the index named total_score, which will contain sum of the math_score and verbal_score field values.

  1. resp = client.indices.put_mapping(
  2. index="my_test_scores",
  3. properties={
  4. "total_score": {
  5. "type": "long"
  6. }
  7. },
  8. )
  9. print(resp)
  1. response = client.indices.put_mapping(
  2. index: 'my_test_scores',
  3. body: {
  4. properties: {
  5. total_score: {
  6. type: 'long'
  7. }
  8. }
  9. }
  10. )
  11. puts response
  1. const response = await client.indices.putMapping({
  2. index: "my_test_scores",
  3. properties: {
  4. total_score: {
  5. type: "long",
  6. },
  7. },
  8. });
  9. console.log(response);
  1. PUT /my_test_scores/_mapping
  2. {
  3. "properties": {
  4. "total_score": {
  5. "type": "long"
  6. }
  7. }
  8. }

Next, use an ingest pipeline containing the script processor to calculate the sum of math_score and verbal_score and index it in the total_score field.

  1. resp = client.ingest.put_pipeline(
  2. id="my_test_scores_pipeline",
  3. description="Calculates the total test score",
  4. processors=[
  5. {
  6. "script": {
  7. "source": "ctx.total_score = (ctx.math_score + ctx.verbal_score)"
  8. }
  9. }
  10. ],
  11. )
  12. print(resp)
  1. response = client.ingest.put_pipeline(
  2. id: 'my_test_scores_pipeline',
  3. body: {
  4. description: 'Calculates the total test score',
  5. processors: [
  6. {
  7. script: {
  8. source: 'ctx.total_score = (ctx.math_score + ctx.verbal_score)'
  9. }
  10. }
  11. ]
  12. }
  13. )
  14. puts response
  1. const response = await client.ingest.putPipeline({
  2. id: "my_test_scores_pipeline",
  3. description: "Calculates the total test score",
  4. processors: [
  5. {
  6. script: {
  7. source: "ctx.total_score = (ctx.math_score + ctx.verbal_score)",
  8. },
  9. },
  10. ],
  11. });
  12. console.log(response);
  1. PUT _ingest/pipeline/my_test_scores_pipeline
  2. {
  3. "description": "Calculates the total test score",
  4. "processors": [
  5. {
  6. "script": {
  7. "source": "ctx.total_score = (ctx.math_score + ctx.verbal_score)"
  8. }
  9. }
  10. ]
  11. }

To update existing data, use this pipeline to reindex any documents from my_test_scores to a new index named my_test_scores_2.

  1. resp = client.reindex(
  2. source={
  3. "index": "my_test_scores"
  4. },
  5. dest={
  6. "index": "my_test_scores_2",
  7. "pipeline": "my_test_scores_pipeline"
  8. },
  9. )
  10. print(resp)
  1. response = client.reindex(
  2. body: {
  3. source: {
  4. index: 'my_test_scores'
  5. },
  6. dest: {
  7. index: 'my_test_scores_2',
  8. pipeline: 'my_test_scores_pipeline'
  9. }
  10. }
  11. )
  12. puts response
  1. const response = await client.reindex({
  2. source: {
  3. index: "my_test_scores",
  4. },
  5. dest: {
  6. index: "my_test_scores_2",
  7. pipeline: "my_test_scores_pipeline",
  8. },
  9. });
  10. console.log(response);
  1. POST /_reindex
  2. {
  3. "source": {
  4. "index": "my_test_scores"
  5. },
  6. "dest": {
  7. "index": "my_test_scores_2",
  8. "pipeline": "my_test_scores_pipeline"
  9. }
  10. }

Continue using the pipeline to index any new documents to my_test_scores_2.

  1. resp = client.index(
  2. index="my_test_scores_2",
  3. pipeline="my_test_scores_pipeline",
  4. document={
  5. "student": "kimchy",
  6. "grad_year": "2099",
  7. "math_score": 1200,
  8. "verbal_score": 800
  9. },
  10. )
  11. print(resp)
  1. response = client.index(
  2. index: 'my_test_scores_2',
  3. pipeline: 'my_test_scores_pipeline',
  4. body: {
  5. student: 'kimchy',
  6. grad_year: '2099',
  7. math_score: 1200,
  8. verbal_score: 800
  9. }
  10. )
  11. puts response
  1. const response = await client.index({
  2. index: "my_test_scores_2",
  3. pipeline: "my_test_scores_pipeline",
  4. document: {
  5. student: "kimchy",
  6. grad_year: "2099",
  7. math_score: 1200,
  8. verbal_score: 800,
  9. },
  10. });
  11. console.log(response);
  1. POST /my_test_scores_2/_doc/?pipeline=my_test_scores_pipeline
  2. {
  3. "student": "kimchy",
  4. "grad_year": "2099",
  5. "math_score": 1200,
  6. "verbal_score": 800
  7. }

These changes slow the index process, but allow for faster searches. Instead of using a script, you can sort searches made on my_test_scores_2 using the total_score field. The response is near real-time! Though this process slows ingest time, it greatly increases queries at search time.

  1. resp = client.search(
  2. index="my_test_scores_2",
  3. query={
  4. "term": {
  5. "grad_year": "2099"
  6. }
  7. },
  8. sort=[
  9. {
  10. "total_score": {
  11. "order": "desc"
  12. }
  13. }
  14. ],
  15. )
  16. print(resp)
  1. response = client.search(
  2. index: 'my_test_scores_2',
  3. body: {
  4. query: {
  5. term: {
  6. grad_year: '2099'
  7. }
  8. },
  9. sort: [
  10. {
  11. total_score: {
  12. order: 'desc'
  13. }
  14. }
  15. ]
  16. }
  17. )
  18. puts response
  1. const response = await client.search({
  2. index: "my_test_scores_2",
  3. query: {
  4. term: {
  5. grad_year: "2099",
  6. },
  7. },
  8. sort: [
  9. {
  10. total_score: {
  11. order: "desc",
  12. },
  13. },
  14. ],
  15. });
  16. console.log(response);
  1. GET /my_test_scores_2/_search
  2. {
  3. "query": {
  4. "term": {
  5. "grad_year": "2099"
  6. }
  7. },
  8. "sort": [
  9. {
  10. "total_score": {
  11. "order": "desc"
  12. }
  13. }
  14. ]
  15. }