千家信息网

Scripting脚本的介绍和使用方法

发表于:2025-01-26 作者:千家信息网编辑
千家信息网最后更新 2025年01月26日,这篇文章主要介绍"Scripting脚本的介绍和使用方法",在日常操作中,相信很多人在Scripting脚本的介绍和使用方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答
千家信息网最后更新 2025年01月26日Scripting脚本的介绍和使用方法

这篇文章主要介绍"Scripting脚本的介绍和使用方法",在日常操作中,相信很多人在Scripting脚本的介绍和使用方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答"Scripting脚本的介绍和使用方法"的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

简介

官方7.9版本:https://www.elastic.co/guide/en/elasticsearch/reference/7.9/modules-scripting.html

通过ES脚本来解决,其允许用户在一些特定的API中对自定义表达式进行求值。

1. ES Scripting历史

版本

使用脚本

< Elasticsearch 1.4

MVEL 脚本

< Elasticsearch 5.0

Groovy 脚本

'>= Elasticsearch 5.0

painless 脚本

2. Painless Scripting 简介

Painless是一种简单,安全的脚本语言,专为与Elasticsearch一起使用而设计。它是Elasticsearch的默认脚本语言,可以安全地用于内联和存储脚本。

Painless特点:

  • 性能优秀:Painless脚本运行速度比备选方案(包括Groovy)快几倍。

  • 安全性强:使用白名单来限制函数与字段的访问,避免了可能的安全隐患。

  • 可选输入:变量和参数可以使用显式类型或动态def类型。

  • 上手容易:扩展了java的基本语法,并兼容groove风格的脚本语言特性。

  • 特定优化:是ES官方专为Elasticsearch脚本编写而设计。

3. Scripting 应用场景

增删改查能解决业务场景80%的问题,Painless脚本操作一般应用于相对复杂的业务场景中。

  • 自定义字段

  • 自定义评分

  • 自定义更新

  • 自定义reindex

  • 聚合

  • 其他自定义操作

在 update, update-by-query, 或 reindex API中使用脚本时,需要通过ctx去访问文档中的字段。ctx._source: 访问文档_source字段ctx.op: 对文档对应的操作,包括index和deletectx._index : 访问文档的 meta-fields其他字段或变量的访问见: update context

4. Scripting 使用模板

"script": {    "lang":   "...",  # 代表language脚本语言,默认指定为:painless    "source" | "id": "...",  # 脚本的核心部分,id应用于:stored script    "params": { ... }  # 传递给脚本使用的变量参数  }

Scripting应用

1. 自定义字段

如:需要给每个doc的complexrank字段翻倍并返回翻倍后的值

POST sphinx-doctor/_search{  "from": 0,  "size": 20,  "query": {    "match_all": {}  },  "_source": "complexrank",  "script_fields": {    "custom_field": {      "script": {        "lang": "expression", # 这里脚本语言选择的expression        "source": "doc['complexrank'] * multiplier",        "params": {          "multiplier": 2        }      }    }  }}说明:对原索引中的complexrank字段值进行2倍返回给custom_field字段{    "took":20,    "timed_out":false,    "_shards":{        "total":1,        "successful":1,        "skipped":0,        "failed":0    },    "hits":{        "total":194354,        "max_score":1,        "hits":[            {                "_index":"sphinx-doctor-20.11.06-103145",                "_type":"_doc",                "_id":"740",                "_score":1,                "_source":{                    "complexrank":"5985"                },                "fields":{                    "custom_field":[                        11970                    ]                }            },            {                "_index":"sphinx-doctor-20.11.06-103145",                "_type":"_doc",                "_id":"742",                "_score":1,                "_source":{                    "complexrank":"5325"                },                "fields":{                    "custom_field":[                        10650                    ]                }            }        ]    }}

如:返回日期字段中的"年"或"月"或"日"等

POST drug/_search{  "from": 0,  "size": 20,  "query": {    "match_all": {}  },  "script_fields": {    "custom_field": {      "script": {        "source": "doc.ctime.value.year" #月:month      }    }  }}{    "took":57,    "timed_out":false,    "_shards":{        "total":1,        "successful":1,        "skipped":0,        "failed":0    },    "hits":{        "total":173536,        "max_score":1,        "hits":[            {                "_index":"drug-20.12.03-151452",                "_type":"_doc",                "_id":"1883795984",                "_score":1,                "fields":{                    "custom_field":[                        2014                    ]                }            },            {                "_index":"drug-20.12.03-151452",                "_type":"_doc",                "_id":"1883795985",                "_score":1,                "fields":{                    "custom_field":[                        2014                    ]                }            }        ]    }}

2. 自定义评分

POST sphinx-doctor/_search{  "from": 0,  "size": 2,  "_source": "hospitalname",   "query": {    "function_score": {      "query": {        "match": {          "hospitalname": {            "query": "北京协和医院"          }        }      },      "functions": [        {          "filter": {            "match_all": {              "boost": 1            }          },          "script_score": {            "script": {              "source": "_score *params._score + doc['rank'].value * params.rank + doc['adminlevel'].value * params.adminlevel",              "lang": "painless",  # # 是Lucene's expressions 脚本语言              "params": {                "adminlevel": 2500,                "rank": 0.5,                "_score": 1              }            }          }        }      ],      "score_mode": "sum",      "max_boost": 3.4028235e+38,      "boost": 1    }  },  "track_scores": true}{  "took" : 487,  "timed_out" : false,  "_shards" : {    "total" : 1,    "successful" : 1,    "skipped" : 0,    "failed" : 0  },  "hits" : {    "total" : 777965,    "max_score" : 449721.3,    "hits" : [      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "175477",        "_score" : 449721.3,        "_source" : {          "hospitalname" : "北京协和医院"        }      },      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "221609",        "_score" : 440269.34,        "_source" : {          "hospitalname" : "北京协和医院"        }      }    ]  }}Java API:排序String scriptText = "_score *params._score + doc['rank'].value * params.rank + doc['adminlevel'].value * params.adminlevel";Map params = new HashMap<>();params.put("_score", 1.0f);params.put("rank", 0.5f);params.put("adminlevel", 2500f);Script script = newScript(ScriptType.INLINE, "painless", scriptText, params);ScriptScoreFunctionBuilder scriptScoreFunctionBuilder = ScoreFunctionBuilders.scriptFunction(script);FunctionScoreQueryBuilder functionScoreQueryBuilder = QueryBuilders.functionScoreQuery(boolQueryBuilder, scriptScoreFunctionBuilder)                        .scoreMode(FunctionScoreQuery.ScoreMode.SUM);builder.query(functionScoreQueryBuilder);builder.from(offset);builder.size(limit);//当使用排序_sort 来指定依据某些字段排序。就不会计算得分,需要设置"track_scores":true,这样分数就会被计算和跟踪。builder.trackScores(true);

3. 自定义更新

update:将已有字段值赋值给其他字段。

POST sphinx-doctor/_doc/102647/_update{  "script": {    "lang": "painless",    "source": "ctx._source.hospitalname = params.hospitalname;ctx._source.name = params.name",    "params": {      "hospitalname": "北京协和医院",      "name": "协和医生"    }  }}

update_by_query

POST sphinx-doctor/_update_by_query{  "query": {    "term": {      "id": {        "value": "102647"      }    }  },  "script": {    "lang": "painless",    "source": """       if (ctx._source.registerordercount >= 500) {          ctx._source.registerordercount += 200;       } else {          ctx.op = 1000;      }   """  }}

4.自定义reindex

POST _reindex{  "source": {    "index": "sphinx-doctor-20.12.02-150616",    "size": 4000  },  "dest": {    "index": "sphinx-doctor-20.12.03-151452"  },  "script": {    "source": "ctx._source.complexrank += 10;\nctx._source.netcaseprice = 100;"  }}

5. 聚合

POST sphinx-doctor/_search{  "from": 0,  "size": 20,   "query": {    "match": {      "hospitalname": "北京协和医院"    }  },  "aggs": {    "grade": {      "terms": {        "script": {          "source": "doc['grade.keyword'].value",          "lang": "painless"        }      }    }  }}{  "took" : 187,  "timed_out" : false,  "_shards" : {    "total" : 1,    "successful" : 1,    "skipped" : 0,    "failed" : 0  },  "hits" : {    "total" : 193445,    "max_score" : 44.18691,    "hits" : [      {        "_index" : "sphinx-doctor-20.11.06-103145",        "_type" : "_doc",        "_id" : "30626487346",        "_score" : 44.18691,        "_source" : {          "registerordercount" : 0,          "hospitalfacultyname" : "神经内科",          "idx_doctoridprimary" : "DoctorIdPrimary_335",          "hospitaldistrict" : "东城",          "caseopened" : true,          "isonline" : null,          "dummy" : "SELECT_ALL",          "facultyid" : "1007000",          "price" : "10",          "rank" : "0",          "ctime" : "1568269475",          "id" : "30626487346",          "clinicprice" : "20.00",          "idx_hospitalfacultyid" : "HospitalFacultyId_335",          "educategrade" : "",          "adminlevel" : "1",          "isprimarydoctor" : 1,          "isexpert" : true,          "bookingvisitnum" : 0,          "primaryid" : "30626487346",          "hospitalfacultyid" : "335",          "spaceid" : "200007660088",          "hospitalcity" : "北京",          "isopenvideo" : true,          "grade" : "主任医师",          "name" : "3 0626487346",          "idx_fix_bigcity" : "北京协和医院北京协和医院北京协和医院北京协和医院测试用",          "phoneopened" : true,          "onlineschedule" : "20201207,20201214,20201221",          "isvideotpl" : false,          "issanjia" : true,          "hospitalprovince" : "北京",          "servicelevel" : 2,          "title" : "",          "confirmed" : "1",          "isvip" : false,          "general" : 0,          "menzhenprice" : "10",          "isopencosvideo" : false,          "hospitalname" : "北京协和医院北京协和医院北京协和医院北京协和医院测试用",          "netcaseprice" : "20.00",          "registrprice" : 0,          "free" : 0,          "workstatus" : "0",          "registtationopened" : null,          "indextype" : "doctor",          "utime" : 1606978403,          "complexrank" : "5985",          "sex" : "1",          "reservationopened" : null,          "idx_doctorid" : "DoctorId_30626487346",          "idx_hospital_info" : "北京协和医院北京协和医院北京协和医院北京协和医院测试用,北京协和医院,33北京协和医院,协和医院,协和东院,协和东院区,北京协和医院东院,协和西院,北京邮电总医院,邮电总医院,北京协和西院,北京协和医院西院区,北京协和医院西院,北京协和,北京市协和,北京邮电医院,协合医院,北京市协和医院,中国医学科学院北京协和医院,北京协合医院45,协和医院,北京协和,北京市协和,协和东院,北京协和医院东院,协和西院,北京邮电医院,北京协和西院,北京协和医院西院,协合医院,北京市协和医院,中国医学科学院北京协和医院,北京协合医院",          "isopenrecipe" : false,          "hits" : "0",          "iscosmetologydoctor" : null,          "hospitalid" : "1",          "idx_facultyid" : "FacultyId_1007000",          "activitytime" : "1568269520",          "hospitalgrade" : "6",          "idx_hospitalid" : "HospitalId_1",          "username" : "francoliu"        }      },      {        "_index" : "sphinx-doctor-20.11.06-103145",        "_type" : "_doc",        "_id" : "310346050824",        "_score" : 40.138744,        "_source" : {          "registerordercount" : 0,          "hospitalfacultyname" : "心血管内科",          "idx_doctoridprimary" : "DoctorIdPrimary_341",          "hospitaldistrict" : "东城",          "caseopened" : false,          "isonline" : null,          "dummy" : "SELECT_ALL",          "facultyid" : "1010000",          "price" : 0,          "rank" : "0",          "ctime" : "1605859848",          "id" : "310346050824",          "clinicprice" : 0,          "idx_hospitalfacultyid" : "HospitalFacultyId_341",          "educategrade" : "教授",          "adminlevel" : "4",          "isprimarydoctor" : 1,          "isexpert" : true,          "bookingvisitnum" : 0,          "primaryid" : "310346050824",          "hospitalfacultyid" : "341",          "spaceid" : "0",          "hospitalcity" : "北京",          "isopenvideo" : null,          "grade" : "主任医师",          "name" : "地名",          "idx_fix_bigcity" : "北京协和医院",          "phoneopened" : false,          "onlineschedule" : "",          "isvideotpl" : false,          "issanjia" : true,          "hospitalprovince" : "北京",          "servicelevel" : 0,          "title" : "",          "confirmed" : "",          "isvip" : false,          "general" : 0,          "menzhenprice" : 0,          "isopencosvideo" : false,          "hospitalname" : "北京协和医院",          "netcaseprice" : 0,          "registrprice" : 0,          "free" : 0,          "workstatus" : "0",          "registtationopened" : null,          "indextype" : "doctor",          "utime" : 1605859882,          "complexrank" : "0",          "sex" : "1",          "reservationopened" : null,          "idx_doctorid" : "DoctorId_310346050824",          "idx_hospital_info" : "北京协和医院,北京协和医院,33北京协和医院,协和医院,协和东院,协和东院区,北京协和医院东院,协和西院,北京邮电总医院,邮电总医院,北京协和西院,北京协和医院西院区,北京协和医院西院,北京协和,北京市协和,北京邮电医院,协合医院,北京市协和医院,中国医学科学院北京协和医院,北京协合医院45,协和医院,北京协和,北京市协和,协和东院,北京协和医院东院,协和西院,北京邮电医院,北京协和西院,北京协和医院西院,协合医院,北京市协和医院,中国医学科学院北京协和医院,北京协合医院",          "isopenrecipe" : null,          "hits" : "0",          "iscosmetologydoctor" : null,          "hospitalid" : "1",          "idx_facultyid" : "FacultyId_1010000",          "activitytime" : null,          "hospitalgrade" : "6",          "idx_hospitalid" : "HospitalId_1",          "username" : ""        }      }    ]  },  "aggregations" : {    "grade" : {      "doc_count_error_upper_bound" : 0,      "sum_other_doc_count" : 479,      "buckets" : [        {          "key" : "主任医师",          "doc_count" : 77699        },        {          "key" : "副主任医师",          "doc_count" : 72939        },        {          "key" : "主治医师",          "doc_count" : 20399        },        {          "key" : "",          "doc_count" : 13558        },        {          "key" : "住院医师",          "doc_count" : 5731        },        {          "key" : "副主任技师",          "doc_count" : 896        },        {          "key" : "副主任药师",          "doc_count" : 612        },        {          "key" : "主管技师",          "doc_count" : 468        },        {          "key" : "主任药师",          "doc_count" : 355        },        {          "key" : "副主任检验师",          "doc_count" : 309        }      ]    }  }}

6.自定义排序

两个或多个字段进行运算返回一个值进行排序

POST sphinx-doctor/_search{  "_source": [    "clinicprice",    "menzhenprice"  ],  "query": {    "match_all": {}  },  "sort": {    "_script": {      "type": "number",  # string      "order": "desc",      "script": {        "lang": "painless",        "source": "doc['clinicprice'].value + doc['menzhenprice'].value"      }    }  }}{  "took" : 334,  "timed_out" : false,  "_shards" : {    "total" : 1,    "successful" : 1,    "skipped" : 0,    "failed" : 0  },  "hits" : {    "total" : 782852,    "max_score" : null,    "hits" : [      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "102647",        "_score" : null,        "_source" : {          "menzhenprice" : "1400",          "clinicprice" : "2700.00"        },        "sort" : [          4100.0        ]      },      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "8231",        "_score" : null,        "_source" : {          "menzhenprice" : "2000.00",          "clinicprice" : "2000.00"        },        "sort" : [          4000.0        ]      }    ]  }}

逻辑运算符进行判断来影响排序

POST sphinx-doctor/_search{  "_source": [    "clinicprice",    "menzhenprice"  ],  "from": 0,  "size": 2,   "query": {    "match_all": {}  },   "sort": [    {      "complexrank": {        "order": "desc"      }    },    {      "_script": {        "script": {          "inline": "'北京 '==doc['hospitalprovince.keyword'].value?0:('陕西'==doc['hospitalprovince.keyword'].value?1:2)"        },        "type": "number",        "order": "asc"      }    }  ]}{  "took" : 80,  "timed_out" : false,  "_shards" : {    "total" : 1,    "successful" : 1,    "skipped" : 0,    "failed" : 0  },  "hits" : {    "total" : 782852,    "max_score" : null,    "hits" : [      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "5279893112",        "_score" : null,        "_source" : {          "menzhenprice" : "500",          "clinicprice" : 0        },        "sort" : [          981,          2.0        ]      },      {        "_index" : "sphinx-doctor-20.12.09-010000",        "_type" : "_doc",        "_id" : "124096",        "_score" : null,        "_source" : {          "menzhenprice" : "200.00",          "clinicprice" : "200.00"        },        "sort" : [          980,          2.0        ]      }    ]  }}

到此,关于"Scripting脚本的介绍和使用方法"的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注网站,小编会继续努力为大家带来更多实用的文章!

0