Date nanoseconds field type

Date nanoseconds field type

This data type is an addition to the date data type. However there is an important distinction between the two. The existing date data type stores dates in millisecond resolution. The date_nanos data type stores dates in nanosecond resolution, which limits its range of dates from roughly 1970 to 2262, as dates are still stored as a long representing nanoseconds since the epoch.

Queries on nanoseconds are internally converted to range queries on this long representation, and the result of aggregations and stored fields is converted back to a string depending on the date format that is associated with the field.

Date formats can be customised, but if no format is specified then it uses the default:

  1. "strict_date_optional_time_nanos||epoch_millis"

For instance:

  1. resp = client.indices.create(
  2. index="my-index-000001",
  3. mappings={
  4. "properties": {
  5. "date": {
  6. "type": "date_nanos"
  7. }
  8. }
  9. },
  10. )
  11. print(resp)
  12. resp1 = client.bulk(
  13. index="my-index-000001",
  14. refresh=True,
  15. operations=[
  16. {
  17. "index": {
  18. "_id": "1"
  19. }
  20. },
  21. {
  22. "date": "2015-01-01"
  23. },
  24. {
  25. "index": {
  26. "_id": "2"
  27. }
  28. },
  29. {
  30. "date": "2015-01-01T12:10:30.123456789Z"
  31. },
  32. {
  33. "index": {
  34. "_id": "3"
  35. }
  36. },
  37. {
  38. "date": 1420070400000
  39. }
  40. ],
  41. )
  42. print(resp1)
  43. resp2 = client.search(
  44. index="my-index-000001",
  45. sort={
  46. "date": "asc"
  47. },
  48. runtime_mappings={
  49. "date_has_nanos": {
  50. "type": "boolean",
  51. "script": "emit(doc['date'].value.nano != 0)"
  52. }
  53. },
  54. fields=[
  55. {
  56. "field": "date",
  57. "format": "strict_date_optional_time_nanos"
  58. },
  59. {
  60. "field": "date_has_nanos"
  61. }
  62. ],
  63. )
  64. print(resp2)
  1. response = client.indices.create(
  2. index: 'my-index-000001',
  3. body: {
  4. mappings: {
  5. properties: {
  6. date: {
  7. type: 'date_nanos'
  8. }
  9. }
  10. }
  11. }
  12. )
  13. puts response
  14. response = client.bulk(
  15. index: 'my-index-000001',
  16. refresh: true,
  17. body: [
  18. {
  19. index: {
  20. _id: '1'
  21. }
  22. },
  23. {
  24. date: '2015-01-01'
  25. },
  26. {
  27. index: {
  28. _id: '2'
  29. }
  30. },
  31. {
  32. date: '2015-01-01T12:10:30.123456789Z'
  33. },
  34. {
  35. index: {
  36. _id: '3'
  37. }
  38. },
  39. {
  40. date: 1_420_070_400_000
  41. }
  42. ]
  43. )
  44. puts response
  45. response = client.search(
  46. index: 'my-index-000001',
  47. body: {
  48. sort: {
  49. date: 'asc'
  50. },
  51. runtime_mappings: {
  52. date_has_nanos: {
  53. type: 'boolean',
  54. script: "emit(doc['date'].value.nano != 0)"
  55. }
  56. },
  57. fields: [
  58. {
  59. field: 'date',
  60. format: 'strict_date_optional_time_nanos'
  61. },
  62. {
  63. field: 'date_has_nanos'
  64. }
  65. ]
  66. }
  67. )
  68. puts response
  1. const response = await client.indices.create({
  2. index: "my-index-000001",
  3. mappings: {
  4. properties: {
  5. date: {
  6. type: "date_nanos",
  7. },
  8. },
  9. },
  10. });
  11. console.log(response);
  12. const response1 = await client.bulk({
  13. index: "my-index-000001",
  14. refresh: "true",
  15. operations: [
  16. {
  17. index: {
  18. _id: "1",
  19. },
  20. },
  21. {
  22. date: "2015-01-01",
  23. },
  24. {
  25. index: {
  26. _id: "2",
  27. },
  28. },
  29. {
  30. date: "2015-01-01T12:10:30.123456789Z",
  31. },
  32. {
  33. index: {
  34. _id: "3",
  35. },
  36. },
  37. {
  38. date: 1420070400000,
  39. },
  40. ],
  41. });
  42. console.log(response1);
  43. const response2 = await client.search({
  44. index: "my-index-000001",
  45. sort: {
  46. date: "asc",
  47. },
  48. runtime_mappings: {
  49. date_has_nanos: {
  50. type: "boolean",
  51. script: "emit(doc['date'].value.nano != 0)",
  52. },
  53. },
  54. fields: [
  55. {
  56. field: "date",
  57. format: "strict_date_optional_time_nanos",
  58. },
  59. {
  60. field: "date_has_nanos",
  61. },
  62. ],
  63. });
  64. console.log(response2);
  1. PUT my-index-000001
  2. {
  3. "mappings": {
  4. "properties": {
  5. "date": {
  6. "type": "date_nanos"
  7. }
  8. }
  9. }
  10. }
  11. PUT my-index-000001/_bulk?refresh
  12. { "index" : { "_id" : "1" } }
  13. { "date": "2015-01-01" }
  14. { "index" : { "_id" : "2" } }
  15. { "date": "2015-01-01T12:10:30.123456789Z" }
  16. { "index" : { "_id" : "3" } }
  17. { "date": 1420070400000 }
  18. GET my-index-000001/_search
  19. {
  20. "sort": { "date": "asc"},
  21. "runtime_mappings": {
  22. "date_has_nanos": {
  23. "type": "boolean",
  24. "script": "emit(doc['date'].value.nano != 0)"
  25. }
  26. },
  27. "fields": [
  28. {
  29. "field": "date",
  30. "format": "strict_date_optional_time_nanos"
  31. },
  32. {
  33. "field": "date_has_nanos"
  34. }
  35. ]
  36. }

The date field uses the default format.

This document uses a plain date.

This document includes a time.

This document uses milliseconds-since-the-epoch.

Note that the sort values that are returned are all in nanoseconds-since-the-epoch.

Use .nano in scripts to return the nanosecond component of the date.

You can specify the format when fetching data using the fields parameter. Use strict_date_optional_time_nanos or you’ll get a rounded result.

You can also specify multiple date formats separated by ||. The same mapping parameters than with the date field can be used.

Date nanoseconds will accept numbers with a decimal point like {"date": 1618249875.123456} but there are some cases (#70085) where we’ll lose precision on those dates so they should be avoided.

Limitations

Aggregations are still on millisecond resolution, even when using a date_nanos field. This limitation also affects transforms.


Synthetic _source

Synthetic _source is Generally Available only for TSDB indices (indices that have index.mode set to time_series). For other indices synthetic _source is in technical preview. Features in technical preview may be changed or removed in a future release. Elastic will work to fix any issues, but features in technical preview are not subject to the support SLA of official GA features.

Synthetic source may sort date_nanos field values. For example:

  1. resp = client.indices.create(
  2. index="idx",
  3. settings={
  4. "index": {
  5. "mapping": {
  6. "source": {
  7. "mode": "synthetic"
  8. }
  9. }
  10. }
  11. },
  12. mappings={
  13. "properties": {
  14. "date": {
  15. "type": "date_nanos"
  16. }
  17. }
  18. },
  19. )
  20. print(resp)
  21. resp1 = client.index(
  22. index="idx",
  23. id="1",
  24. document={
  25. "date": [
  26. "2015-01-01T12:10:30.000Z",
  27. "2014-01-01T12:10:30.000Z"
  28. ]
  29. },
  30. )
  31. print(resp1)
  1. const response = await client.indices.create({
  2. index: "idx",
  3. settings: {
  4. index: {
  5. mapping: {
  6. source: {
  7. mode: "synthetic",
  8. },
  9. },
  10. },
  11. },
  12. mappings: {
  13. properties: {
  14. date: {
  15. type: "date_nanos",
  16. },
  17. },
  18. },
  19. });
  20. console.log(response);
  21. const response1 = await client.index({
  22. index: "idx",
  23. id: 1,
  24. document: {
  25. date: ["2015-01-01T12:10:30.000Z", "2014-01-01T12:10:30.000Z"],
  26. },
  27. });
  28. console.log(response1);
  1. PUT idx
  2. {
  3. "settings": {
  4. "index": {
  5. "mapping": {
  6. "source": {
  7. "mode": "synthetic"
  8. }
  9. }
  10. }
  11. },
  12. "mappings": {
  13. "properties": {
  14. "date": { "type": "date_nanos" }
  15. }
  16. }
  17. }
  18. PUT idx/_doc/1
  19. {
  20. "date": ["2015-01-01T12:10:30.000Z", "2014-01-01T12:10:30.000Z"]
  21. }

Will become:

  1. {
  2. "date": ["2014-01-01T12:10:30.000Z", "2015-01-01T12:10:30.000Z"]
  3. }